From d86a518b7db3b57485f992cdbe9e9811d2d99a4c Mon Sep 17 00:00:00 2001 From: Abdelrahman Date: Mon, 25 May 2026 11:04:30 +0100 Subject: [PATCH] Vulkan spec --- .gitignore | 6 + 09_shader_base.slang | 43 + 09_shader_base.spv | Bin 0 -> 1592 bytes build | 6 + cglm/.gitattributes | 1 + cglm/.github/FUNDING.yml | 8 + cglm/.github/workflows/ci.yml | 645 +++++ cglm/.github/workflows/cmake-wasm.yml | 107 + cglm/.github/workflows/meson-wasm.yml | 79 + cglm/.gitignore | 83 + cglm/.readthedocs.yaml | 39 + cglm/.travis.yml | 68 + cglm/BUILDING.md | 199 ++ cglm/CMakeLists.txt | 197 ++ cglm/CONTRIBUTING.md | 53 + cglm/CREDITS | 98 + cglm/LICENSE | 21 + cglm/Makefile.am | 312 +++ cglm/Package.swift | 44 + cglm/README.md | 211 ++ cglm/_config.yml | 1 + cglm/autogen.sh | 21 + cglm/cglm.pc.in | 11 + cglm/cglm.png | Bin 0 -> 387622 bytes cglm/cglm.podspec | 37 + cglm/configure.ac | 75 + cglm/docs/make.bat | 36 + cglm/docs/requirements.txt | 4 + cglm/docs/source/aabb2d.rst | 198 ++ cglm/docs/source/affine-common.rst | 129 + cglm/docs/source/affine-mat.rst | 99 + cglm/docs/source/affine-post.rst | 127 + cglm/docs/source/affine-pre.rst | 154 ++ cglm/docs/source/affine.rst | 196 ++ cglm/docs/source/affine2d.rst | 140 + cglm/docs/source/api.rst | 28 + cglm/docs/source/api_call.rst | 11 + cglm/docs/source/api_inline_array.rst | 78 + cglm/docs/source/api_simd.rst | 12 + cglm/docs/source/api_struct.rst | 101 + cglm/docs/source/bezier.rst | 89 + cglm/docs/source/box.rst | 181 ++ cglm/docs/source/build.rst | 153 ++ cglm/docs/source/call.rst | 19 + cglm/docs/source/cam.rst | 313 +++ cglm/docs/source/cglm-intro.png | Bin 0 -> 74053 bytes cglm/docs/source/color.rst | 34 + cglm/docs/source/conf.py | 207 ++ cglm/docs/source/curve.rst | 41 + cglm/docs/source/euler.rst | 182 ++ cglm/docs/source/features.rst | 29 + cglm/docs/source/frustum.rst | 168 ++ cglm/docs/source/getting_started.rst | 105 + cglm/docs/source/index.rst | 53 + cglm/docs/source/io.rst | 147 + cglm/docs/source/ivec2.rst | 260 ++ cglm/docs/source/ivec3.rst | 272 ++ cglm/docs/source/ivec4.rst | 179 ++ cglm/docs/source/mat2.rst | 191 ++ cglm/docs/source/mat2x3.rst | 140 + cglm/docs/source/mat2x4.rst | 145 + cglm/docs/source/mat3.rst | 201 ++ cglm/docs/source/mat3x2.rst | 137 + cglm/docs/source/mat3x4.rst | 147 + cglm/docs/source/mat4.rst | 315 +++ cglm/docs/source/mat4x2.rst | 141 + cglm/docs/source/mat4x3.rst | 144 + cglm/docs/source/noise.rst | 60 + cglm/docs/source/opengl.rst | 61 + cglm/docs/source/opt.rst | 132 + cglm/docs/source/plane.rst | 33 + cglm/docs/source/project.rst | 119 + cglm/docs/source/quat.rst | 445 +++ cglm/docs/source/ray.rst | 69 + cglm/docs/source/sphere.rst | 74 + .../source/sphinx-static/theme_overrides.css | 12 + cglm/docs/source/troubleshooting.rst | 99 + cglm/docs/source/util.rst | 182 ++ cglm/docs/source/vec2-ext.rst | 161 ++ cglm/docs/source/vec2.rst | 424 +++ cglm/docs/source/vec3-ext.rst | 178 ++ cglm/docs/source/vec3.rst | 554 ++++ cglm/docs/source/vec4-ext.rst | 171 ++ cglm/docs/source/vec4.rst | 458 ++++ cglm/docs/source/version.rst | 15 + cglm/include/cglm/aabb2d.h | 270 ++ cglm/include/cglm/affine-mat.h | 189 ++ cglm/include/cglm/affine-post.h | 247 ++ cglm/include/cglm/affine-pre.h | 304 ++ cglm/include/cglm/affine.h | 238 ++ cglm/include/cglm/affine2d.h | 268 ++ cglm/include/cglm/applesimd.h | 95 + cglm/include/cglm/bezier.h | 154 ++ cglm/include/cglm/box.h | 281 ++ cglm/include/cglm/call.h | 51 + cglm/include/cglm/call/aabb2d.h | 89 + cglm/include/cglm/call/affine.h | 167 ++ cglm/include/cglm/call/affine2d.h | 67 + cglm/include/cglm/call/bezier.h | 31 + cglm/include/cglm/call/box.h | 78 + cglm/include/cglm/call/cam.h | 133 + .../include/cglm/call/clipspace/ortho_lh_no.h | 46 + .../include/cglm/call/clipspace/ortho_lh_zo.h | 46 + .../include/cglm/call/clipspace/ortho_rh_no.h | 46 + .../include/cglm/call/clipspace/ortho_rh_zo.h | 46 + .../include/cglm/call/clipspace/persp_lh_no.h | 87 + .../include/cglm/call/clipspace/persp_lh_zo.h | 87 + .../include/cglm/call/clipspace/persp_rh_no.h | 87 + .../include/cglm/call/clipspace/persp_rh_zo.h | 87 + cglm/include/cglm/call/clipspace/project_no.h | 31 + cglm/include/cglm/call/clipspace/project_zo.h | 31 + cglm/include/cglm/call/clipspace/view_lh_no.h | 31 + cglm/include/cglm/call/clipspace/view_lh_zo.h | 31 + cglm/include/cglm/call/clipspace/view_rh_no.h | 31 + cglm/include/cglm/call/clipspace/view_rh_zo.h | 31 + cglm/include/cglm/call/curve.h | 23 + cglm/include/cglm/call/ease.h | 143 + cglm/include/cglm/call/euler.h | 80 + cglm/include/cglm/call/frustum.h | 41 + cglm/include/cglm/call/io.h | 45 + cglm/include/cglm/call/ivec2.h | 179 ++ cglm/include/cglm/call/ivec3.h | 183 ++ cglm/include/cglm/call/ivec4.h | 147 + cglm/include/cglm/call/mat2.h | 83 + cglm/include/cglm/call/mat2x3.h | 47 + cglm/include/cglm/call/mat2x4.h | 47 + cglm/include/cglm/call/mat3.h | 90 + cglm/include/cglm/call/mat3x2.h | 47 + cglm/include/cglm/call/mat3x4.h | 47 + cglm/include/cglm/call/mat4.h | 131 + cglm/include/cglm/call/mat4x2.h | 47 + cglm/include/cglm/call/mat4x3.h | 47 + cglm/include/cglm/call/noise.h | 31 + cglm/include/cglm/call/plane.h | 23 + cglm/include/cglm/call/project.h | 39 + cglm/include/cglm/call/quat.h | 175 ++ cglm/include/cglm/call/ray.h | 39 + cglm/include/cglm/call/sphere.h | 39 + cglm/include/cglm/call/vec2.h | 251 ++ cglm/include/cglm/call/vec3.h | 369 +++ cglm/include/cglm/call/vec4.h | 342 +++ cglm/include/cglm/cam.h | 582 ++++ cglm/include/cglm/cglm.h | 47 + cglm/include/cglm/clipspace/ortho_lh_no.h | 183 ++ cglm/include/cglm/clipspace/ortho_lh_zo.h | 177 ++ cglm/include/cglm/clipspace/ortho_rh_no.h | 183 ++ cglm/include/cglm/clipspace/ortho_rh_zo.h | 181 ++ cglm/include/cglm/clipspace/persp.h | 48 + cglm/include/cglm/clipspace/persp_lh_no.h | 395 +++ cglm/include/cglm/clipspace/persp_lh_zo.h | 387 +++ cglm/include/cglm/clipspace/persp_rh_no.h | 395 +++ cglm/include/cglm/clipspace/persp_rh_zo.h | 389 +++ cglm/include/cglm/clipspace/project_no.h | 109 + cglm/include/cglm/clipspace/project_zo.h | 111 + cglm/include/cglm/clipspace/view_lh.h | 99 + cglm/include/cglm/clipspace/view_lh_no.h | 74 + cglm/include/cglm/clipspace/view_lh_zo.h | 74 + cglm/include/cglm/clipspace/view_rh.h | 99 + cglm/include/cglm/clipspace/view_rh_no.h | 74 + cglm/include/cglm/clipspace/view_rh_zo.h | 74 + cglm/include/cglm/color.h | 26 + cglm/include/cglm/common.h | 128 + cglm/include/cglm/curve.h | 40 + cglm/include/cglm/ease.h | 317 +++ cglm/include/cglm/euler.h | 601 ++++ cglm/include/cglm/frustum.h | 255 ++ cglm/include/cglm/handed/euler_to_quat_lh.h | 167 ++ cglm/include/cglm/handed/euler_to_quat_rh.h | 170 ++ cglm/include/cglm/io.h | 440 +++ cglm/include/cglm/ivec2.h | 659 +++++ cglm/include/cglm/ivec3.h | 713 +++++ cglm/include/cglm/ivec4.h | 608 ++++ cglm/include/cglm/mat2.h | 364 +++ cglm/include/cglm/mat2x3.h | 161 ++ cglm/include/cglm/mat2x4.h | 170 ++ cglm/include/cglm/mat3.h | 451 +++ cglm/include/cglm/mat3x2.h | 155 ++ cglm/include/cglm/mat3x4.h | 179 ++ cglm/include/cglm/mat4.h | 802 ++++++ cglm/include/cglm/mat4x2.h | 168 ++ cglm/include/cglm/mat4x3.h | 192 ++ cglm/include/cglm/noise.h | 734 +++++ cglm/include/cglm/plane.h | 44 + cglm/include/cglm/project.h | 172 ++ cglm/include/cglm/quat.h | 949 +++++++ cglm/include/cglm/ray.h | 174 ++ cglm/include/cglm/simd/arm.h | 204 ++ cglm/include/cglm/simd/avx/affine.h | 66 + cglm/include/cglm/simd/avx/mat4.h | 115 + cglm/include/cglm/simd/intrin.h | 153 ++ cglm/include/cglm/simd/neon/affine.h | 121 + cglm/include/cglm/simd/neon/mat2.h | 44 + cglm/include/cglm/simd/neon/mat4.h | 468 ++++ cglm/include/cglm/simd/neon/quat.h | 57 + cglm/include/cglm/simd/sse2/affine.h | 115 + cglm/include/cglm/simd/sse2/mat2.h | 48 + cglm/include/cglm/simd/sse2/mat3.h | 76 + cglm/include/cglm/simd/sse2/mat4.h | 573 ++++ cglm/include/cglm/simd/sse2/quat.h | 54 + cglm/include/cglm/simd/wasm.h | 197 ++ cglm/include/cglm/simd/wasm/affine.h | 127 + cglm/include/cglm/simd/wasm/mat2.h | 50 + cglm/include/cglm/simd/wasm/mat3.h | 85 + cglm/include/cglm/simd/wasm/mat4.h | 454 +++ cglm/include/cglm/simd/wasm/quat.h | 55 + cglm/include/cglm/simd/x86.h | 365 +++ cglm/include/cglm/sphere.h | 99 + cglm/include/cglm/struct.h | 50 + cglm/include/cglm/struct/aabb2d.h | 253 ++ cglm/include/cglm/struct/affine-mat.h | 90 + cglm/include/cglm/struct/affine-post.h | 184 ++ cglm/include/cglm/struct/affine-pre.h | 184 ++ cglm/include/cglm/struct/affine.h | 201 ++ cglm/include/cglm/struct/affine2d.h | 177 ++ cglm/include/cglm/struct/box.h | 259 ++ cglm/include/cglm/struct/cam.h | 646 +++++ .../cglm/struct/clipspace/ortho_lh_no.h | 154 ++ .../cglm/struct/clipspace/ortho_lh_zo.h | 154 ++ .../cglm/struct/clipspace/ortho_rh_no.h | 154 ++ .../cglm/struct/clipspace/ortho_rh_zo.h | 154 ++ .../cglm/struct/clipspace/persp_lh_no.h | 312 +++ .../cglm/struct/clipspace/persp_lh_zo.h | 312 +++ .../cglm/struct/clipspace/persp_rh_no.h | 312 +++ .../cglm/struct/clipspace/persp_rh_zo.h | 312 +++ .../cglm/struct/clipspace/project_no.h | 98 + .../cglm/struct/clipspace/project_zo.h | 98 + .../cglm/struct/clipspace/view_lh_no.h | 89 + .../cglm/struct/clipspace/view_lh_zo.h | 89 + .../cglm/struct/clipspace/view_rh_no.h | 89 + .../cglm/struct/clipspace/view_rh_zo.h | 89 + cglm/include/cglm/struct/color.h | 27 + cglm/include/cglm/struct/curve.h | 40 + cglm/include/cglm/struct/euler.h | 249 ++ cglm/include/cglm/struct/frustum.h | 155 ++ .../cglm/struct/handed/euler_to_quat_lh.h | 115 + .../cglm/struct/handed/euler_to_quat_rh.h | 115 + cglm/include/cglm/struct/io.h | 107 + cglm/include/cglm/struct/ivec2.h | 708 +++++ cglm/include/cglm/struct/ivec3.h | 725 +++++ cglm/include/cglm/struct/ivec4.h | 588 ++++ cglm/include/cglm/struct/mat2.h | 276 ++ cglm/include/cglm/struct/mat2x3.h | 127 + cglm/include/cglm/struct/mat2x4.h | 127 + cglm/include/cglm/struct/mat3.h | 303 ++ cglm/include/cglm/struct/mat3x2.h | 127 + cglm/include/cglm/struct/mat3x4.h | 127 + cglm/include/cglm/struct/mat4.h | 477 ++++ cglm/include/cglm/struct/mat4x2.h | 128 + cglm/include/cglm/struct/mat4x3.h | 127 + cglm/include/cglm/struct/noise.h | 57 + cglm/include/cglm/struct/plane.h | 40 + cglm/include/cglm/struct/project.h | 162 ++ cglm/include/cglm/struct/quat.h | 601 ++++ cglm/include/cglm/struct/ray.h | 86 + cglm/include/cglm/struct/sphere.h | 93 + cglm/include/cglm/struct/vec2-ext.h | 337 +++ cglm/include/cglm/struct/vec2.h | 747 +++++ cglm/include/cglm/struct/vec3-ext.h | 325 +++ cglm/include/cglm/struct/vec3.h | 1132 ++++++++ cglm/include/cglm/struct/vec4-ext.h | 325 +++ cglm/include/cglm/struct/vec4.h | 961 +++++++ cglm/include/cglm/types-struct.h | 303 ++ cglm/include/cglm/types.h | 132 + cglm/include/cglm/util.h | 375 +++ cglm/include/cglm/vec2-ext.h | 337 +++ cglm/include/cglm/vec2.h | 798 ++++++ cglm/include/cglm/vec3-ext.h | 345 +++ cglm/include/cglm/vec3.h | 1264 +++++++++ cglm/include/cglm/vec4-ext.h | 400 +++ cglm/include/cglm/vec4.h | 1348 +++++++++ cglm/include/cglm/version.h | 15 + cglm/include/module.modulemap | 14 + cglm/meson.build | 142 + cglm/meson_options.txt | 2 + cglm/src/aabb2d.c | 108 + cglm/src/affine.c | 225 ++ cglm/src/affine2d.c | 81 + cglm/src/bezier.c | 27 + cglm/src/box.c | 96 + cglm/src/cam.c | 171 ++ cglm/src/clipspace/ortho_lh_no.c | 51 + cglm/src/clipspace/ortho_lh_zo.c | 51 + cglm/src/clipspace/ortho_rh_no.c | 51 + cglm/src/clipspace/ortho_rh_zo.c | 51 + cglm/src/clipspace/persp_lh_no.c | 110 + cglm/src/clipspace/persp_lh_zo.c | 110 + cglm/src/clipspace/persp_rh_no.c | 110 + cglm/src/clipspace/persp_rh_zo.c | 110 + cglm/src/clipspace/project_no.c | 27 + cglm/src/clipspace/project_zo.c | 27 + cglm/src/clipspace/view_lh_no.c | 27 + cglm/src/clipspace/view_lh_zo.c | 27 + cglm/src/clipspace/view_rh_no.c | 27 + cglm/src/clipspace/view_rh_zo.c | 27 + cglm/src/config.h | 22 + cglm/src/curve.c | 15 + cglm/src/ease.c | 195 ++ cglm/src/euler.c | 99 + cglm/src/frustum.c | 42 + cglm/src/io.c | 46 + cglm/src/ivec2.c | 249 ++ cglm/src/ivec3.c | 255 ++ cglm/src/ivec4.c | 201 ++ cglm/src/mat2.c | 105 + cglm/src/mat2x3.c | 51 + cglm/src/mat2x4.c | 51 + cglm/src/mat3.c | 111 + cglm/src/mat3x2.c | 51 + cglm/src/mat3x4.c | 51 + cglm/src/mat4.c | 171 ++ cglm/src/mat4x2.c | 51 + cglm/src/mat4x3.c | 51 + cglm/src/noise.c | 27 + cglm/src/plane.c | 15 + cglm/src/project.c | 39 + cglm/src/quat.c | 243 ++ cglm/src/ray.c | 29 + cglm/src/sphere.c | 39 + cglm/src/swift/empty.c | 1 + cglm/src/vec2.c | 358 +++ cglm/src/vec3.c | 503 ++++ cglm/src/vec4.c | 461 ++++ cglm/test/CMakeLists.txt | 48 + cglm/test/include/common.h | 159 ++ cglm/test/runner.c | 100 + cglm/test/src/test_aabb2d.h | 28 + cglm/test/src/test_affine.h | 634 +++++ cglm/test/src/test_affine2d.h | 310 +++ cglm/test/src/test_affine_mat.h | 112 + cglm/test/src/test_bezier.c | 66 + cglm/test/src/test_cam.h | 115 + cglm/test/src/test_cam_lh_no.h | 38 + cglm/test/src/test_cam_lh_zo.h | 38 + cglm/test/src/test_cam_rh_no.h | 38 + cglm/test/src/test_cam_rh_zo.h | 38 + cglm/test/src/test_clamp.c | 31 + cglm/test/src/test_common.c | 581 ++++ cglm/test/src/test_common.h | 176 ++ cglm/test/src/test_euler.c | 45 + cglm/test/src/test_euler_to_quat_lh.h | 493 ++++ cglm/test/src/test_euler_to_quat_rh.h | 573 ++++ cglm/test/src/test_ivec2.h | 536 ++++ cglm/test/src/test_ivec3.h | 582 ++++ cglm/test/src/test_ivec4.h | 488 ++++ cglm/test/src/test_mat2.h | 302 ++ cglm/test/src/test_mat2x3.h | 156 ++ cglm/test/src/test_mat2x4.h | 159 ++ cglm/test/src/test_mat3.h | 329 +++ cglm/test/src/test_mat3x2.h | 155 ++ cglm/test/src/test_mat3x4.h | 161 ++ cglm/test/src/test_mat4.h | 506 ++++ cglm/test/src/test_mat4x2.h | 158 ++ cglm/test/src/test_mat4x3.h | 162 ++ cglm/test/src/test_noise.h | 118 + cglm/test/src/test_plane.h | 39 + cglm/test/src/test_project.h | 112 + cglm/test/src/test_quat.h | 1108 ++++++++ cglm/test/src/test_ray.h | 73 + cglm/test/src/test_struct.c | 150 + cglm/test/src/test_vec2.h | 951 +++++++ cglm/test/src/test_vec3.h | 2032 ++++++++++++++ cglm/test/src/test_vec4.h | 1705 ++++++++++++ cglm/test/src/tests.c | 91 + cglm/test/tests.h | 2437 +++++++++++++++++ cglm/win/.gitignore | 9 + cglm/win/build.bat | 1 + cglm/win/cglm-test.vcxproj | 432 +++ cglm/win/cglm-test.vcxproj.filters | 140 + cglm/win/cglm.sln | 71 + cglm/win/cglm.vcxproj | 633 +++++ cglm/win/cglm.vcxproj.filters | 748 +++++ main.c | 1310 +++++++++ mat.c | 18 + mat.cc | 19 + wapp | 1 + 375 files changed, 77089 insertions(+) create mode 100644 .gitignore create mode 100644 09_shader_base.slang create mode 100644 09_shader_base.spv create mode 100755 build create mode 100644 cglm/.gitattributes create mode 100644 cglm/.github/FUNDING.yml create mode 100644 cglm/.github/workflows/ci.yml create mode 100644 cglm/.github/workflows/cmake-wasm.yml create mode 100644 cglm/.github/workflows/meson-wasm.yml create mode 100644 cglm/.gitignore create mode 100644 cglm/.readthedocs.yaml create mode 100644 cglm/.travis.yml create mode 100644 cglm/BUILDING.md create mode 100644 cglm/CMakeLists.txt create mode 100644 cglm/CONTRIBUTING.md create mode 100644 cglm/CREDITS create mode 100644 cglm/LICENSE create mode 100644 cglm/Makefile.am create mode 100644 cglm/Package.swift create mode 100644 cglm/README.md create mode 100644 cglm/_config.yml create mode 100755 cglm/autogen.sh create mode 100644 cglm/cglm.pc.in create mode 100644 cglm/cglm.png create mode 100644 cglm/cglm.podspec create mode 100644 cglm/configure.ac create mode 100755 cglm/docs/make.bat create mode 100644 cglm/docs/requirements.txt create mode 100644 cglm/docs/source/aabb2d.rst create mode 100644 cglm/docs/source/affine-common.rst create mode 100644 cglm/docs/source/affine-mat.rst create mode 100644 cglm/docs/source/affine-post.rst create mode 100644 cglm/docs/source/affine-pre.rst create mode 100644 cglm/docs/source/affine.rst create mode 100644 cglm/docs/source/affine2d.rst create mode 100644 cglm/docs/source/api.rst create mode 100644 cglm/docs/source/api_call.rst create mode 100644 cglm/docs/source/api_inline_array.rst create mode 100644 cglm/docs/source/api_simd.rst create mode 100644 cglm/docs/source/api_struct.rst create mode 100644 cglm/docs/source/bezier.rst create mode 100644 cglm/docs/source/box.rst create mode 100644 cglm/docs/source/build.rst create mode 100644 cglm/docs/source/call.rst create mode 100644 cglm/docs/source/cam.rst create mode 100644 cglm/docs/source/cglm-intro.png create mode 100644 cglm/docs/source/color.rst create mode 100644 cglm/docs/source/conf.py create mode 100644 cglm/docs/source/curve.rst create mode 100644 cglm/docs/source/euler.rst create mode 100644 cglm/docs/source/features.rst create mode 100644 cglm/docs/source/frustum.rst create mode 100644 cglm/docs/source/getting_started.rst create mode 100644 cglm/docs/source/index.rst create mode 100644 cglm/docs/source/io.rst create mode 100644 cglm/docs/source/ivec2.rst create mode 100644 cglm/docs/source/ivec3.rst create mode 100644 cglm/docs/source/ivec4.rst create mode 100644 cglm/docs/source/mat2.rst create mode 100644 cglm/docs/source/mat2x3.rst create mode 100644 cglm/docs/source/mat2x4.rst create mode 100644 cglm/docs/source/mat3.rst create mode 100644 cglm/docs/source/mat3x2.rst create mode 100644 cglm/docs/source/mat3x4.rst create mode 100644 cglm/docs/source/mat4.rst create mode 100644 cglm/docs/source/mat4x2.rst create mode 100644 cglm/docs/source/mat4x3.rst create mode 100644 cglm/docs/source/noise.rst create mode 100644 cglm/docs/source/opengl.rst create mode 100644 cglm/docs/source/opt.rst create mode 100644 cglm/docs/source/plane.rst create mode 100644 cglm/docs/source/project.rst create mode 100644 cglm/docs/source/quat.rst create mode 100644 cglm/docs/source/ray.rst create mode 100644 cglm/docs/source/sphere.rst create mode 100644 cglm/docs/source/sphinx-static/theme_overrides.css create mode 100644 cglm/docs/source/troubleshooting.rst create mode 100644 cglm/docs/source/util.rst create mode 100644 cglm/docs/source/vec2-ext.rst create mode 100644 cglm/docs/source/vec2.rst create mode 100644 cglm/docs/source/vec3-ext.rst create mode 100644 cglm/docs/source/vec3.rst create mode 100644 cglm/docs/source/vec4-ext.rst create mode 100644 cglm/docs/source/vec4.rst create mode 100644 cglm/docs/source/version.rst create mode 100644 cglm/include/cglm/aabb2d.h create mode 100644 cglm/include/cglm/affine-mat.h create mode 100644 cglm/include/cglm/affine-post.h create mode 100644 cglm/include/cglm/affine-pre.h create mode 100644 cglm/include/cglm/affine.h create mode 100644 cglm/include/cglm/affine2d.h create mode 100644 cglm/include/cglm/applesimd.h create mode 100644 cglm/include/cglm/bezier.h create mode 100644 cglm/include/cglm/box.h create mode 100644 cglm/include/cglm/call.h create mode 100644 cglm/include/cglm/call/aabb2d.h create mode 100644 cglm/include/cglm/call/affine.h create mode 100644 cglm/include/cglm/call/affine2d.h create mode 100644 cglm/include/cglm/call/bezier.h create mode 100644 cglm/include/cglm/call/box.h create mode 100644 cglm/include/cglm/call/cam.h create mode 100644 cglm/include/cglm/call/clipspace/ortho_lh_no.h create mode 100644 cglm/include/cglm/call/clipspace/ortho_lh_zo.h create mode 100644 cglm/include/cglm/call/clipspace/ortho_rh_no.h create mode 100644 cglm/include/cglm/call/clipspace/ortho_rh_zo.h create mode 100644 cglm/include/cglm/call/clipspace/persp_lh_no.h create mode 100644 cglm/include/cglm/call/clipspace/persp_lh_zo.h create mode 100644 cglm/include/cglm/call/clipspace/persp_rh_no.h create mode 100644 cglm/include/cglm/call/clipspace/persp_rh_zo.h create mode 100644 cglm/include/cglm/call/clipspace/project_no.h create mode 100644 cglm/include/cglm/call/clipspace/project_zo.h create mode 100644 cglm/include/cglm/call/clipspace/view_lh_no.h create mode 100644 cglm/include/cglm/call/clipspace/view_lh_zo.h create mode 100644 cglm/include/cglm/call/clipspace/view_rh_no.h create mode 100644 cglm/include/cglm/call/clipspace/view_rh_zo.h create mode 100644 cglm/include/cglm/call/curve.h create mode 100644 cglm/include/cglm/call/ease.h create mode 100644 cglm/include/cglm/call/euler.h create mode 100644 cglm/include/cglm/call/frustum.h create mode 100644 cglm/include/cglm/call/io.h create mode 100644 cglm/include/cglm/call/ivec2.h create mode 100644 cglm/include/cglm/call/ivec3.h create mode 100644 cglm/include/cglm/call/ivec4.h create mode 100644 cglm/include/cglm/call/mat2.h create mode 100644 cglm/include/cglm/call/mat2x3.h create mode 100644 cglm/include/cglm/call/mat2x4.h create mode 100644 cglm/include/cglm/call/mat3.h create mode 100644 cglm/include/cglm/call/mat3x2.h create mode 100644 cglm/include/cglm/call/mat3x4.h create mode 100644 cglm/include/cglm/call/mat4.h create mode 100644 cglm/include/cglm/call/mat4x2.h create mode 100644 cglm/include/cglm/call/mat4x3.h create mode 100644 cglm/include/cglm/call/noise.h create mode 100644 cglm/include/cglm/call/plane.h create mode 100644 cglm/include/cglm/call/project.h create mode 100644 cglm/include/cglm/call/quat.h create mode 100644 cglm/include/cglm/call/ray.h create mode 100644 cglm/include/cglm/call/sphere.h create mode 100644 cglm/include/cglm/call/vec2.h create mode 100644 cglm/include/cglm/call/vec3.h create mode 100644 cglm/include/cglm/call/vec4.h create mode 100644 cglm/include/cglm/cam.h create mode 100644 cglm/include/cglm/cglm.h create mode 100644 cglm/include/cglm/clipspace/ortho_lh_no.h create mode 100644 cglm/include/cglm/clipspace/ortho_lh_zo.h create mode 100644 cglm/include/cglm/clipspace/ortho_rh_no.h create mode 100644 cglm/include/cglm/clipspace/ortho_rh_zo.h create mode 100644 cglm/include/cglm/clipspace/persp.h create mode 100644 cglm/include/cglm/clipspace/persp_lh_no.h create mode 100644 cglm/include/cglm/clipspace/persp_lh_zo.h create mode 100644 cglm/include/cglm/clipspace/persp_rh_no.h create mode 100644 cglm/include/cglm/clipspace/persp_rh_zo.h create mode 100644 cglm/include/cglm/clipspace/project_no.h create mode 100644 cglm/include/cglm/clipspace/project_zo.h create mode 100644 cglm/include/cglm/clipspace/view_lh.h create mode 100644 cglm/include/cglm/clipspace/view_lh_no.h create mode 100644 cglm/include/cglm/clipspace/view_lh_zo.h create mode 100644 cglm/include/cglm/clipspace/view_rh.h create mode 100644 cglm/include/cglm/clipspace/view_rh_no.h create mode 100644 cglm/include/cglm/clipspace/view_rh_zo.h create mode 100644 cglm/include/cglm/color.h create mode 100644 cglm/include/cglm/common.h create mode 100644 cglm/include/cglm/curve.h create mode 100644 cglm/include/cglm/ease.h create mode 100644 cglm/include/cglm/euler.h create mode 100644 cglm/include/cglm/frustum.h create mode 100644 cglm/include/cglm/handed/euler_to_quat_lh.h create mode 100644 cglm/include/cglm/handed/euler_to_quat_rh.h create mode 100644 cglm/include/cglm/io.h create mode 100644 cglm/include/cglm/ivec2.h create mode 100644 cglm/include/cglm/ivec3.h create mode 100644 cglm/include/cglm/ivec4.h create mode 100644 cglm/include/cglm/mat2.h create mode 100644 cglm/include/cglm/mat2x3.h create mode 100644 cglm/include/cglm/mat2x4.h create mode 100644 cglm/include/cglm/mat3.h create mode 100644 cglm/include/cglm/mat3x2.h create mode 100644 cglm/include/cglm/mat3x4.h create mode 100644 cglm/include/cglm/mat4.h create mode 100644 cglm/include/cglm/mat4x2.h create mode 100644 cglm/include/cglm/mat4x3.h create mode 100644 cglm/include/cglm/noise.h create mode 100644 cglm/include/cglm/plane.h create mode 100644 cglm/include/cglm/project.h create mode 100644 cglm/include/cglm/quat.h create mode 100644 cglm/include/cglm/ray.h create mode 100644 cglm/include/cglm/simd/arm.h create mode 100644 cglm/include/cglm/simd/avx/affine.h create mode 100644 cglm/include/cglm/simd/avx/mat4.h create mode 100644 cglm/include/cglm/simd/intrin.h create mode 100644 cglm/include/cglm/simd/neon/affine.h create mode 100644 cglm/include/cglm/simd/neon/mat2.h create mode 100644 cglm/include/cglm/simd/neon/mat4.h create mode 100644 cglm/include/cglm/simd/neon/quat.h create mode 100644 cglm/include/cglm/simd/sse2/affine.h create mode 100644 cglm/include/cglm/simd/sse2/mat2.h create mode 100644 cglm/include/cglm/simd/sse2/mat3.h create mode 100644 cglm/include/cglm/simd/sse2/mat4.h create mode 100644 cglm/include/cglm/simd/sse2/quat.h create mode 100644 cglm/include/cglm/simd/wasm.h create mode 100644 cglm/include/cglm/simd/wasm/affine.h create mode 100644 cglm/include/cglm/simd/wasm/mat2.h create mode 100644 cglm/include/cglm/simd/wasm/mat3.h create mode 100644 cglm/include/cglm/simd/wasm/mat4.h create mode 100644 cglm/include/cglm/simd/wasm/quat.h create mode 100644 cglm/include/cglm/simd/x86.h create mode 100644 cglm/include/cglm/sphere.h create mode 100644 cglm/include/cglm/struct.h create mode 100644 cglm/include/cglm/struct/aabb2d.h create mode 100644 cglm/include/cglm/struct/affine-mat.h create mode 100644 cglm/include/cglm/struct/affine-post.h create mode 100644 cglm/include/cglm/struct/affine-pre.h create mode 100644 cglm/include/cglm/struct/affine.h create mode 100644 cglm/include/cglm/struct/affine2d.h create mode 100644 cglm/include/cglm/struct/box.h create mode 100644 cglm/include/cglm/struct/cam.h create mode 100644 cglm/include/cglm/struct/clipspace/ortho_lh_no.h create mode 100644 cglm/include/cglm/struct/clipspace/ortho_lh_zo.h create mode 100644 cglm/include/cglm/struct/clipspace/ortho_rh_no.h create mode 100644 cglm/include/cglm/struct/clipspace/ortho_rh_zo.h create mode 100644 cglm/include/cglm/struct/clipspace/persp_lh_no.h create mode 100644 cglm/include/cglm/struct/clipspace/persp_lh_zo.h create mode 100644 cglm/include/cglm/struct/clipspace/persp_rh_no.h create mode 100644 cglm/include/cglm/struct/clipspace/persp_rh_zo.h create mode 100644 cglm/include/cglm/struct/clipspace/project_no.h create mode 100644 cglm/include/cglm/struct/clipspace/project_zo.h create mode 100644 cglm/include/cglm/struct/clipspace/view_lh_no.h create mode 100644 cglm/include/cglm/struct/clipspace/view_lh_zo.h create mode 100644 cglm/include/cglm/struct/clipspace/view_rh_no.h create mode 100644 cglm/include/cglm/struct/clipspace/view_rh_zo.h create mode 100644 cglm/include/cglm/struct/color.h create mode 100644 cglm/include/cglm/struct/curve.h create mode 100644 cglm/include/cglm/struct/euler.h create mode 100644 cglm/include/cglm/struct/frustum.h create mode 100644 cglm/include/cglm/struct/handed/euler_to_quat_lh.h create mode 100644 cglm/include/cglm/struct/handed/euler_to_quat_rh.h create mode 100644 cglm/include/cglm/struct/io.h create mode 100644 cglm/include/cglm/struct/ivec2.h create mode 100644 cglm/include/cglm/struct/ivec3.h create mode 100644 cglm/include/cglm/struct/ivec4.h create mode 100644 cglm/include/cglm/struct/mat2.h create mode 100644 cglm/include/cglm/struct/mat2x3.h create mode 100644 cglm/include/cglm/struct/mat2x4.h create mode 100644 cglm/include/cglm/struct/mat3.h create mode 100644 cglm/include/cglm/struct/mat3x2.h create mode 100644 cglm/include/cglm/struct/mat3x4.h create mode 100644 cglm/include/cglm/struct/mat4.h create mode 100644 cglm/include/cglm/struct/mat4x2.h create mode 100644 cglm/include/cglm/struct/mat4x3.h create mode 100644 cglm/include/cglm/struct/noise.h create mode 100644 cglm/include/cglm/struct/plane.h create mode 100644 cglm/include/cglm/struct/project.h create mode 100644 cglm/include/cglm/struct/quat.h create mode 100644 cglm/include/cglm/struct/ray.h create mode 100644 cglm/include/cglm/struct/sphere.h create mode 100644 cglm/include/cglm/struct/vec2-ext.h create mode 100644 cglm/include/cglm/struct/vec2.h create mode 100644 cglm/include/cglm/struct/vec3-ext.h create mode 100644 cglm/include/cglm/struct/vec3.h create mode 100644 cglm/include/cglm/struct/vec4-ext.h create mode 100644 cglm/include/cglm/struct/vec4.h create mode 100644 cglm/include/cglm/types-struct.h create mode 100644 cglm/include/cglm/types.h create mode 100644 cglm/include/cglm/util.h create mode 100644 cglm/include/cglm/vec2-ext.h create mode 100644 cglm/include/cglm/vec2.h create mode 100644 cglm/include/cglm/vec3-ext.h create mode 100644 cglm/include/cglm/vec3.h create mode 100644 cglm/include/cglm/vec4-ext.h create mode 100644 cglm/include/cglm/vec4.h create mode 100644 cglm/include/cglm/version.h create mode 100644 cglm/include/module.modulemap create mode 100644 cglm/meson.build create mode 100644 cglm/meson_options.txt create mode 100644 cglm/src/aabb2d.c create mode 100644 cglm/src/affine.c create mode 100644 cglm/src/affine2d.c create mode 100644 cglm/src/bezier.c create mode 100644 cglm/src/box.c create mode 100644 cglm/src/cam.c create mode 100644 cglm/src/clipspace/ortho_lh_no.c create mode 100644 cglm/src/clipspace/ortho_lh_zo.c create mode 100644 cglm/src/clipspace/ortho_rh_no.c create mode 100644 cglm/src/clipspace/ortho_rh_zo.c create mode 100644 cglm/src/clipspace/persp_lh_no.c create mode 100644 cglm/src/clipspace/persp_lh_zo.c create mode 100644 cglm/src/clipspace/persp_rh_no.c create mode 100644 cglm/src/clipspace/persp_rh_zo.c create mode 100644 cglm/src/clipspace/project_no.c create mode 100644 cglm/src/clipspace/project_zo.c create mode 100644 cglm/src/clipspace/view_lh_no.c create mode 100644 cglm/src/clipspace/view_lh_zo.c create mode 100644 cglm/src/clipspace/view_rh_no.c create mode 100644 cglm/src/clipspace/view_rh_zo.c create mode 100644 cglm/src/config.h create mode 100644 cglm/src/curve.c create mode 100644 cglm/src/ease.c create mode 100644 cglm/src/euler.c create mode 100644 cglm/src/frustum.c create mode 100644 cglm/src/io.c create mode 100644 cglm/src/ivec2.c create mode 100644 cglm/src/ivec3.c create mode 100644 cglm/src/ivec4.c create mode 100644 cglm/src/mat2.c create mode 100644 cglm/src/mat2x3.c create mode 100644 cglm/src/mat2x4.c create mode 100644 cglm/src/mat3.c create mode 100644 cglm/src/mat3x2.c create mode 100644 cglm/src/mat3x4.c create mode 100644 cglm/src/mat4.c create mode 100644 cglm/src/mat4x2.c create mode 100644 cglm/src/mat4x3.c create mode 100644 cglm/src/noise.c create mode 100644 cglm/src/plane.c create mode 100644 cglm/src/project.c create mode 100644 cglm/src/quat.c create mode 100644 cglm/src/ray.c create mode 100644 cglm/src/sphere.c create mode 100644 cglm/src/swift/empty.c create mode 100644 cglm/src/vec2.c create mode 100644 cglm/src/vec3.c create mode 100644 cglm/src/vec4.c create mode 100644 cglm/test/CMakeLists.txt create mode 100644 cglm/test/include/common.h create mode 100644 cglm/test/runner.c create mode 100644 cglm/test/src/test_aabb2d.h create mode 100644 cglm/test/src/test_affine.h create mode 100644 cglm/test/src/test_affine2d.h create mode 100644 cglm/test/src/test_affine_mat.h create mode 100644 cglm/test/src/test_bezier.c create mode 100644 cglm/test/src/test_cam.h create mode 100644 cglm/test/src/test_cam_lh_no.h create mode 100644 cglm/test/src/test_cam_lh_zo.h create mode 100644 cglm/test/src/test_cam_rh_no.h create mode 100644 cglm/test/src/test_cam_rh_zo.h create mode 100644 cglm/test/src/test_clamp.c create mode 100644 cglm/test/src/test_common.c create mode 100644 cglm/test/src/test_common.h create mode 100644 cglm/test/src/test_euler.c create mode 100644 cglm/test/src/test_euler_to_quat_lh.h create mode 100644 cglm/test/src/test_euler_to_quat_rh.h create mode 100644 cglm/test/src/test_ivec2.h create mode 100644 cglm/test/src/test_ivec3.h create mode 100644 cglm/test/src/test_ivec4.h create mode 100644 cglm/test/src/test_mat2.h create mode 100644 cglm/test/src/test_mat2x3.h create mode 100644 cglm/test/src/test_mat2x4.h create mode 100644 cglm/test/src/test_mat3.h create mode 100644 cglm/test/src/test_mat3x2.h create mode 100644 cglm/test/src/test_mat3x4.h create mode 100644 cglm/test/src/test_mat4.h create mode 100644 cglm/test/src/test_mat4x2.h create mode 100644 cglm/test/src/test_mat4x3.h create mode 100644 cglm/test/src/test_noise.h create mode 100644 cglm/test/src/test_plane.h create mode 100644 cglm/test/src/test_project.h create mode 100644 cglm/test/src/test_quat.h create mode 100644 cglm/test/src/test_ray.h create mode 100644 cglm/test/src/test_struct.c create mode 100644 cglm/test/src/test_vec2.h create mode 100644 cglm/test/src/test_vec3.h create mode 100644 cglm/test/src/test_vec4.h create mode 100644 cglm/test/src/tests.c create mode 100644 cglm/test/tests.h create mode 100644 cglm/win/.gitignore create mode 100755 cglm/win/build.bat create mode 100644 cglm/win/cglm-test.vcxproj create mode 100644 cglm/win/cglm-test.vcxproj.filters create mode 100644 cglm/win/cglm.sln create mode 100644 cglm/win/cglm.vcxproj create mode 100644 cglm/win/cglm.vcxproj.filters create mode 100644 main.c create mode 100644 mat.c create mode 100644 mat.cc create mode 160000 wapp diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..e3ce75b --- /dev/null +++ b/.gitignore @@ -0,0 +1,6 @@ +.cache +ccmat +cmat +main +compile_commands.json +rgb_u8_to_f32.py diff --git a/09_shader_base.slang b/09_shader_base.slang new file mode 100644 index 0000000..0263cf7 --- /dev/null +++ b/09_shader_base.slang @@ -0,0 +1,43 @@ +struct Vertex { + float3 position; + float3 color; +}; + +static float4x4 model = { + { 1.00000, 0.00000, 0.00000, 0.00000}, + { 0.00000, 1.00000, 0.00000, 0.00000}, + { 0.00000, 0.00000, 1.00000, 0.00000}, + { 0.00000, 0.00000, 0.00000, 1.00000}, +}; + +static float4x4 view = { +{ 0.80000, 0.00000, -0.60000, -0.00000}, +{ 0.22283, 0.92848, 0.29711, -0.00000}, +{ 0.55709, -0.37139, 0.74278, -5.38516}, +{ 0.00000, 0.00000, 0.00000, 1.00000}, +}; + +static float4x4 proj = { +{ 1.81066, 0.00000, 0.00000, 0.00000}, +{ 0.00000, 2.41421, 0.00000, 0.00000}, +{ 0.00000, 0.00000, -1.02020, -0.20202}, +{ 0.00000, 0.00000, -1.00000, 0.00000}, +}; + +struct VertexOutput { + float3 color; + float4 sv_position : SV_Position; +}; + +[shader("vertex")] +VertexOutput vertMain(Vertex vertex) { + VertexOutput output; + output.color = vertex.color; + output.sv_position = mul(proj, mul(view, mul(model, float4(vertex.position, 1.0)))); + return output; +} + +[shader("fragment")] +float4 fragMain(VertexOutput inVert) : SV_Target { + return float4(inVert.color, 1.0); +} diff --git a/09_shader_base.spv b/09_shader_base.spv new file mode 100644 index 0000000000000000000000000000000000000000..9051e85a8be9319d2f27feffca0fa46dcb5df4bc GIT binary patch literal 1592 zcmZ9M%}*0i6vdyd1;m68QN9)gMG@sg-59i5TqrICY=Rg`G^UYKCrWEdV{}JYyKrNo zCjJL5+_)_!Ze{6~t%-ku8xzlOoYPGDnmcpPz2}~H@0*!sGZUR5cPNA~6`tzK>JOPf zq$gx!jNco&V?U$o%W}Q3T#Bj$)@19lCE22^H*|%=x<8a{)=L%hD#B%1pESvJ%Ch<$ z)gAp@=!omlF29<4QhQ#jD;&qsPU0_W+fgH`RTb}!_byi(_1DE(RBaSX_0ra3_hyBh znC(5Y7rjK)``XP+_Bs;cKHD_YbD-Bg+igwE>|-)!0n;;>1?HU6m@{K`7J2$f=jp?E zAbB1q;}iS8$DsTUamHmss^>hxJTY)rf_WCh!t6y1JgqC-W3sQ`e=Qgf2(xCS2De`Z zhq%q$6N)|C9lc@h@q|OXojau1?{~X7@fB(l)Yt}QuA3uz=dYTm0+VMUU0@R=nm%O#6Wm;KKQcL z(mpS|KlZ+0t+T?^z$2H|ni1aq)YGb+75+KdY+8FR#)Ve?f^dHI^EYcx3!_0DT=K5H zBwr&fn0rykXUCm9InM5sjG5^bO}MVRA|GzL*14-mJe_wgpQ>@*xofFhdoT0Ku_xMS zz#Z2+drNn|FWz7ZVjUObKcQ>E#7v)o9%y(%_J>Ml74Xs>~DNWKKmC{OOMMkVw}y}B({>oIRnS;$|ueVV!78x^4;5pd}eiToAQa#8=A~b%$olIFIa{Q literal 0 HcmV?d00001 diff --git a/build b/build new file mode 100755 index 0000000..398f8a5 --- /dev/null +++ b/build @@ -0,0 +1,6 @@ +#!/bin/bash + +slangc 09_shader_base.slang -entry vertMain -entry fragMain -o 09_shader_base.spv +bear -- gcc -g -DNDEBUG -DVK_NO_PROTOTYPES -I$VULKAN_SDK/include -Iwapp/dist/include/ -Icglm/include -Lwapp/dist/lib/ $VULKAN_SDK/include/volk/volk.c main.c -lSDL3 -lwapp -lm -o main +clang -Icglm/include mat.c -lm -o cmat +clang++ mat.cc -lglm -o ccmat diff --git a/cglm/.gitattributes b/cglm/.gitattributes new file mode 100644 index 0000000..15a5c58 --- /dev/null +++ b/cglm/.gitattributes @@ -0,0 +1 @@ +*.h linguist-language=C diff --git a/cglm/.github/FUNDING.yml b/cglm/.github/FUNDING.yml new file mode 100644 index 0000000..2138628 --- /dev/null +++ b/cglm/.github/FUNDING.yml @@ -0,0 +1,8 @@ +# These are supported funding model platforms + +github: [recp] +patreon: recp +open_collective: cglm +ko_fi: # Replace with a single Ko-fi username +tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel +custom: # Replace with a single custom sponsorship URL diff --git a/cglm/.github/workflows/ci.yml b/cglm/.github/workflows/ci.yml new file mode 100644 index 0000000..f4a0f6b --- /dev/null +++ b/cglm/.github/workflows/ci.yml @@ -0,0 +1,645 @@ +name: CI + +on: + push: + branches: [ "master" ] + pull_request: + branches: [ "master" ] + +jobs: + build_autotools: + name: Autotools / ${{ matrix.os }} / ${{ matrix.simd }} + runs-on: ${{ matrix.os }} + strategy: + fail-fast: false + matrix: + include: + # x86/x64 builds + - { os: macos-13, simd: none } + - { os: macos-13, simd: sse } + - { os: macos-13, simd: sse2 } + - { os: macos-13, simd: sse3 } + - { os: macos-13, simd: sse4 } + - { os: macos-13, simd: avx } + - { os: macos-13, simd: avx2 } + - { os: macos-14, simd: none } + - { os: macos-14, simd: sse } + - { os: macos-14, simd: sse2 } + - { os: macos-14, simd: sse3 } + - { os: macos-14, simd: sse4 } + - { os: macos-14, simd: avx } + - { os: macos-14, simd: avx2 } + - { os: ubuntu-22.04, simd: none } + - { os: ubuntu-22.04, simd: sse } + - { os: ubuntu-22.04, simd: sse2 } + - { os: ubuntu-22.04, simd: sse3 } + - { os: ubuntu-22.04, simd: sse4 } + - { os: ubuntu-22.04, simd: avx } + - { os: ubuntu-22.04, simd: avx2 } + - { os: ubuntu-24.04, simd: none } + - { os: ubuntu-24.04, simd: sse } + - { os: ubuntu-24.04, simd: sse2 } + - { os: ubuntu-24.04, simd: sse3 } + - { os: ubuntu-24.04, simd: sse4 } + - { os: ubuntu-24.04, simd: avx } + - { os: ubuntu-24.04, simd: avx2 } + # ARM64 builds + - { os: ubuntu-latest-arm64, simd: neon } + + steps: + - uses: actions/checkout@v4 + + - name: Install Autotools on macOS + if: runner.os == 'macOS' + run: brew upgrade && brew install autoconf automake libtool + + - name: Install Autotools on Ubuntu + if: matrix.os == 'ubuntu-22.04' || matrix.os == 'ubuntu-24.04' + run: sudo apt-get install -y autoconf automake libtool + + - name: Set SIMD flags + run: | + if [ "${{ matrix.simd }}" == "none" ]; then + export CFLAGS="" + elif [ "${{ matrix.simd }}" == "sse" ]; then + export CFLAGS="-msse" + elif [ "${{ matrix.simd }}" == "sse2" ]; then + export CFLAGS="-msse2" + elif [ "${{ matrix.simd }}" == "sse3" ]; then + export CFLAGS="-msse3" + elif [ "${{ matrix.simd }}" == "sse4" ]; then + export CFLAGS="-msse4" + elif [ "${{ matrix.simd }}" == "avx" ]; then + export CFLAGS="-mavx" + elif [ "${{ matrix.simd }}" == "avx2" ]; then + export CFLAGS="-mavx2" + elif [ "${{ matrix.simd }}" == "neon" ]; then + export CFLAGS="-mfpu=neon" + fi + + - name: Generate Autotools + run: ./autogen.sh + + - name: Configure Autotools + run: ./configure CFLAGS="$CFLAGS" + + - name: Build + run: make + + - name: Test + run: make check + + build_cmake_ios: + name: CMake / iOS + runs-on: macos-14 + + steps: + - uses: actions/checkout@v4 + + - name: Configure CMake + run: | + cmake \ + -B build \ + -GXcode \ + -DCMAKE_SYSTEM_NAME=iOS \ + -DCMAKE_BUILD_TYPE=Release \ + -DCMAKE_XCODE_ATTRIBUTE_CODE_SIGNING_ALLOWED=NO \ + -DCGLM_STATIC=ON \ + -DCGLM_USE_TEST=ON + + - name: Build + run: cmake --build build + + build_cmake_ubuntu: + name: CMake / ${{ matrix.target.os }} / ${{ matrix.target.cc }} / ${{ matrix.target.arch }} / ${{ matrix.target.simd }} + runs-on: ${{ matrix.target.arch == 'arm64' && 'ubuntu-latest-arm64' || matrix.target.os }} + strategy: + fail-fast: false + matrix: + target: + # GCC 11 builds + - { os: ubuntu-20.04, cc: gcc-11, arch: x64, simd: none } + - { os: ubuntu-20.04, cc: gcc-11, arch: x64, simd: sse } + - { os: ubuntu-20.04, cc: gcc-11, arch: x64, simd: sse2 } + - { os: ubuntu-20.04, cc: gcc-11, arch: x64, simd: sse3 } + - { os: ubuntu-20.04, cc: gcc-11, arch: x64, simd: sse4 } + - { os: ubuntu-20.04, cc: gcc-11, arch: x64, simd: avx } + - { os: ubuntu-20.04, cc: gcc-11, arch: x64, simd: avx2 } + # GCC 12 builds + - { os: ubuntu-22.04, cc: gcc-12, arch: x64, simd: none } + - { os: ubuntu-22.04, cc: gcc-12, arch: x64, simd: sse } + - { os: ubuntu-22.04, cc: gcc-12, arch: x64, simd: sse2 } + - { os: ubuntu-22.04, cc: gcc-12, arch: x64, simd: sse3 } + - { os: ubuntu-22.04, cc: gcc-12, arch: x64, simd: sse4 } + - { os: ubuntu-22.04, cc: gcc-12, arch: x64, simd: avx } + - { os: ubuntu-22.04, cc: gcc-12, arch: x64, simd: avx2 } + # GCC 13 builds + - { os: ubuntu-24.04, cc: gcc-13, arch: x64, simd: none } + - { os: ubuntu-24.04, cc: gcc-13, arch: x64, simd: sse } + - { os: ubuntu-24.04, cc: gcc-13, arch: x64, simd: sse2 } + - { os: ubuntu-24.04, cc: gcc-13, arch: x64, simd: sse3 } + - { os: ubuntu-24.04, cc: gcc-13, arch: x64, simd: sse4 } + - { os: ubuntu-24.04, cc: gcc-13, arch: x64, simd: avx } + - { os: ubuntu-24.04, cc: gcc-13, arch: x64, simd: avx2 } + # Clang 12 builds + - { os: ubuntu-20.04, cc: clang-12, arch: x64, simd: none } + - { os: ubuntu-20.04, cc: clang-12, arch: x64, simd: sse } + - { os: ubuntu-20.04, cc: clang-12, arch: x64, simd: sse2 } + - { os: ubuntu-20.04, cc: clang-12, arch: x64, simd: sse3 } + - { os: ubuntu-20.04, cc: clang-12, arch: x64, simd: sse4 } + - { os: ubuntu-20.04, cc: clang-12, arch: x64, simd: avx } + - { os: ubuntu-20.04, cc: clang-12, arch: x64, simd: avx2 } + # Clang 15 builds + - { os: ubuntu-22.04, cc: clang-15, arch: x64, simd: none } + - { os: ubuntu-22.04, cc: clang-15, arch: x64, simd: sse } + - { os: ubuntu-22.04, cc: clang-15, arch: x64, simd: sse2 } + - { os: ubuntu-22.04, cc: clang-15, arch: x64, simd: sse3 } + - { os: ubuntu-22.04, cc: clang-15, arch: x64, simd: sse4 } + - { os: ubuntu-22.04, cc: clang-15, arch: x64, simd: avx } + - { os: ubuntu-22.04, cc: clang-15, arch: x64, simd: avx2 } + # ARM64 builds + - { os: ubuntu-latest, cc: gcc-12, arch: arm64, simd: neon } + - { os: ubuntu-latest, cc: gcc-13, arch: arm64, simd: neon } + # ARMv7 builds + - { os: ubuntu-latest-arm64, cc: gcc-12, arch: armv7, simd: neon } + - { os: ubuntu-latest-arm64, cc: gcc-12, arch: armv7, simd: none } + + steps: + - uses: actions/checkout@v4 + + - name: Add Ubuntu Toolchain PPA + if: matrix.target.os == 'ubuntu-20.04' + run: | + sudo apt-get update + sudo apt-get install -y software-properties-common + sudo add-apt-repository -y ppa:ubuntu-toolchain-r/test + sudo apt-get update + + - name: Install Compiler and Ninja + run: | + sudo apt-get install -y ${{ matrix.target.cc }} ninja-build + + - name: Set SIMD flags + run: | + if [ "${{ matrix.simd }}" == "none" ]; then + export CFLAGS="" + elif [ "${{ matrix.simd }}" == "sse" ]; then + export CFLAGS="-msse" + elif [ "${{ matrix.simd }}" == "sse2" ]; then + export CFLAGS="-msse2" + elif [ "${{ matrix.simd }}" == "sse3" ]; then + export CFLAGS="-msse3" + elif [ "${{ matrix.simd }}" == "sse4" ]; then + export CFLAGS="-msse4" + elif [ "${{ matrix.simd }}" == "avx" ]; then + export CFLAGS="-mavx" + elif [ "${{ matrix.simd }}" == "avx2" ]; then + export CFLAGS="-mavx2" + elif [ "${{ matrix.simd }}" == "neon" ]; then + export CFLAGS="-mfpu=neon" + fi + + - name: Configure CMake + run: | + if [ "${{ matrix.target.arch }}" == "armv7" ]; then + # Build for ARMv7 + neon_flags="" + if [ "${{ matrix.simd }}" == "neon" ]; then + neon_flags="-mfpu=neon -mfloat-abi=hard" + fi + cmake -B build -GNinja -DCMAKE_BUILD_TYPE=Release \ + -DCMAKE_C_COMPILER=${{ matrix.target.cc }} \ + -DCMAKE_C_FLAGS="$CFLAGS -m32 -march=armv7-a ${neon_flags}" \ + -DCGLM_STATIC=ON -DCGLM_USE_TEST=ON + elif [ "${{ matrix.target.arch }}" == "arm64" ]; then + # Build for ARM64 (AArch64) + neon_flags="" + if [ "${{ matrix.simd }}" == "neon" ]; then + neon_flags="+simd" # Enable SIMD/NEON features on ARM64 + else + neon_flags="+nosimd" # Explicitly disable SIMD/NEON + fi + cmake -B build -GNinja -DCMAKE_BUILD_TYPE=Release \ + -DCMAKE_C_COMPILER=${{ matrix.target.cc }} \ + -DCMAKE_C_FLAGS="$CFLAGS -march=armv8-a${neon_flags}" \ + -DCGLM_STATIC=ON -DCGLM_USE_TEST=ON + else + # Normal build (x86/x64) + cmake -B build -GNinja -DCMAKE_BUILD_TYPE=Release \ + -DCMAKE_C_COMPILER=${{ matrix.target.cc }} \ + -DCMAKE_C_FLAGS="$CFLAGS" \ + -DCGLM_STATIC=ON -DCGLM_USE_TEST=ON + fi + + - name: Build + run: cmake --build build + + - name: Test + working-directory: build + run: ./tests + + build_cmake_macos: + name: CMake / ${{ matrix.os }} + runs-on: ${{ matrix.os }} + strategy: + fail-fast: false + matrix: + os: [macos-13, macos-14] + + steps: + - uses: actions/checkout@v4 + + - name: Install Ninja + if: runner.os == 'macOS' + run: brew upgrade && brew install ninja + + - name: Configure CMake + run: | + cmake \ + -B build \ + -GNinja \ + -DCMAKE_BUILD_TYPE=Release \ + -DCGLM_STATIC=ON \ + -DCGLM_USE_TEST=ON + + - name: Build + run: cmake --build build + + - name: Test + working-directory: build + run: ./tests + + build_cmake: + name: CMake / ${{ matrix.os }} / ${{ matrix.simd }} + runs-on: ${{ matrix.os }} + strategy: + fail-fast: false + matrix: + include: + # x86/x64 builds + - { os: macos-13, simd: none } + - { os: macos-13, simd: sse } + - { os: macos-13, simd: sse2 } + - { os: macos-13, simd: sse3 } + - { os: macos-13, simd: sse4 } + - { os: macos-13, simd: avx } + - { os: macos-13, simd: avx2 } + - { os: macos-14, simd: none } + - { os: macos-14, simd: sse } + - { os: macos-14, simd: sse2 } + - { os: macos-14, simd: sse3 } + - { os: macos-14, simd: sse4 } + - { os: macos-14, simd: avx } + - { os: macos-14, simd: avx2 } + - { os: windows-2022, simd: none } + - { os: windows-2022, simd: sse } + - { os: windows-2022, simd: sse2 } + - { os: windows-2022, simd: sse3 } + - { os: windows-2022, simd: sse4 } + - { os: windows-2022, simd: avx } + - { os: windows-2022, simd: avx2 } + # ARM64 builds + - { os: macos-14-arm64, simd: neon } + + steps: + - uses: actions/checkout@v4 + + - name: Install Ninja on macOS + if: runner.os == 'macOS' + run: brew upgrade && brew install ninja + + - name: Set SIMD flags (Windows) + if: runner.os == 'Windows' + shell: pwsh + run: | + $simd = "${{ matrix.simd }}" + if ($simd -eq "none") { + $env:CFLAGS = "" + } elseif ($simd -eq "sse") { + $env:CFLAGS = "-arch:SSE" + } elseif ($simd -eq "sse2") { + $env:CFLAGS = "-arch:SSE2" + } elseif ($simd -eq "sse3") { + $env:CFLAGS = "-arch:SSE3" + } elseif ($simd -eq "sse4") { + $env:CFLAGS = "-arch:SSE4" + } elseif ($simd -eq "avx") { + $env:CFLAGS = "-arch:AVX" + } elseif ($simd -eq "avx2") { + $env:CFLAGS = "-arch:AVX2" + } elseif ($simd -eq "neon") { + $env:CFLAGS = "-arch:NEON" + } + + - name: Set SIMD flags (Unix) + if: runner.os != 'Windows' + shell: bash + run: | + if [ "${{ matrix.simd }}" == "none" ]; then + export CFLAGS="" + elif [ "${{ matrix.simd }}" == "sse" ]; then + export CFLAGS="-msse" + elif [ "${{ matrix.simd }}" == "sse2" ]; then + export CFLAGS="-msse2" + elif [ "${{ matrix.simd }}" == "sse3" ]; then + export CFLAGS="-msse3" + elif [ "${{ matrix.simd }}" == "sse4" ]; then + export CFLAGS="-msse4" + elif [ "${{ matrix.simd }}" == "avx" ]; then + export CFLAGS="-mavx" + elif [ "${{ matrix.simd }}" == "avx2" ]; then + export CFLAGS="-mavx2" + elif [ "${{ matrix.simd }}" == "neon" ]; then + export CFLAGS="-mfpu=neon" + fi + + - name: Configure CMake (Windows) + if: runner.os == 'Windows' + shell: pwsh + run: cmake -B build -G "Visual Studio 17 2022" -A x64 -T host=x64 -DCMAKE_BUILD_TYPE=Release -DCMAKE_C_FLAGS="$env:CFLAGS" -DCGLM_STATIC=ON -DCGLM_USE_TEST=ON + + - name: Configure CMake (Unix) + if: runner.os != 'Windows' + shell: bash + run: cmake -B build -GNinja -DCMAKE_BUILD_TYPE=Release -DCMAKE_C_FLAGS="$CFLAGS" -DCGLM_STATIC=ON -DCGLM_USE_TEST=ON + + - name: Build + run: cmake --build build + + - name: Test (Windows) + if: runner.os == 'Windows' + shell: pwsh + working-directory: build + run: .\Debug\tests.exe + + - name: Test (Unix) + if: runner.os != 'Windows' + shell: bash + working-directory: build + run: ./tests + + build_meson: + name: Meson / ${{ matrix.os }} / ${{ matrix.simd }} + runs-on: ${{ contains(matrix.os, 'arm64') && 'ubuntu-latest-arm64' || matrix.os }} + strategy: + fail-fast: false + matrix: + include: + # x86/x64 builds + - { os: macos-14, simd: none } + - { os: macos-14, simd: sse } + - { os: macos-14, simd: sse2 } + - { os: macos-14, simd: sse3 } + - { os: macos-14, simd: sse4 } + - { os: macos-14, simd: avx } + - { os: macos-14, simd: avx2 } + - { os: ubuntu-22.04, simd: none } + - { os: ubuntu-22.04, simd: sse } + - { os: ubuntu-22.04, simd: sse2 } + - { os: ubuntu-22.04, simd: sse3 } + - { os: ubuntu-22.04, simd: sse4 } + - { os: ubuntu-22.04, simd: avx } + - { os: ubuntu-22.04, simd: avx2 } + - { os: ubuntu-24.04, simd: none } + - { os: ubuntu-24.04, simd: sse } + - { os: ubuntu-24.04, simd: sse2 } + - { os: ubuntu-24.04, simd: sse3 } + - { os: ubuntu-24.04, simd: sse4 } + - { os: ubuntu-24.04, simd: avx } + - { os: ubuntu-24.04, simd: avx2 } + - { os: windows-2022, simd: none } + - { os: windows-2022, simd: sse } + - { os: windows-2022, simd: sse2 } + - { os: windows-2022, simd: sse3 } + - { os: windows-2022, simd: sse4 } + - { os: windows-2022, simd: avx } + - { os: windows-2022, simd: avx2 } + # ARM64 builds + - { os: ubuntu-latest-arm64, simd: neon } + + steps: + - uses: actions/checkout@v4 + + - uses: actions/setup-python@v5 + with: + python-version: '3.12' + cache: 'pip' + + - name: Install meson + run: python3 -m pip install meson ninja + + - name: Set SIMD flags (Windows) + if: runner.os == 'Windows' + shell: pwsh + run: | + $simd = "${{ matrix.simd }}" + if ($simd -eq "none") { + $env:CFLAGS = "" + } elseif ($simd -eq "sse") { + $env:CFLAGS = "-arch:SSE" + } elseif ($simd -eq "sse2") { + $env:CFLAGS = "-arch:SSE2" + } elseif ($simd -eq "sse3") { + $env:CFLAGS = "-arch:SSE3" + } elseif ($simd -eq "sse4") { + $env:CFLAGS = "-arch:SSE4" + } elseif ($simd -eq "avx") { + $env:CFLAGS = "-arch:AVX" + } elseif ($simd -eq "avx2") { + $env:CFLAGS = "-arch:AVX2" + } elseif ($simd -eq "neon") { + $env:CFLAGS = "-arch:NEON" + } + + - name: Set SIMD flags (Unix) + if: runner.os != 'Windows' + shell: bash + run: | + if [ "${{ matrix.simd }}" == "none" ]; then + export CFLAGS="" + elif [ "${{ matrix.simd }}" == "sse" ]; then + export CFLAGS="-msse" + elif [ "${{ matrix.simd }}" == "sse2" ]; then + export CFLAGS="-msse2" + elif [ "${{ matrix.simd }}" == "sse3" ]; then + export CFLAGS="-msse3" + elif [ "${{ matrix.simd }}" == "sse4" ]; then + export CFLAGS="-msse4" + elif [ "${{ matrix.simd }}" == "avx" ]; then + export CFLAGS="-mavx" + elif [ "${{ matrix.simd }}" == "avx2" ]; then + export CFLAGS="-mavx2" + elif [ "${{ matrix.simd }}" == "neon" ]; then + export CFLAGS="-mfpu=neon" + fi + + - name: Build with meson (Windows) + if: runner.os == 'Windows' + shell: pwsh + run: | + meson setup build -Dbuildtype=release --default-library=static -Dbuild_tests=true -Dc_args="$env:CFLAGS" + meson test -C build + + - name: Build with meson (Unix) + if: runner.os != 'Windows' + shell: bash + run: | + meson setup build -Dbuildtype=release --default-library=static -Dbuild_tests=true -Dc_args="$CFLAGS" + meson test -C build + + build_msbuild: + name: MSBuild / Windows / ${{ matrix.simd }} + runs-on: windows-2022 + strategy: + fail-fast: false + matrix: + simd: [none, sse, sse2, sse3, sse4, avx, avx2, neon] + + steps: + - uses: actions/checkout@v4 + + - uses: microsoft/setup-msbuild@v2 + + - name: Retarget solution + run: | + vswhere -latest -products * -requires Microsoft.VisualStudio.Component.VC.Tools.x86.x64 -property installationPath + $vsInstallPath = vswhere -latest -products * -requires Microsoft.VisualStudio.Component.VC.Tools.x86.x64 -property installationPath + & "$vsInstallPath\Common7\IDE\devenv.com" cglm.sln /Upgrade + + - name: Set SIMD flags + run: | + if ($Env:SIMD -eq 'none') { + $Env:CFLAGS="" + } elseif ($Env:SIMD -eq 'sse') { + $Env:CFLAGS="-arch:SSE" + } elseif ($Env:SIMD -eq 'sse2') { + $Env:CFLAGS="-arch:SSE2" + } elseif ($Env:SIMD -eq 'sse3') { + $Env:CFLAGS="-arch:SSE3" + } elseif ($Env:SIMD -eq 'sse4') { + $Env:CFLAGS="-arch:SSE4" + } elseif ($Env:SIMD -eq 'avx') { + $Env:CFLAGS="-arch:AVX" + } elseif ($Env:SIMD -eq 'avx2') { + $Env:CFLAGS="-arch:AVX2" + } elseif ($Env:SIMD -eq 'neon') { + $Env:CFLAGS="-arch:NEON" + } + + - name: Build (x86) + working-directory: win + run: msbuild cglm.vcxproj /p:Configuration=Release /p:Platform=x86 /p:PlatformToolset=v143 /p:BuildInParallel=true /p:AdditionalOptions="$Env:CFLAGS" + + - name: Build (x64) + working-directory: win + run: msbuild cglm.vcxproj /p:Configuration=Release /p:Platform=x64 /p:PlatformToolset=v143 /p:BuildInParallel=true /p:AdditionalOptions="$Env:CFLAGS" + + build_documentation: + name: Documentation + runs-on: ubuntu-22.04 + + steps: + - uses: actions/checkout@v4 + + - uses: actions/setup-python@v5 + with: + python-version: '3.12' + + - name: Install Dependencies + working-directory: docs + run: python3 -m pip install -r requirements.txt + + - name: Build + working-directory: docs + run: sphinx-build -W --keep-going source build + + build_swift: + name: Swift ${{ matrix.swift }} / ${{ matrix.os }} + runs-on: ${{ matrix.os }} + strategy: + fail-fast: false + matrix: + os: [macos-13, macos-14, ubuntu-22.04] + + # This has no test yet. + steps: + - uses: actions/checkout@v4 + + - name: Build + run: swift build + + build_cmake_arm: + name: CMake / ARM / ${{ matrix.os }} / ${{ matrix.arch }} / ${{ matrix.simd }} + runs-on: ${{ matrix.os }} + strategy: + fail-fast: false + matrix: + include: + # Linux ARM builds + - os: ubuntu-latest-arm64 + arch: arm64 + simd: neon + - os: ubuntu-latest-arm64 + arch: armv7 + simd: neon + - os: ubuntu-latest-arm64 + arch: armv7 + simd: none + # Windows ARM builds + - os: windows-latest-arm64 + arch: arm64 + simd: neon + - os: windows-latest-arm64 + arch: arm + simd: neon + - os: windows-latest-arm64 + arch: arm + simd: none + + steps: + - uses: actions/checkout@v4 + + - name: Configure CMake (Windows) + if: runner.os == 'Windows' + shell: pwsh + run: | + $flags = "" + if ("${{ matrix.arch }}" -eq "arm") { + $flags = "-m32 -march=armv7-a" + if ("${{ matrix.simd }}" -eq "neon") { + $flags += " -mfpu=neon" + } + } + elseif ("${{ matrix.simd }}" -eq "neon") { + $flags = "-march=armv8-a+simd" + } + + cmake -B build -G "Visual Studio 17 2022" -A ${{ matrix.arch == 'arm64' && 'ARM64' || 'ARM' }} ` + -DCMAKE_BUILD_TYPE=Release ` + -DCMAKE_C_FLAGS="$flags" ` + -DCGLM_STATIC=ON -DCGLM_USE_TEST=ON + + - name: Configure CMake (Unix) + if: runner.os != 'Windows' + shell: bash + run: | + flags="" + if [ "${{ matrix.arch }}" = "armv7" ]; then + flags="-m32 -march=armv7-a" + if [ "${{ matrix.simd }}" = "neon" ]; then + flags="$flags -mfpu=neon -mfloat-abi=hard" + fi + elif [ "${{ matrix.simd }}" = "neon" ]; then + flags="-march=armv8-a+simd" + fi + + cmake -B build -GNinja -DCMAKE_BUILD_TYPE=Release \ + -DCMAKE_C_FLAGS="$flags" \ + -DCGLM_STATIC=ON -DCGLM_USE_TEST=ON + + - name: Build + run: cmake --build build + + - name: Test + working-directory: build + run: ./tests diff --git a/cglm/.github/workflows/cmake-wasm.yml b/cglm/.github/workflows/cmake-wasm.yml new file mode 100644 index 0000000..cdccaa6 --- /dev/null +++ b/cglm/.github/workflows/cmake-wasm.yml @@ -0,0 +1,107 @@ +name: CMake WebAssembly + +on: + push: + branches: [ "master" ] + pull_request: + branches: [ "master" ] + +env: + wasmtime_version: v7.0.0 + wasmer_version: v3.1.1 + +jobs: + build_wasi_sdk: + strategy: + matrix: + BUILD_TYPE: [Release, Debug, RelWithDebInfo, MinSizeRel] + C_FLAGS: ['', '-msimd128'] + wasi_sdk_version: [19, 20] + # The CMake configure and build commands are platform agnostic and should work equally well on Windows or Mac. + # You can convert this to a matrix build if you need cross-platform coverage. + # See: https://docs.github.com/en/free-pro-team@latest/actions/learn-github-actions/managing-complex-workflows#using-a-build-matrix + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v4 + + - name: Downloading wasi-sdk + run: | + cd ${{github.workspace}} + wget --no-verbose https://github.com/WebAssembly/wasi-sdk/releases/download/wasi-sdk-${{matrix.wasi_sdk_version}}/wasi-sdk-${{matrix.wasi_sdk_version}}.0-linux.tar.gz + tar xf wasi-sdk-${{matrix.wasi_sdk_version}}.0-linux.tar.gz + + # Building a wasm library without needing to define a main(): + # https://github.com/WebAssembly/wasi-sdk/issues/332 + - name: Modify CMakeLists.txt for WASI + run: | + echo 'if (CMAKE_SYSTEM_NAME STREQUAL "WASI")' >> CMakeLists.txt + echo ' target_link_options(${PROJECT_NAME} PRIVATE -mexec-model=reactor)' >> CMakeLists.txt + echo 'endif()' >> CMakeLists.txt + + - name: Configure CMake + # Configure CMake in a 'build' subdirectory. `CMAKE_BUILD_TYPE` is only required if you are using a single-configuration generator such as make. + # See https://cmake.org/cmake/help/latest/variable/CMAKE_BUILD_TYPE.html?highlight=cmake_build_type + # Below suppress <<'clock' is deprecated: WASI lacks process-associated clocks; ...>> warns: + # -D_WASI_EMULATED_PROCESS_CLOCKS" -DCMAKE_EXE_LINKER_FLAGS="-lwasi-emulated-process-clocks + run: cmake -B ${{github.workspace}}/build -DCMAKE_BUILD_TYPE=${{matrix.BUILD_TYPE}} -DCMAKE_C_FLAGS="${{matrix.C_FLAGS}} -D_WASI_EMULATED_PROCESS_CLOCKS" -DCMAKE_EXE_LINKER_FLAGS="-lwasi-emulated-process-clocks" -DCMAKE_TOOLCHAIN_FILE=${{github.workspace}}/wasi-sdk-${{matrix.wasi_sdk_version}}.0/share/cmake/wasi-sdk.cmake -DWASI_SDK_PREFIX=${{github.workspace}}/wasi-sdk-${{matrix.wasi_sdk_version}}.0 -DCGLM_STATIC=ON -DCGLM_SHARED=OFF -DCGLM_USE_TEST=ON + + - name: Build + # Build your program with the given configuration + run: cmake --build ${{github.workspace}}/build --config ${{matrix.BUILD_TYPE}} + + - name: Test with wasmtime + run: | + cd ${{github.workspace}} + ls -lh ${{github.workspace}}/build/ + wget --no-verbose https://github.com/bytecodealliance/wasmtime/releases/download/${{env.wasmtime_version}}/wasmtime-${{env.wasmtime_version}}-x86_64-linux.tar.xz + tar xf wasmtime-${{env.wasmtime_version}}-x86_64-linux.tar.xz + ./wasmtime-${{env.wasmtime_version}}-x86_64-linux/wasmtime run --wasm-features simd ${{github.workspace}}/build/tests + + - name: Test with wasmer + run: | + cd ${{github.workspace}} + mkdir wasmer + cd wasmer + wget --no-verbose https://github.com/wasmerio/wasmer/releases/download/${{env.wasmer_version}}/wasmer-linux-amd64.tar.gz + tar xf wasmer-linux-amd64.tar.gz + ./bin/wasmer run --enable-simd ${{github.workspace}}/build/tests + + build_emsdk: + strategy: + matrix: + BUILD_TYPE: [Release, Debug, RelWithDebInfo, MinSizeRel] + C_FLAGS: ['', '-msimd128', '-msse -msse2 -msimd128', '-msse -msse2 -msse3 -msse4 -msimd128'] + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - name: Setup emsdk + uses: mymindstorm/setup-emsdk@v12 + + - name: Verify emsdk + run: emcc -v + + - name: Configure CMake + # Configure CMake in a 'build' subdirectory. `CMAKE_BUILD_TYPE` is only required if you are using a single-configuration generator such as make. + # See https://cmake.org/cmake/help/latest/variable/CMAKE_BUILD_TYPE.html?highlight=cmake_build_type + run: emcmake cmake -B ${{github.workspace}}/build -DCMAKE_BUILD_TYPE=${{matrix.BUILD_TYPE}} -DCMAKE_C_FLAGS="${{matrix.C_FLAGS}}" -DCMAKE_EXE_LINKER_FLAGS="-s STANDALONE_WASM" -DCGLM_STATIC=ON -DCGLM_USE_TEST=ON + + - name: Build + # Build your program with the given configuration + run: cmake --build ${{github.workspace}}/build --config ${{matrix.BUILD_TYPE}} + + - name: Test with wasmtime + run: | + cd ${{github.workspace}} + ls -lh ${{github.workspace}}/build/ + wget --no-verbose https://github.com/bytecodealliance/wasmtime/releases/download/${{env.wasmtime_version}}/wasmtime-${{env.wasmtime_version}}-x86_64-linux.tar.xz + tar xf wasmtime-${{env.wasmtime_version}}-x86_64-linux.tar.xz + ./wasmtime-${{env.wasmtime_version}}-x86_64-linux/wasmtime run --wasm-features simd ${{github.workspace}}/build/tests.wasm + - name: Test with wasmer + run: | + cd ${{github.workspace}} + mkdir wasmer + cd wasmer + wget --no-verbose https://github.com/wasmerio/wasmer/releases/download/${{env.wasmer_version}}/wasmer-linux-amd64.tar.gz + tar xf wasmer-linux-amd64.tar.gz + ./bin/wasmer run --enable-simd ${{github.workspace}}/build/tests.wasm diff --git a/cglm/.github/workflows/meson-wasm.yml b/cglm/.github/workflows/meson-wasm.yml new file mode 100644 index 0000000..bd913a2 --- /dev/null +++ b/cglm/.github/workflows/meson-wasm.yml @@ -0,0 +1,79 @@ +name: Meson WebAssembly + +on: + push: + branches: [ "master" ] + pull_request: + branches: [ "master" ] + +env: + wasmtime_version: v7.0.0 + wasmer_version: v3.1.1 + +jobs: + build_emsdk: + strategy: + matrix: + BUILD_TYPE: [debug, debugoptimized, release, minsize] + C_FLAGS: ['', '-msimd128', '-msse -msse2 -msimd128', '-msse -msse2 -msse3 -msse4 -msimd128'] + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - name: Setup emsdk + uses: mymindstorm/setup-emsdk@v13 + + - name: Verify emsdk + run: emcc -v + + - name: Creating cross file + run: | + cat << EOF > ${{github.workspace}}/meson_cross_emsdk.txt + [binaries] + c = '`which emcc`' + cpp = '`which em++`' + ar = '`which emar`' + + [built-in options] + + c_args = ['-Wno-unused-parameter'] + c_link_args = ['-s', 'STANDALONE_WASM'] + cpp_args = ['-Wno-unused-parameter'] + cpp_link_args = ['-s', 'STANDALONE_WASM'] + + [host_machine] + + system = 'emscripten' + cpu_family = 'wasm32' + cpu = 'wasm32' + endian = 'little' + + EOF + cat ${{github.workspace}}/meson_cross_emsdk.txt + + - uses: actions/setup-python@v4 + + - name: Install meson + run: | + sudo python3 -m pip install meson ninja + + - name: Build with meson + run: | + meson setup build -Dbuildtype=${{matrix.BUILD_TYPE}} --cross-file ${{github.workspace}}/meson_cross_emsdk.txt --default-library=static -Dbuild_tests=true + meson test -C build + + - name: Test with wasmtime + run: | + cd ${{github.workspace}} + ls -lh ${{github.workspace}}/build/ + wget --no-verbose https://github.com/bytecodealliance/wasmtime/releases/download/${{env.wasmtime_version}}/wasmtime-${{env.wasmtime_version}}-x86_64-linux.tar.xz + tar xf wasmtime-${{env.wasmtime_version}}-x86_64-linux.tar.xz + ./wasmtime-${{env.wasmtime_version}}-x86_64-linux/wasmtime run --wasm-features simd ${{github.workspace}}/build/tests.wasm + + - name: Test with wasmer + run: | + cd ${{github.workspace}} + mkdir wasmer + cd wasmer + wget --no-verbose https://github.com/wasmerio/wasmer/releases/download/${{env.wasmer_version}}/wasmer-linux-amd64.tar.gz + tar xf wasmer-linux-amd64.tar.gz + ./bin/wasmer run --enable-simd ${{github.workspace}}/build/tests.wasm diff --git a/cglm/.gitignore b/cglm/.gitignore new file mode 100644 index 0000000..879f459 --- /dev/null +++ b/cglm/.gitignore @@ -0,0 +1,83 @@ +*.xcodeproj +*.xcworkspace +*.sln +*.vcxproj +*.vcxproj.* +*.suo +*.sdf +*.opensdf +ipch/ +Debug/ +Release/ +.DS_Store +.vs +*.nupkg +*.opendb +packages.config +/aclocal.m4 +/ar-lib +/autom4te.cache/ +/compile +/config.guess +/config.log +/config.status +/config.sub +/configure +/depcomp +/install-sh +/ltmain.sh +/missing +/libtool +/.libs/ +.deps/ +*.[oa] +*.l[oa] +Makefile +Makefile.in +m4/*.m4 +.buildstamp +.dirstamp +packages/ +.anjuta/* +*.anjuta* +config.h.* +/config.h +stamp* +COPYING +.idea/* +*.VC.db +cscope.* +*-git-ignored-file.* +test/*.trs +test/test_* +*.log +test/.libs/* +test/tests +cglm_arm/* +cglm_test_ios/* +cglm_test_iosTests/* +docs/build/* +win/cglm_test_* +* copy.* +*.o +*.obj +*codeanalysis.*.xml +*codeanalysis.xml +*.lib +*.tlog +win/x64 +win/x85 +win/Debug +cglm-test-ios* +/cglm.pc +test-driver +Default-568h@2x.png +build/ +conftest.dir/* +confdefs.h +*.xcuserdatad +.idea +cmake-build-debug +*.o.tmp +xcode/* +.vscode \ No newline at end of file diff --git a/cglm/.readthedocs.yaml b/cglm/.readthedocs.yaml new file mode 100644 index 0000000..cf6b4b3 --- /dev/null +++ b/cglm/.readthedocs.yaml @@ -0,0 +1,39 @@ +# Read the Docs configuration file for Sphinx projects +# See https://docs.readthedocs.io/en/stable/config-file/v2.html for details + +# Required +version: 2 + +# Set the OS, Python version and other tools you might need +build: + os: ubuntu-22.04 + tools: + python: "3.12" + # You can also specify other tool versions: + # nodejs: "20" + # rust: "1.70" + # golang: "1.20" + +# Build documentation in the "docs/" directory with Sphinx +sphinx: + configuration: docs/source/conf.py + # You can configure Sphinx to use a different builder, for instance use the dirhtml builder for simpler URLs + # builder: "dirhtml" + # Fail on all warnings to avoid broken references + # fail_on_warning: true + +# Optionally build your docs in additional formats such as PDF and ePub +# formats: +# - pdf +# - epub + +# Optional but recommended, declare the Python requirements required +# to build your documentation +# See https://docs.readthedocs.io/en/stable/guides/reproducible-builds.html +# python: +# install: +# - requirements: docs/requirements.txt + +python: + install: + - requirements: docs/requirements.txt \ No newline at end of file diff --git a/cglm/.travis.yml b/cglm/.travis.yml new file mode 100644 index 0000000..7d3d6e0 --- /dev/null +++ b/cglm/.travis.yml @@ -0,0 +1,68 @@ +language: c + +os: + - linux + - osx + +arch: + - amd64 + - ppc64le + - s390x + - arm64 + +sudo: required +dist: trusty + +compiler: + - clang + - gcc + +matrix: + fast_finish: true + exclude: + # Skip GCC builds on macOS. + - os: osx + compiler: gcc + include: + # Additional GCC builds for code coverage. + - os: linux + compiler: gcc + env: CODE_COVERAGE=ON + +cache: + apt: true + +addons: + apt: + packages: + - clang-3.6 + - lcov + +branches: + only: + - master + +script: + - sh ./autogen.sh + - if [[ "$CC" == "gcc" && "$CODE_COVERAGE" == "ON" ]]; then + ./configure CFLAGS="-ftest-coverage -fprofile-arcs -coverage"; + else + ./configure; + fi + - make + - make check + +after_success: + - if [[ "$CC" == "gcc" && "$CODE_COVERAGE" == "ON" ]]; then + pip install --user cpp-coveralls && + coveralls + --build-root . + --exclude lib + --exclude test + --gcov-options '\-lp' + --verbose && + bash <(curl -s https://codecov.io/bash); + fi + +# after_failure: +# - cat ./test-suite.log diff --git a/cglm/BUILDING.md b/cglm/BUILDING.md new file mode 100644 index 0000000..01452f5 --- /dev/null +++ b/cglm/BUILDING.md @@ -0,0 +1,199 @@ +# Building the library + +cglm can be built using one of the following build systems: + +## CMake (All platforms) +```bash +$ mkdir build +$ cd build +$ cmake .. # [Optional] -DCGLM_SHARED=ON +$ make +$ sudo make install # [Optional] +``` + +### Options with defaults + +```CMake +option(CGLM_SHARED "Shared build" ON) +option(CGLM_STATIC "Static build" OFF) +option(CGLM_USE_C99 "" OFF) # C11 +option(CGLM_USE_TEST "Enable Tests" OFF) # for make check - make test +``` + +### Including in a CMake project + +#### Header only + +This requires no building or installation of cglm. + +* Example: + +``` cmake +cmake_minimum_required(VERSION 3.8.2) + +project() + +add_executable(${PROJECT_NAME} src/main.c) +target_link_libraries(${LIBRARY_NAME} PRIVATE + cglm_headers) + +add_subdirectory(external/cglm/ EXCLUDE_FROM_ALL) +``` + +#### Linked + +* Example: +```cmake +cmake_minimum_required(VERSION 3.8.2) + +project() + +add_executable(${PROJECT_NAME} src/main.c) +target_link_libraries(${LIBRARY_NAME} PRIVATE + cglm) + +add_subdirectory(external/cglm/) + +# or you can use find_package to configure cglm +``` + +### Using CMake to build for WebAssembly + +Since math functions like `sinf` are used, this can not be targeted at `wasm32-unknown-unknown`, one of [wasi-sdk](https://github.com/WebAssembly/wasi-sdk) or [emscripten](https://github.com/emscripten-core/emsdk) should be used. + +Should note that shared build is not yet supported for WebAssembly. + +For [simd128](https://github.com/WebAssembly/simd) support, add `-msimd128` to `CMAKE_C_FLAGS`, in command line `-DCMAKE_C_FLAGS="-msimd128"`. + +For tests, the cmake option `CGLM_USE_TEST` would still work, you'll need a wasi runtime for running tests, see our [ci config file](.github/workflows/cmake-wasm.yml) for a detailed example. + +#### WASI SDK + +```bash +$ cmake .. \ + -DCMAKE_TOOLCHAIN_FILE=/path/to/wasi-sdk-19.0/share/cmake/wasi-sdk.cmake \ + -DWASI_SDK_PREFIX=/path/to/wasi-sdk-19.0 +``` + +Where `/path/to/wasi-sdk-19.0/` is the path to extracted [wasi sdk](https://github.com/WebAssembly/wasi-sdk). + +In this case it would by default make a static build. + +#### Emscripten + +```bash +$ emcmake cmake .. \ + -DCMAKE_EXE_LINKER_FLAGS="-s STANDALONE_WASM" \ + -DCGLM_STATIC=ON +``` + +The `emcmake` here is the cmake wrapper for Emscripten from installed [emsdk](https://github.com/emscripten-core/emsdk). + +## Meson (All platforms) + +```bash +$ meson build # [Optional] --default-library=static +$ cd build +$ ninja +$ sudo ninja install # [Optional] +``` + +### Options with Defaults: + +```meson +c_std=c11 +buildtype=release +default_library=shared +build_tests=true # to run tests: ninja test +``` +### Including in a Meson project +* Example: +```meson +# Clone cglm or create a cglm.wrap under /subprojects +project('name', 'c') + +cglm_dep = dependency('cglm', fallback : 'cglm', 'cglm_dep') + +executable('exe', 'src/main.c', dependencies : cglm_dep) +``` + +## Swift (Swift Package Manager) + +Currently only default build options are supported. Add **cglm** dependency to your project: + +```swift +... +Package( + ... + dependencies: [ + ... + .package(url: "https://github.com/recp/cglm", .branch("master")), + ] + ... +) +``` + +Now add **cgml** as a dependency to your target. Product choices are: +- **cglm** for inlined version of the library which can be linked only statically +- **cglmc** for a compiled version of the library with no linking limitation + +```swift +... +.target( + ... + dependencies: [ + ... + .product(name: "cglm", package: "cglm"), + ] + ... +) +... +``` + +## Unix (Autotools) + +```bash +$ sh autogen.sh +$ ./configure +$ make +$ make check # [Optional] +$ [sudo] make install # [Optional] +``` + +This will also install pkg-config files so you can use +`pkg-config --cflags cglm` and `pkg-config --libs cglm` to retrieve compiler +and linker flags. + +The files will be installed into the given prefix (usually `/usr/local` by +default on Linux), but your pkg-config may not be configured to actually check +there. You can figure out where it's looking by running `pkg-config --variable +pc_path pkg-config` and change the path the files are installed to via +`./configure --with-pkgconfigdir=/your/path`. Alternatively, you can add the +prefix path to your `PKG_CONFIG_PATH` environment variable. + +## Windows (MSBuild) +Windows related build file and project files are located in `win` folder, +make sure you are inside `cglm/win` folder. +Code Analysis is enabled, so it may take awhile to build. + +```Powershell +$ cd win +$ .\build.bat +``` +if `msbuild` won't work (because of multi version VS) then try to build with `devenv`: +```Powershell +$ devenv cglm.sln /Build Release +``` + +### Running Tests on Windows + +You can see test project in same visual studio solution file. It is enough to run that project to run tests. + +# Building the documentation +First you need install Sphinx: http://www.sphinx-doc.org/en/master/usage/installation.html +then: +```bash +$ cd docs +$ sphinx-build source build +``` +it will compile docs into build folder, you can run index.html inside that function. \ No newline at end of file diff --git a/cglm/CMakeLists.txt b/cglm/CMakeLists.txt new file mode 100644 index 0000000..eaef8fa --- /dev/null +++ b/cglm/CMakeLists.txt @@ -0,0 +1,197 @@ +cmake_minimum_required(VERSION 3.13) +project(cglm + VERSION 0.9.6 + HOMEPAGE_URL https://github.com/recp/cglm + DESCRIPTION "OpenGL Mathematics (glm) for C" + LANGUAGES C +) + +set(CMAKE_C_STANDARD 11) +set(CMAKE_C_STANDARD_REQUIRED YES) +set(DEFAULT_BUILD_TYPE "Release") + +set(CGLM_BUILD) +option(CGLM_SHARED "Shared build" ON) +option(CGLM_STATIC "Static build" OFF) +option(CGLM_USE_C99 "" OFF) +option(CGLM_USE_TEST "Enable Tests" OFF) + +if(CMAKE_SYSTEM_NAME STREQUAL WASI) + set(CGLM_STATIC ON CACHE BOOL "Static option" FORCE) + set(CGLM_SHARED OFF CACHE BOOL "Shared option" FORCE) +endif() + +if(NOT CGLM_STATIC AND CGLM_SHARED) + set(CGLM_BUILD SHARED) +else(CGLM_STATIC) + set(CGLM_BUILD STATIC) +endif() + +if(CGLM_USE_C99) + set(CMAKE_C_STANDARD 99) +endif() + +if(MSVC) + add_definitions(-D_WINDOWS -D_USRDLL) + + if(NOT CMAKE_BUILD_TYPE MATCHES Debug) + add_definitions(-DNDEBUG) + add_compile_options(/W3 /Ox /Gy /Oi /TC) + foreach(flag_var + CMAKE_C_FLAGS CMAKE_C_FLAGS_DEBUG CMAKE_C_FLAGS_RELEASE + CMAKE_C_FLAGS_MINSIZEREL CMAKE_C_FLAGS_RELWITHDEBINFO) + string(REGEX REPLACE "/RTC(su|[1su])" "" ${flag_var} "${${flag_var}}") + endforeach(flag_var) + endif() +else() + add_compile_options(-Wall) + + if(NOT CMAKE_BUILD_TYPE MATCHES Debug) + add_compile_options(-O3) + endif() +endif() + +get_directory_property(hasParent PARENT_DIRECTORY) + +if(NOT hasParent AND NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES) + message(STATUS "Setting build type to '${DEFAULT_BUILD_TYPE}' as none was specified.") + set(CMAKE_BUILD_TYPE "${DEFAULT_BUILD_TYPE}" CACHE STRING "Choose the type of build." FORCE) + # Set the possible values of build type for cmake-gui + set_property(CACHE CMAKE_BUILD_TYPE PROPERTY STRINGS "Debug" "Release" "MinSizeRel" "RelWithDebInfo") +endif() + +include(GNUInstallDirs) + +set(CPACK_PROJECT_NAME ${PROJECT_NAME}) +set(CPACK_PROJECT_VERSION ${PROJECT_VERSION}) + +if(NOT CPack_CMake_INCLUDED) + include(CPack) +endif() + +# Target Start +add_library(${PROJECT_NAME} + ${CGLM_BUILD} + src/euler.c + src/affine.c + src/io.c + src/quat.c + src/cam.c + src/vec2.c + src/ivec2.c + src/vec3.c + src/ivec3.c + src/vec4.c + src/ivec4.c + src/mat2.c + src/mat2x3.c + src/mat2x4.c + src/mat3.c + src/mat3x2.c + src/mat3x4.c + src/mat4.c + src/mat4x2.c + src/mat4x3.c + src/plane.c + src/noise.c + src/frustum.c + src/box.c + src/aabb2d.c + src/project.c + src/sphere.c + src/ease.c + src/curve.c + src/bezier.c + src/ray.c + src/affine2d.c + src/clipspace/ortho_lh_no.c + src/clipspace/ortho_lh_zo.c + src/clipspace/ortho_rh_no.c + src/clipspace/ortho_rh_zo.c + src/clipspace/persp_lh_no.c + src/clipspace/persp_lh_zo.c + src/clipspace/persp_rh_no.c + src/clipspace/persp_rh_zo.c + src/clipspace/view_lh_no.c + src/clipspace/view_lh_zo.c + src/clipspace/view_rh_no.c + src/clipspace/view_rh_zo.c + src/clipspace/project_no.c + src/clipspace/project_zo.c + ) + +if(CGLM_SHARED) + add_definitions(-DCGLM_EXPORTS) +else() + target_compile_definitions(${PROJECT_NAME} PUBLIC -DCGLM_STATIC) +endif() + +set_target_properties(${PROJECT_NAME} PROPERTIES + VERSION ${PROJECT_VERSION} + SOVERSION ${PROJECT_VERSION_MAJOR}) + +if(WIN32) + # Because SOVERSION has no effect to file naming on Windows + set_target_properties(${PROJECT_NAME} PROPERTIES + RUNTIME_OUTPUT_NAME ${PROJECT_NAME}-${PROJECT_VERSION_MAJOR}) +endif() + +target_include_directories(${PROJECT_NAME} + PUBLIC + $ + $ + PRIVATE + ${CMAKE_CURRENT_SOURCE_DIR}/src +) + +# Target for header-only usage +add_library(${PROJECT_NAME}_headers INTERFACE) +target_include_directories(${PROJECT_NAME}_headers INTERFACE + ${CMAKE_CURRENT_SOURCE_DIR}/include) + +# Test Configuration +if(CGLM_USE_TEST) + include(CTest) + enable_testing() + add_subdirectory(test) +endif() + +# Install +install(TARGETS ${PROJECT_NAME} + EXPORT ${PROJECT_NAME} + LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR} + ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR} + RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR}) + +install(DIRECTORY include/${PROJECT_NAME} DESTINATION ${CMAKE_INSTALL_INCLUDEDIR} + PATTERN ".*" EXCLUDE) + +# Config +export(TARGETS ${PROJECT_NAME} + NAMESPACE ${PROJECT_NAME}:: + FILE "${CMAKE_CURRENT_BINARY_DIR}/${PROJECT_NAME}Config.cmake" +) + +install(EXPORT ${PROJECT_NAME} + FILE "${PROJECT_NAME}Config.cmake" + NAMESPACE ${PROJECT_NAME}:: + DESTINATION ${CMAKE_INSTALL_LIBDIR}/cmake/${PROJECT_NAME}) + +set(PACKAGE_NAME ${PROJECT_NAME}) +set(prefix ${CMAKE_INSTALL_PREFIX}) +set(exec_prefix ${CMAKE_INSTALL_PREFIX}) +if (IS_ABSOLUTE "${CMAKE_INSTALL_INCLUDEDIR}") + set(includedir "${CMAKE_INSTALL_INCLUDEDIR}") +else() + set(includedir "\${prefix}/${CMAKE_INSTALL_INCLUDEDIR}") +endif() +if (IS_ABSOLUTE "${CMAKE_INSTALL_LIBDIR}") + set(libdir "${CMAKE_INSTALL_LIBDIR}") +else() + set(libdir "\${exec_prefix}/${CMAKE_INSTALL_LIBDIR}") +endif() +set(PACKAGE_VERSION "${PROJECT_VERSION}") +configure_file(cglm.pc.in cglm.pc @ONLY) + +install(FILES ${CMAKE_CURRENT_BINARY_DIR}/cglm.pc + DESTINATION ${CMAKE_INSTALL_LIBDIR}/pkgconfig) diff --git a/cglm/CONTRIBUTING.md b/cglm/CONTRIBUTING.md new file mode 100644 index 0000000..f25a6ab --- /dev/null +++ b/cglm/CONTRIBUTING.md @@ -0,0 +1,53 @@ +# CONTRIBUTING + +Any contributions (code, documentation, ...) are welcome. + +# New Features +- This library may not accept all new features, it is better to create an issue and get approval before coding +- You must add test for every new feature +- The feature must be compiled on both UNIX/POSIX systems (e.g. macos, linux...) and Windows + +# Code Style +This library is written with C99, don't try to add C++ files (yes it can compiled into lib), +if you have enough reason to add C++ files than create an issue and get approval before coding, + +- All functions must have `glm` prefix +- Lines should be wrapped at 80 characters. +- Don't invent new style for existing ones +- Use C89 style comments (`/* comments */`) not C++ style comments (`// comments`) +- Don't use TABs instead use 2 spaces for TABs +- All indents must be 2 spaces, not 1 nor 4 space +- All functions in `include` folder must be exported by `CGLM_EXPORT` and wrapped by `extern "C" {` for C++ +- Crate new line for return type, attribs: + +```C +CGLM_INLINE +void +glm_mul(mat4 m1, mat4 m2, mat4 dest) +``` + +not acceptable: + +```C +CGLM_INLINE void glm_mul(mat4 m1, mat4 m2, mat4 dest) +``` +- Variables must be declared at the top of a scope before usage: +```C +int x; +int y; + +x = y = 0; +``` + +not acceptable: + +```C +int x; + +x = 0; +int y = 0; +``` + +- All files must retain same LICENSE statement +- Code with warnings will not be accepted, please suppress them (not by disabling them) +- Run code anaylysis before submitting pull requests, if you use Xcode you can enable Sanitizer in scheme, you can use valgrind in linux diff --git a/cglm/CREDITS b/cglm/CREDITS new file mode 100644 index 0000000..f875d1a --- /dev/null +++ b/cglm/CREDITS @@ -0,0 +1,98 @@ +This library [initially] used some [piece of] implementations +(may include codes) from these open source projects/resources: + +1. Initial Affine Transforms +The original glm repo (g-truc), url: https://github.com/g-truc/glm + +LICENSE[S]: + The Happy Bunny License (Modified MIT License) + The MIT License + Copyright (c) 2005 - 2016 G-Truc Creation + +FULL LICENSE: https://github.com/g-truc/glm/blob/master/copying.txt + +2. Initial Quaternions +Anton's OpenGL 4 Tutorials book source code: + +LICENSE: + OpenGL 4 Example Code. + Accompanies written series "Anton's OpenGL 4 Tutorials" + Email: anton at antongerdelan dot net + First version 27 Jan 2014 + Copyright Dr Anton Gerdelan, Trinity College Dublin, Ireland. + +3. Euler Angles + David Eberly + Geometric Tools, LLC http://www.geometrictools.com/ + Copyright (c) 1998-2016. All Rights Reserved. + + Computing Euler angles from a rotation matrix (euler.pdf) + Gregory G. Slabaugh + +4. Extracting Planes +Fast Extraction of Viewing Frustum Planes from the World-View-Projection Matrix +Authors: + Gil Gribb (ggribb@ravensoft.com) + Klaus Hartmann (k_hartmann@osnabrueck.netsurf.de) + +5. Transform AABB +Transform Axis Aligned Bounding Boxes: +http://dev.theomader.com/transform-bounding-boxes/ +https://github.com/erich666/GraphicsGems/blob/master/gems/TransBox.c + +6. Cull frustum +http://www.txutxi.com/?p=584 +http://old.cescg.org/CESCG-2002/DSykoraJJelinek/ + +7. Quaternions +Initial mat4_quat is borrowed from Apple's simd library + +8. Vector Rotation using Quaternion +https://gamedev.stackexchange.com/questions/28395/rotating-vector3-by-a-quaternion + +9. Sphere AABB intersect +https://github.com/erich666/GraphicsGems/blob/master/gems/BoxSphere.c + +10. Horizontal add +https://stackoverflow.com/questions/6996764/fastest-way-to-do-horizontal-float-vector-sum-on-x86 + +11. de casteljau implementation and comments +https://forums.khronos.org/showthread.php/10264-Animations-in-1-4-1-release-notes-revision-A/page2?highlight=bezier +https://forums.khronos.org/showthread.php/10644-Animation-Bezier-interpolation +https://forums.khronos.org/showthread.php/10387-2D-Tangents-in-Bezier-Splines?p=34164&viewfull=1#post34164 +https://forums.khronos.org/showthread.php/10651-Animation-TCB-Spline-Interpolation-in-COLLADA?highlight=bezier + +12. vec2 cross product +http://allenchou.net/2013/07/cross-product-of-2d-vectors/ + +13. Ray triangle intersect +Möller–Trumbore ray-triangle intersection algorithm, from "Fast, Minimum Storage Ray/Triangle Intersection" +Authors: + Thomas Möller (tompa@clarus.se) + Ben Trumbore (wbt@graphics.cornell.edu) +Link to paper: http://webserver2.tecgraf.puc-rio.br/~mgattass/cg/trbRR/Fast%20MinimumStorage%20RayTriangle%20Intersection.pdf + +14. ARM NEON: Matrix Vector Multiplication +https://stackoverflow.com/a/57793352/2676533 + +16. ARM NEON Div + +http://github.com/microsoft/DirectXMath + +17. Pick Matrix + +glu project -> project.c + +18. Ray sphere intersection + +RAY TRACING GEMS +HIGH-QUALITY AND REAL-TIME RENDERING WITH DXR AND OTHER APIS + +CHAPTER 7 +Precision Improvements for Ray/Sphere Intersection +Eric Haines (1), Johannes Günther (2), and Tomas Akenine-Möller (1) + (1) NVIDIA + (2) Intel + +Wyman, C., and Haines, E. Getting Started with RTX Ray Tracing. +https://github.com/NVIDIAGameWorks/GettingStartedWithRTXRayTracing diff --git a/cglm/LICENSE b/cglm/LICENSE new file mode 100644 index 0000000..c92e559 --- /dev/null +++ b/cglm/LICENSE @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) 2015 Recep Aslantas + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/cglm/Makefile.am b/cglm/Makefile.am new file mode 100644 index 0000000..94510a6 --- /dev/null +++ b/cglm/Makefile.am @@ -0,0 +1,312 @@ +#****************************************************************************** +# Copyright (c), Recep Aslantas. * +# * +# MIT License (MIT), http://opensource.org/licenses/MIT * +# Full license can be found in the LICENSE file * +# * +#****************************************************************************** + +ACLOCAL_AMFLAGS = -I m4 +AM_CFLAGS = -Wall \ + -std=gnu11 \ + -O3 \ + -Wstrict-aliasing=2 \ + -fstrict-aliasing + +lib_LTLIBRARIES = libcglm.la +libcglm_la_LDFLAGS = -no-undefined -version-info 0:1:0 + +checkLDFLAGS = -L./.libs \ + -lm \ + -lcglm +checkCFLAGS = $(AM_CFLAGS) \ + -std=gnu11 \ + -O3 \ + -DCGLM_DEFINE_PRINTS \ + -I./include + +check_PROGRAMS = test/tests +TESTS = $(check_PROGRAMS) + +test_tests_LDFLAGS = $(checkLDFLAGS) +test_tests_CFLAGS = $(checkCFLAGS) + +cglmdir=$(includedir)/cglm +cglm_HEADERS = include/cglm/version.h \ + include/cglm/common.h \ + include/cglm/types.h \ + include/cglm/types-struct.h \ + include/cglm/cglm.h \ + include/cglm/call.h \ + include/cglm/struct.h \ + include/cglm/cam.h \ + include/cglm/io.h \ + include/cglm/mat4.h \ + include/cglm/mat4x2.h \ + include/cglm/mat4x3.h \ + include/cglm/mat3.h \ + include/cglm/mat3x2.h \ + include/cglm/mat3x4.h \ + include/cglm/mat2.h \ + include/cglm/mat2x3.h \ + include/cglm/mat2x4.h \ + include/cglm/affine-pre.h \ + include/cglm/affine-post.h \ + include/cglm/affine.h \ + include/cglm/affine-mat.h \ + include/cglm/vec2.h \ + include/cglm/vec2-ext.h \ + include/cglm/ivec2.h \ + include/cglm/vec3.h \ + include/cglm/vec3-ext.h \ + include/cglm/ivec3.h \ + include/cglm/vec4.h \ + include/cglm/vec4-ext.h \ + include/cglm/ivec4.h \ + include/cglm/euler.h \ + include/cglm/util.h \ + include/cglm/quat.h \ + include/cglm/plane.h \ + include/cglm/noise.h \ + include/cglm/frustum.h \ + include/cglm/box.h \ + include/cglm/aabb2d.h \ + include/cglm/color.h \ + include/cglm/project.h \ + include/cglm/sphere.h \ + include/cglm/ease.h \ + include/cglm/curve.h \ + include/cglm/bezier.h \ + include/cglm/applesimd.h \ + include/cglm/ray.h \ + include/cglm/affine2d.h + +cglm_clipspacedir=$(includedir)/cglm/clipspace +cglm_clipspace_HEADERS = include/cglm/clipspace/persp.h \ + include/cglm/clipspace/persp_lh_no.h \ + include/cglm/clipspace/persp_lh_zo.h \ + include/cglm/clipspace/persp_rh_no.h \ + include/cglm/clipspace/persp_rh_zo.h \ + include/cglm/clipspace/ortho_lh_no.h \ + include/cglm/clipspace/ortho_lh_zo.h \ + include/cglm/clipspace/ortho_rh_no.h \ + include/cglm/clipspace/ortho_rh_zo.h \ + include/cglm/clipspace/view_lh.h \ + include/cglm/clipspace/view_rh.h \ + include/cglm/clipspace/view_lh_no.h \ + include/cglm/clipspace/view_lh_zo.h \ + include/cglm/clipspace/view_rh_no.h \ + include/cglm/clipspace/view_rh_zo.h \ + include/cglm/clipspace/project_no.h \ + include/cglm/clipspace/project_zo.h + +cglm_calldir=$(includedir)/cglm/call +cglm_call_HEADERS = include/cglm/call/mat4.h \ + include/cglm/call/mat4x2.h \ + include/cglm/call/mat4x3.h \ + include/cglm/call/mat3.h \ + include/cglm/call/mat3x2.h \ + include/cglm/call/mat3x4.h \ + include/cglm/call/mat2.h \ + include/cglm/call/mat2x3.h \ + include/cglm/call/mat2x4.h \ + include/cglm/call/vec2.h \ + include/cglm/call/vec3.h \ + include/cglm/call/vec4.h \ + include/cglm/call/ivec2.h \ + include/cglm/call/ivec3.h \ + include/cglm/call/ivec4.h \ + include/cglm/call/io.h \ + include/cglm/call/cam.h \ + include/cglm/call/quat.h \ + include/cglm/call/euler.h \ + include/cglm/call/plane.h \ + include/cglm/call/noise.h \ + include/cglm/call/frustum.h \ + include/cglm/call/box.h \ + include/cglm/call/project.h \ + include/cglm/call/sphere.h \ + include/cglm/call/ease.h \ + include/cglm/call/curve.h \ + include/cglm/call/bezier.h \ + include/cglm/call/ray.h \ + include/cglm/call/affine.h \ + include/cglm/call/affine2d.h \ + include/cglm/call/aabb2d.h + +cglm_call_clipspacedir=$(includedir)/cglm/call/clipspace +cglm_call_clipspace_HEADERS = include/cglm/call/clipspace/persp_lh_no.h \ + include/cglm/call/clipspace/persp_lh_zo.h \ + include/cglm/call/clipspace/persp_rh_no.h \ + include/cglm/call/clipspace/persp_rh_zo.h \ + include/cglm/call/clipspace/ortho_lh_no.h \ + include/cglm/call/clipspace/ortho_lh_zo.h \ + include/cglm/call/clipspace/ortho_rh_no.h \ + include/cglm/call/clipspace/ortho_rh_zo.h \ + include/cglm/call/clipspace/view_lh_no.h \ + include/cglm/call/clipspace/view_lh_zo.h \ + include/cglm/call/clipspace/view_rh_no.h \ + include/cglm/call/clipspace/view_rh_zo.h \ + include/cglm/call/clipspace/project_no.h \ + include/cglm/call/clipspace/project_zo.h + +cglm_simddir=$(includedir)/cglm/simd +cglm_simd_HEADERS = include/cglm/simd/intrin.h \ + include/cglm/simd/x86.h \ + include/cglm/simd/arm.h \ + include/cglm/simd/wasm.h + +cglm_simd_sse2dir=$(includedir)/cglm/simd/sse2 +cglm_simd_sse2_HEADERS = include/cglm/simd/sse2/affine.h \ + include/cglm/simd/sse2/mat4.h \ + include/cglm/simd/sse2/mat3.h \ + include/cglm/simd/sse2/mat2.h \ + include/cglm/simd/sse2/quat.h + +cglm_simd_avxdir=$(includedir)/cglm/simd/avx +cglm_simd_avx_HEADERS = include/cglm/simd/avx/mat4.h \ + include/cglm/simd/avx/affine.h + +cglm_simd_neondir=$(includedir)/cglm/simd/neon +cglm_simd_neon_HEADERS = include/cglm/simd/neon/affine.h \ + include/cglm/simd/neon/mat2.h \ + include/cglm/simd/neon/mat4.h \ + include/cglm/simd/neon/quat.h + +cglm_simd_wasmdir=$(includedir)/cglm/simd/wasm +cglm_simd_wasm_HEADERS = include/cglm/simd/wasm/affine.h \ + include/cglm/simd/wasm/mat2.h \ + include/cglm/simd/wasm/mat3.h \ + include/cglm/simd/wasm/mat4.h \ + include/cglm/simd/wasm/quat.h + +cglm_handeddir=$(includedir)/cglm/handed +cglm_handed_HEADERS = include/cglm/handed/euler_to_quat_lh.h \ + include/cglm/handed/euler_to_quat_rh.h + +cglm_structdir=$(includedir)/cglm/struct +cglm_struct_HEADERS = include/cglm/struct/mat4.h \ + include/cglm/struct/mat4x2.h \ + include/cglm/struct/mat4x3.h \ + include/cglm/struct/mat3.h \ + include/cglm/struct/mat3x2.h \ + include/cglm/struct/mat3x4.h \ + include/cglm/struct/mat2.h \ + include/cglm/struct/mat2x3.h \ + include/cglm/struct/mat2x4.h \ + include/cglm/struct/affine-pre.h \ + include/cglm/struct/affine-post.h \ + include/cglm/struct/affine-mat.h \ + include/cglm/struct/affine.h \ + include/cglm/struct/affine2d.h \ + include/cglm/struct/vec2.h \ + include/cglm/struct/vec2-ext.h \ + include/cglm/struct/ivec2.h \ + include/cglm/struct/vec3.h \ + include/cglm/struct/vec3-ext.h \ + include/cglm/struct/ivec3.h \ + include/cglm/struct/vec4.h \ + include/cglm/struct/vec4-ext.h \ + include/cglm/struct/ivec4.h \ + include/cglm/struct/io.h \ + include/cglm/struct/cam.h \ + include/cglm/struct/quat.h \ + include/cglm/struct/euler.h \ + include/cglm/struct/plane.h \ + include/cglm/struct/noise.h \ + include/cglm/struct/frustum.h \ + include/cglm/struct/box.h \ + include/cglm/struct/aabb2d.h \ + include/cglm/struct/project.h \ + include/cglm/struct/sphere.h \ + include/cglm/struct/color.h \ + include/cglm/struct/curve.h \ + include/cglm/struct/ray.h + +cglm_struct_clipspacedir=$(includedir)/cglm/struct/clipspace +cglm_struct_clipspace_HEADERS = include/cglm/struct/clipspace/persp_lh_no.h \ + include/cglm/struct/clipspace/persp_lh_zo.h \ + include/cglm/struct/clipspace/persp_rh_no.h \ + include/cglm/struct/clipspace/persp_rh_zo.h \ + include/cglm/struct/clipspace/ortho_lh_no.h \ + include/cglm/struct/clipspace/ortho_lh_zo.h \ + include/cglm/struct/clipspace/ortho_rh_no.h \ + include/cglm/struct/clipspace/ortho_rh_zo.h \ + include/cglm/struct/clipspace/view_lh_no.h \ + include/cglm/struct/clipspace/view_lh_zo.h \ + include/cglm/struct/clipspace/view_rh_no.h \ + include/cglm/struct/clipspace/view_rh_zo.h \ + include/cglm/struct/clipspace/project_no.h \ + include/cglm/struct/clipspace/project_zo.h + +cglm_struct_handeddir=$(includedir)/cglm/struct/handed +cglm_struct_handed_HEADERS = include/cglm/struct/handed/euler_to_quat_lh.h \ + include/cglm/struct/handed/euler_to_quat_rh.h + +libcglm_la_SOURCES=\ + src/euler.c \ + src/affine.c \ + src/io.c \ + src/quat.c \ + src/cam.c \ + src/vec2.c \ + src/ivec2.c \ + src/vec3.c \ + src/ivec3.c \ + src/vec4.c \ + src/ivec4.c \ + src/mat2.c \ + src/mat2x3.c \ + src/mat2x4.c \ + src/mat3.c \ + src/mat3x2.c \ + src/mat3x4.c \ + src/mat4.c \ + src/mat4x2.c \ + src/mat4x3.c \ + src/plane.c \ + src/noise.c \ + src/frustum.c \ + src/box.c \ + src/project.c \ + src/sphere.c \ + src/ease.c \ + src/curve.c \ + src/bezier.c \ + src/ray.c \ + src/affine2d.c \ + src/aabb2d.c \ + src/clipspace/ortho_lh_no.c \ + src/clipspace/ortho_lh_zo.c \ + src/clipspace/ortho_rh_no.c \ + src/clipspace/ortho_rh_zo.c \ + src/clipspace/persp_lh_no.c \ + src/clipspace/persp_lh_zo.c \ + src/clipspace/persp_rh_no.c \ + src/clipspace/persp_rh_zo.c \ + src/clipspace/view_lh_no.c \ + src/clipspace/view_lh_zo.c \ + src/clipspace/view_rh_no.c \ + src/clipspace/view_rh_zo.c \ + src/clipspace/project_no.c \ + src/clipspace/project_zo.c + +test_tests_SOURCES=\ + test/runner.c \ + test/src/test_common.c \ + test/src/tests.c \ + test/src/test_clamp.c \ + test/src/test_euler.c \ + test/src/test_bezier.c \ + test/src/test_struct.c + +pkgconfig_DATA=cglm.pc + +# When running configure with --prefix, $VPATH references +# the source directory that post-build.sh is in. When not +# using a prefix, $VPATH will be unset, so we need to fall +# back to using . to run the script. +#export VPATH + +# all-local: +# sh $${VPATH:-.}/post-build.sh diff --git a/cglm/Package.swift b/cglm/Package.swift new file mode 100644 index 0000000..ac0f7e4 --- /dev/null +++ b/cglm/Package.swift @@ -0,0 +1,44 @@ +// swift-tools-version:5.2 + +import PackageDescription + +let package = Package( + name: "cglm", + products: [ + .library(name: "cglm", type: .static, targets: ["cglmHeader"]), + .library(name: "cglmc", targets: ["cglmCompiled"]), + ], + dependencies: [], + targets: [ + .target( + name: "cglmCompiled", + path: "./", + exclude: [ + "./docs", + "./src/swift", + "./include", + "./test", + "./win", + ], + sources: [ + "./src", + ], + publicHeadersPath: "./include" + ), + .target( + name: "cglmHeader", + path: "./", + exclude: [ + "./docs", + "./include", + "./test", + "./win", + ], + sources: [ + "./src/swift", + ], + publicHeadersPath: "./include" + ), + ], + cLanguageStandard: .c11 +) diff --git a/cglm/README.md b/cglm/README.md new file mode 100644 index 0000000..0f02789 --- /dev/null +++ b/cglm/README.md @@ -0,0 +1,211 @@ +# 🎥 OpenGL Mathematics (glm) for `C` + +

+ +

+
+

+ + Build Status + + + Documentation Status + + + Codacy Badge + + + Coverage Status + + + Coverage Status + +

+ + Sponsors on Open Collective + + + Backers on Open Collective + +

+ +
+ +

+A highly optimized 2D|3D math library. Also known as OpenGL Mathematics (glm) for C. cglm provides fast and ergonomic math functions to ease graphics programming. It is community friendly – feel free to report any bugs and issues you face.
+If you're using C++, you might want to check out GLM +

+ + - Allocation-free + - Header-only + - SIMD-optimized + - API-agnostic + +--- + +### 📚 Documentation + +All functions and their parameters are documented above their declaration inside their corresponding headers.
+Alternatively, you can read the complete documentation [here](http://cglm.readthedocs.io). + +### 🔨 Building + +cglm can be used in it's entirety as a header-only library simply by including `cglm/cglm.h`. If you wish to link against it instead, it can be built using one of the supported build systems. Detailed information about building on individual platforms and build systems along with the instructions for building the documentation can be found in [BUILDING.md](./BUILDING.md). + +### ✅ Usage + +#### Header-only + +Include the `cglm/cglm.h` header and use functions with the `glm_` prefix. +```c +#include "cglm/cglm.h" + +// ... + +vec2 vector; +glm_vec2_zero(vector); +``` + +#### Struct API + +Include `cglm/struct.h` and use `glms_`. +```c +#include "cglm/struct.h" + +// ... + +vec2s vector = glms_vec2_zero(); +``` + +#### Linked + +Include `cglm/call.h` and use `glmc_`. +```c +#include "cglm/call.h" + +// ... + +vec2 vector; +glmc_vec2_zero(vector); +``` + +### ❗ Alignment + +While cglm by default aligns what's necessary, it is possible to disable this by defining `CGLM_ALL_UNALIGNED`. If you're targeting machines with any kind of SIMD support, make sure that all `vec4`, `mat4` and `mat2` arguments you pass to cglm functions are aligned to prevent unexpected crashes, alternatively use the unaligned versions if present. + +### Struct API + +The struct API works as follows (note the `s` suffix on types, `glms_` prefix on functions and `GLMS_` on constants): + +```C +#include + +mat4s mat = GLMS_MAT4_IDENTITY_INIT; +mat4s inv = glms_mat4_inv(mat); +``` + +Struct functions generally take parameters *by copy* and *return* the results rather than taking pointers and writing to out parameters. That means your variables can usually be `const`, if you're into that. + +The types used are actually unions that allow access to the same data in multiple ways. One of these involves anonymous structures available since C11. MSVC supports them in earlier versions out of the box and GCC/Clang as well if you enable `-fms-extensions`. +To explicitly enable anonymous structures `#define CGLM_USE_ANONYMOUS_STRUCT 1`, or `0` to disable them. +For backwards compatibility, you can also `#define CGLM_NO_ANONYMOUS_STRUCT` to disable them. If you don't specify explicitly, cglm will attempt a best guess based on your compiler and C version. + +### 📌 Migration notes: + +- `_dup` (duplicate) functions were renamed to `_copy`. For instance: `glm_vec_dup` -> `glm_vec3_copy`. +- OpenGL related functions were dropped to make cglm API independent. +- **[bugfix]** Euler angles had been previously implemented in reverse order (extrinsic). This was fixed to be intrinsic. +- **[major change]** Starting with **v0.4.0**, quaternions are stored as [x, y, z, w]. Previously it was [w, x, y, z]. +- **[api rename]** Starting with **v0.4.5**, `glm_simd_` functions are renamed to `glmm_`. +- **[new option]** Starting with **v0.4.5**, alignment requirements can be disabled. Read more in the documentation. +- **[major change]** Starting with **v0.5.0**, vec3 functions occupy the **glm_vec3_** namespace. This used to be **glm_vec_** in earlier versions. +- **[major change]** Starting with **v0.5.1**, `vec3` and `mat3` types are not aligned by default. +- **[major change]** Starting with **v0.7.3**, inline print functions are disabled by default in release mode to eliminate printing costs (see the Options chapter of the docs).
Colored output can be disabled (see documentation). +- **[major change]** Starting with **v0.8.3**, alternate clipspace configurations are supported. The `CGLM_FORCE_DEPTH_ZERO_TO_ONE` and `CGLM_FORCE_LEFT_HANDED` flags are provided to control clip depth and handedness. This makes it easier to incorporate cglm into projects using graphics APIs such as Vulkan or Metal. See https://cglm.readthedocs.io/en/latest/opt.html#clipspace-option-s + +### 🚀 Features + +- scalar and simd (sse, avx, neon...) optimizations +- general purpose matrix operations (mat4, mat3) +- chain matrix multiplication (square only) +- general purpose vector operations (cross, dot, rotate, proj, angle...) +- affine transformations +- matrix decomposition (extract rotation, scaling factor) +- optimized affine transform matrices (mul, rigid-body inverse) +- camera (lookat) +- projections (ortho, perspective) +- quaternions +- euler angles / yaw-pitch-roll to matrix +- extract euler angles +- frustum (extract view frustum planes, corners...) +- bounding box (AABB in Frustum (culling), crop, merge...) +- bounding sphere +- project, unproject +- easing functions +- curves +- curve interpolation helpers (SMC, deCasteljau...) +- comversion helpers from cglm types to Apple's simd library to pass cglm types to Metal GL without packing them on both sides +- ray intersection helpers +--- + + + + + + + + +
+
Like other graphics libraries (especially OpenGL), cglm uses column-major layout to keep matrices in memory.
+
 
+
While we might support row-major matrices in the future, currently if you need your matrices to be in row-major layout you have to transpose them.
+
+ +
+ +--- + +cglm contains general purpose mat4 product and inverse functions but also provides optimized versions for affine transformations. If you want to multiply two affine transformation matrices you can use glm_mul instead of glm_mat4_mul and glm_inv_tr (ROT + TR) instead glm_mat4_inv. +```C +/* multiplication */ +mat4 modelMat; +glm_mul(T, R, modelMat); + +/* othonormal rot + tr matrix inverse (rigid-body) */ +glm_inv_tr(modelMat); +``` + +## Contributors + +This project exists thanks to all the people who contribute. [[Contribute](CONTRIBUTING.md)] + + + +## Backers + +Thank you to all our backers! 🙏 [[Become a backer](https://opencollective.com/cglm#backer)] + + + + +## Sponsors + +Support this project by becoming a sponsor. Your logo will show up here with a link to your website. [[Become a sponsor](https://opencollective.com/cglm#sponsor)] + + + + + + + + + + + diff --git a/cglm/_config.yml b/cglm/_config.yml new file mode 100644 index 0000000..2f7efbe --- /dev/null +++ b/cglm/_config.yml @@ -0,0 +1 @@ +theme: jekyll-theme-minimal \ No newline at end of file diff --git a/cglm/autogen.sh b/cglm/autogen.sh new file mode 100755 index 0000000..b3b8d82 --- /dev/null +++ b/cglm/autogen.sh @@ -0,0 +1,21 @@ +#! /bin/sh +# +# Copyright (c), Recep Aslantas. +# +# MIT License (MIT), http://opensource.org/licenses/MIT +# Full license can be found in the LICENSE file +# + +cd $(dirname "$0") + +autoheader + +if [ "$(uname)" = "Darwin" ]; then + glibtoolize +else + libtoolize +fi + +aclocal -I m4 +autoconf +automake --add-missing --copy diff --git a/cglm/cglm.pc.in b/cglm/cglm.pc.in new file mode 100644 index 0000000..b80b401 --- /dev/null +++ b/cglm/cglm.pc.in @@ -0,0 +1,11 @@ +prefix=@prefix@ +exec_prefix=@exec_prefix@ +libdir=@libdir@ +includedir=@includedir@ + +Name: @PACKAGE_NAME@ +Description: OpenGL Mathematics (glm) for C +URL: https://github.com/recp/cglm +Version: @PACKAGE_VERSION@ +Cflags: -I${includedir} +Libs: -L${libdir} -lcglm @LIBS@ diff --git a/cglm/cglm.png b/cglm/cglm.png new file mode 100644 index 0000000000000000000000000000000000000000..b450f6aafcd90837f5960fe208e062ffdfc7eeb2 GIT binary patch literal 387622 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 z

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

XtPnl+Q&t61nBJkf->pe>qBb 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 z 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ⅅ#ZIae5x= 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 "MIT", :file => "LICENSE" } + s.author = { "Recep Aslantas" => "recp@acm.org" } + + # Sources + s.source = { :git => "https://github.com/recp/cglm.git", :tag => "v#{s.version}" } + s.source_files = "src", "include/cglm/**/*.h" + s.public_header_files = "include", "include/cglm/**/*.h" + s.exclude_files = "src/win/*", "src/dllmain.c", "src/**/*.h" + s.preserve_paths = "include", "src" + s.header_mappings_dir = "include" + + # Linking + s.library = "m" + + # Configuration + s.pod_target_xcconfig = { + 'CLANG_ENABLE_MODULES' => 'NO', + 'CLANG_ALLOW_NON_MODULAR_INCLUDES_IN_FRAMEWORK_MODULES' => 'YES', + 'CLANG_WARN_DOCUMENTATION_COMMENTS' => 'NO', + 'GCC_C_LANGUAGE_STANDARD' => 'gnu11', + 'GCC_PREPROCESSOR_DEFINITIONS' => '$(inherited) GLM_TESTS_NO_COLORFUL_OUTPUT' + } +end diff --git a/cglm/configure.ac b/cglm/configure.ac new file mode 100644 index 0000000..e0151e8 --- /dev/null +++ b/cglm/configure.ac @@ -0,0 +1,75 @@ +#***************************************************************************** +# Copyright (c), Recep Aslantas. * +# * +# MIT License (MIT), http://opensource.org/licenses/MIT * +# Full license can be found in the LICENSE file * +# * +#***************************************************************************** + +AC_PREREQ([2.69]) +AC_INIT([cglm], [0.9.6], [info@recp.me]) +AM_INIT_AUTOMAKE([-Wall foreign subdir-objects serial-tests]) + +# Don't use the default cflags (-O2 -g), we set ours manually in Makefile.am. +: ${CFLAGS=""} + +AC_CONFIG_MACRO_DIR([m4]) +AC_CONFIG_SRCDIR([src/]) +AC_CONFIG_HEADERS([config.h]) + +# Dependencies for pkg-config. +PKG_PROG_PKG_CONFIG +# Ancient versions of pkg-config (such as the one used in Travis CI) +# don't have this macro, so we need to do it manually. +m4_ifdef([PKG_INSTALLDIR], [ + PKG_INSTALLDIR +], [ + AC_ARG_WITH([pkgconfigdir], + [AS_HELP_STRING([--with-pkgconfigdir], + [pkg-config installation directory ['${libdir}/pkgconfig']])],, + [with_pkgconfigdir=]'${libdir}/pkgconfig') + AC_SUBST([pkgconfigdir], [$with_pkgconfigdir]) +]) + +# Checks for programs. +AC_PROG_CC +AM_PROG_CC_C_O + +AC_PROG_INSTALL +AM_PROG_AR + +AC_ENABLE_SHARED +AC_ENABLE_STATIC + +LT_INIT + +# Checks for libraries. +AC_CHECK_LIB([m], [floor]) + +m4_ifdef([AM_SILENT_RULES], [AM_SILENT_RULES([yes])]) +AC_SYS_LARGEFILE + +# Checks for header files. +AC_CHECK_HEADERS([limits.h \ + stddef.h \ + stdint.h \ + stdlib.h \ + string.h ]) + +# Checks for typedefs, structures, and compiler characteristics. +AC_CHECK_HEADER_STDBOOL +AC_C_INLINE +AC_TYPE_INT32_T +AC_TYPE_INT64_T +AC_TYPE_SIZE_T +AC_TYPE_UINT16_T +AC_TYPE_UINT32_T +AC_TYPE_UINT64_T +AC_TYPE_UINT8_T + +# Checks for library functions. +AC_FUNC_ERROR_AT_LINE + +AC_CONFIG_FILES([Makefile cglm.pc]) + +AC_OUTPUT diff --git a/cglm/docs/make.bat b/cglm/docs/make.bat new file mode 100755 index 0000000..f0aed6a --- /dev/null +++ b/cglm/docs/make.bat @@ -0,0 +1,36 @@ +@ECHO OFF + +pushd %~dp0 + +REM Command file for Sphinx documentation + +if "%SPHINXBUILD%" == "" ( + set SPHINXBUILD=python -msphinx +) +set SOURCEDIR=source +set BUILDDIR=build +set SPHINXPROJ=cglm + +if "%1" == "" goto help + +%SPHINXBUILD% >NUL 2>NUL +if errorlevel 9009 ( + echo. + echo.The Sphinx module was not found. Make sure you have Sphinx installed, + echo.then set the SPHINXBUILD environment variable to point to the full + echo.path of the 'sphinx-build' executable. Alternatively you may add the + echo.Sphinx directory to PATH. + echo. + echo.If you don't have Sphinx installed, grab it from + echo.http://sphinx-doc.org/ + exit /b 1 +) + +%SPHINXBUILD% -M %1 %SOURCEDIR% %BUILDDIR% %SPHINXOPTS% +goto end + +:help +%SPHINXBUILD% -M help %SOURCEDIR% %BUILDDIR% %SPHINXOPTS% + +:end +popd diff --git a/cglm/docs/requirements.txt b/cglm/docs/requirements.txt new file mode 100644 index 0000000..2ca19a4 --- /dev/null +++ b/cglm/docs/requirements.txt @@ -0,0 +1,4 @@ +# Defining the exact version will make sure things don't break +sphinx==7.2.6 +sphinx_rtd_theme==2.0.0 +readthedocs-sphinx-search==0.3.2 diff --git a/cglm/docs/source/aabb2d.rst b/cglm/docs/source/aabb2d.rst new file mode 100644 index 0000000..16d18c6 --- /dev/null +++ b/cglm/docs/source/aabb2d.rst @@ -0,0 +1,198 @@ +.. default-domain:: C + +2d axis aligned bounding box (AABB) +================================================================================ + +Header: cglm/aabb2d.h + +Some convenient functions provided for AABB. + +**Definition of aabb:** + +cglm defines an aabb as a two dimensional array of vec2's. +The first element is the **min** point and the second one is the **max** point. +If you have another type e.g. struct or even another representation then you must +convert it before and after calling a cglm aabb2d function. + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. :c:func:`glm_aabb2d_size` + +Functions: + +1. :c:func:`glm_aabb2d_copy` +#. :c:func:`glm_aabb2d_zero` +#. :c:func:`glm_aabb2d_transform` +#. :c:func:`glm_aabb2d_merge` +#. :c:func:`glm_aabb2d_crop` +#. :c:func:`glm_aabb2d_crop_until` +#. :c:func:`glm_aabb2d_invalidate` +#. :c:func:`glm_aabb2d_isvalid` +#. :c:func:`glm_aabb2d_diag` +#. :c:func:`glm_aabb2d_sizev` +#. :c:func:`glm_aabb2d_radius` +#. :c:func:`glm_aabb2d_center` +#. :c:func:`glm_aabb2d_aabb` +#. :c:func:`glm_aabb2d_circle` +#. :c:func:`glm_aabb2d_point` +#. :c:func:`glm_aabb2d_contains` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_aabb2d_copy(vec2 aabb[2], vec2 dest[2]) + + | copy all members of [aabb] to [dest] + + Parameters: + | *[in]* **aabb** bounding box + | *[out]* **dest** destination + +.. c:function:: void glm_aabb2d_zero(vec2 aabb[2]) + + | makes all members of [aabb] 0.0f (zero) + + Parameters: + | *[in, out]* **aabb** bounding box + +.. c:function:: void glm_aabb2d_transform(vec2 aabb[2], mat3 m, vec2 dest[2]) + + | apply transform to Axis-Aligned Bounding Box + + Parameters: + | *[in]* **aabb** bounding box + | *[in]* **m** transform matrix + | *[out]* **dest** transformed bounding box + +.. c:function:: void glm_aabb2d_merge(vec2 aabb1[2], vec2 aabb2[2], vec2 dest[2]) + + | merges two AABB bounding box and creates new one + + two aabb must be in the same space + + Parameters: + | *[in]* **aabb1** bounding box 1 + | *[in]* **aabb2** bounding box 2 + | *[out]* **dest** merged bounding box + +.. c:function:: void glm_aabb2d_crop(vec2 aabb[2], vec2 cropAabb[2], vec2 dest[2]) + + | crops a bounding box with another one. + + this could be useful for getting a bbox which fits with view frustum and + object bounding boxes. In this case you crop view frustum box with objects + box + + Parameters: + | *[in]* **aabb** bounding box 1 + | *[in]* **cropAabb** crop box + | *[out]* **dest** cropped bounding box + +.. c:function:: void glm_aabb2d_crop_until(vec2 aabb[2], vec2 cropAabb[2], vec2 clampAabb[2], vec2 dest[2]) + + | crops a bounding box with another one. + + this could be useful for getting a bbox which fits with view frustum and + object bounding boxes. In this case you crop view frustum box with objects + box + + Parameters: + | *[in]* **aabb** bounding box + | *[in]* **cropAabb** crop box + | *[in]* **clampAabb** minimum box + | *[out]* **dest** cropped bounding box + +.. c:function:: void glm_aabb2d_invalidate(vec2 aabb[2]) + + | invalidate AABB min and max values + + | It fills *max* values with -FLT_MAX and *min* values with +FLT_MAX + + Parameters: + | *[in, out]* **aabb** bounding box + +.. c:function:: bool glm_aabb2d_isvalid(vec2 aabb[2]) + + | check if AABB is valid or not + + Parameters: + | *[in]* **aabb** bounding box + + Returns: + returns true if aabb is valid otherwise false + +.. c:function:: float glm_aabb2d_diag(vec2 aabb[2]) + + | distance between min and max + + Parameters: + | *[in]* **aabb** bounding box + + Returns: + distance between min - max + + +.. c:function:: void glm_aabb2d_sizev(vec2 aabb[2], vec2 dest) + + | size vector of aabb + + Parameters: + | *[in]* **aabb** bounding box + | *[out]* **dest** size vector + + Returns: + size vector of aabb max - min + +.. c:function:: float glm_aabb2d_radius(vec2 aabb[2]) + + | radius of sphere which surrounds AABB + + Parameters: + | *[in]* **aabb** bounding box + +.. c:function:: void glm_aabb2d_center(vec2 aabb[2], vec2 dest) + + | computes center point of AABB + + Parameters: + | *[in]* **aabb** bounding box + | *[out]* **dest** center of bounding box + +.. c:function:: bool glm_aabb2d_aabb(vec2 aabb[2], vec2 other[2]) + + | check if two AABB intersects + + Parameters: + | *[in]* **aabb** bounding box + | *[out]* **other** other bounding box + +.. c:function:: bool glm_aabb2d_circle(vec2 aabb[2], vec3 c) + + | check if AABB intersects with sphere + + | https://github.com/erich666/GraphicsGems/blob/master/gems/BoxSphere.c + | Solid Box - Solid Sphere test. + + Parameters: + | *[in]* **aabb** solid bounding box + | *[out]* **c** solid circle + +.. c:function:: bool glm_aabb2d_point(vec2 aabb[2], vec2 point) + + | check if point is inside of AABB + + Parameters: + | *[in]* **aabb** bounding box + | *[out]* **point** point + +.. c:function:: bool glm_aabb2d_contains(vec2 aabb[2], vec2 other[2]) + + | check if AABB contains other AABB + + Parameters: + | *[in]* **aabb** bounding box + | *[out]* **other** other bounding box + diff --git a/cglm/docs/source/affine-common.rst b/cglm/docs/source/affine-common.rst new file mode 100644 index 0000000..562e612 --- /dev/null +++ b/cglm/docs/source/affine-common.rst @@ -0,0 +1,129 @@ +.. default-domain:: C + +3D Affine Transforms (common) +================================================================================ + +Common transform functions. + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_translate_make` +#. :c:func:`glm_scale_to` +#. :c:func:`glm_scale_make` +#. :c:func:`glm_scale` +#. :c:func:`glm_scale_uni` +#. :c:func:`glm_rotate_make` +#. :c:func:`glm_rotate_atm` +#. :c:func:`glm_decompose_scalev` +#. :c:func:`glm_uniscaled` +#. :c:func:`glm_decompose_rs` +#. :c:func:`glm_decompose` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_translate_make(mat4 m, vec3 v) + + creates NEW translate transform matrix by *v* vector. + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **v** translate vector [x, y, z] + +.. c:function:: void glm_scale_to(mat4 m, vec3 v, mat4 dest) + + scale existing transform matrix by *v* vector and store result in dest + + Parameters: + | *[in]* **m** affine transform + | *[in]* **v** scale vector [x, y, z] + | *[out]* **dest** scaled matrix + +.. c:function:: void glm_scale_make(mat4 m, vec3 v) + + creates NEW scale matrix by v vector + + Parameters: + | *[out]* **m** affine transform + | *[in]* **v** scale vector [x, y, z] + +.. c:function:: void glm_scale(mat4 m, vec3 v) + + scales existing transform matrix by v vector + and stores result in same matrix + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **v** scale vector [x, y, z] + +.. c:function:: void glm_scale_uni(mat4 m, float s) + + applies uniform scale to existing transform matrix v = [s, s, s] + and stores result in same matrix + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **v** scale factor + +.. c:function:: void glm_rotate_make(mat4 m, float angle, vec3 axis) + + creates NEW rotation matrix by angle and axis, + axis will be normalized so you don't need to normalize it + + Parameters: + | *[out]* **m** affine transform + | *[in]* **axis** angle (radians) + | *[in]* **axis** axis + +.. c:function:: void glm_rotate_atm(mat4 m, vec3 pivot, float angle, vec3 axis) + + | creates NEW rotation matrix by angle and axis at given point + | this creates rotation matrix, it assumes you don't have a matrix + + | this should work faster than glm_rotate_at because it reduces one glm_translate. + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **pivot** pivot, anchor point, rotation center + | *[in]* **angle** angle (radians) + | *[in]* **axis** axis + +.. c:function:: void glm_decompose_scalev(mat4 m, vec3 s) + + decompose scale vector + + Parameters: + | *[in]* **m** affine transform + | *[out]* **s** scale vector (Sx, Sy, Sz) + +.. c:function:: bool glm_uniscaled(mat4 m) + + returns true if matrix is uniform scaled. + This is helpful for creating normal matrix. + + Parameters: + | *[in]* **m** matrix + +.. c:function:: void glm_decompose_rs(mat4 m, mat4 r, vec3 s) + + decompose rotation matrix (mat4) and scale vector [Sx, Sy, Sz] + DON'T pass projected matrix here + + Parameters: + | *[in]* **m** affine transform + | *[out]* **r** rotation matrix + | *[out]* **s** scale matrix + +.. c:function:: void glm_decompose(mat4 m, vec4 t, mat4 r, vec3 s) + + decompose affine transform, TODO: extract shear factors. + DON'T pass projected matrix here + + Parameters: + | *[in]* **m** affine transform + | *[out]* **t** translation vector + | *[out]* **r** rotation matrix (mat4) + | *[out]* **s** scaling vector [X, Y, Z] diff --git a/cglm/docs/source/affine-mat.rst b/cglm/docs/source/affine-mat.rst new file mode 100644 index 0000000..7c0721c --- /dev/null +++ b/cglm/docs/source/affine-mat.rst @@ -0,0 +1,99 @@ +.. default-domain:: C + +3D Affine Transform Matrix (specialized functions) +================================================================================ + +Header: cglm/affine-mat.h + +We mostly use glm_mat4_* for 4x4 general and transform matrices. **cglm** +provides optimized version of some functions. Because affine transform matrix is +a known format, for instance all last item of first three columns is zero. + +You should be careful when using these functions. For instance :c:func:`glm_mul` +assumes matrix will be this format: + +.. code-block:: text + + R R R X + R R R Y + R R R Z + 0 0 0 W + +if you override zero values here then use :c:func:`glm_mat4_mul` version. +You cannot use :c:func:`glm_mul` anymore. + +Same is also true for :c:func:`glm_inv_tr` if you only have rotation and +translation then it will work as expected, otherwise you cannot use that. + +In the future it may accept scale factors too but currently it does not. + +Table of contents (click func go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_mul` +#. :c:func:`glm_mul_rot` +#. :c:func:`glm_inv_tr` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_mul(mat4 m1, mat4 m2, mat4 dest) + + | this is similar to glm_mat4_mul but specialized to affine transform + + Matrix format should be: + + .. code-block:: text + + R R R X + R R R Y + R R R Z + 0 0 0 W + + this reduces some multiplications. It should be faster than mat4_mul. + if you are not sure about matrix format then DON'T use this! use mat4_mul + + Parameters: + | *[in]* **m1** affine matrix 1 + | *[in]* **m2** affine matrix 2 + | *[out]* **dest** result matrix + +.. c:function:: void glm_mul_rot(mat4 m1, mat4 m2, mat4 dest) + + | this is similar to glm_mat4_mul but specialized to rotation matrix + + Right Matrix format should be (left is free): + + .. code-block:: text + + R R R 0 + R R R 0 + R R R 0 + 0 0 0 1 + + this reduces some multiplications. It should be faster than mat4_mul. + if you are not sure about matrix format then DON'T use this! use mat4_mul + + Parameters: + | *[in]* **m1** affine matrix 1 + | *[in]* **m2** affine matrix 2 + | *[out]* **dest** result matrix + +.. c:function:: void glm_inv_tr(mat4 mat) + + | inverse orthonormal rotation + translation matrix (ridig-body) + + .. code-block:: text + + X = | R T | X' = | R' -R'T | + | 0 1 | | 0 1 | + + use this if you only have rotation + translation, this should work faster + than :c:func:`glm_mat4_inv` + + Don't use this if your matrix includes other things e.g. scale, shear... + + Parameters: + | *[in,out]* **mat** affine matrix diff --git a/cglm/docs/source/affine-post.rst b/cglm/docs/source/affine-post.rst new file mode 100644 index 0000000..f298e79 --- /dev/null +++ b/cglm/docs/source/affine-post.rst @@ -0,0 +1,127 @@ +.. default-domain:: C + +3D Affine Transforms (post) +================================================================================ + +Post transform functions are similar to pre transform functions except order of application is reversed. +Post transform functions are applied after the object is transformed with given (model matrix) transform. + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_translated_to` +#. :c:func:`glm_translated` +#. :c:func:`glm_translated_x` +#. :c:func:`glm_translated_y` +#. :c:func:`glm_translated_z` +#. :c:func:`glm_rotated_x` +#. :c:func:`glm_rotated_y` +#. :c:func:`glm_rotated_z` +#. :c:func:`glm_rotated` +#. :c:func:`glm_rotated_at` +#. :c:func:`glm_spinned` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_translated_to(mat4 m, vec3 v, mat4 dest) + + translate existing transform matrix by *v* vector and store result in dest + + Parameters: + | *[in]* **m** affine transform + | *[in]* **v** translate vector [x, y, z] + | *[out]* **dest** translated matrix + +.. c:function:: void glm_translated(mat4 m, vec3 v) + + translate existing transform matrix by *v* vector + and stores result in same matrix + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **v** translate vector [x, y, z] + +.. c:function:: void glm_translated_x(mat4 m, float x) + + translate existing transform matrix by x factor + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **v** x factor + +.. c:function:: void glm_translated_y(mat4 m, float y) + + translate existing transform matrix by *y* factor + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **v** y factor + +.. c:function:: void glm_translated_z(mat4 m, float z) + + translate existing transform matrix by *z* factor + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **v** z factor + +.. c:function:: void glm_rotated_x(mat4 m, float angle, mat4 dest) + + rotate existing transform matrix around X axis by angle + and store result in dest + + Parameters: + | *[in]* **m** affine transform + | *[in]* **angle** angle (radians) + | *[out]* **dest** rotated matrix + +.. c:function:: void glm_rotated_y(mat4 m, float angle, mat4 dest) + + rotate existing transform matrix around Y axis by angle + and store result in dest + + Parameters: + | *[in]* **m** affine transform + | *[in]* **angle** angle (radians) + | *[out]* **dest** rotated matrix + +.. c:function:: void glm_rotated_z(mat4 m, float angle, mat4 dest) + + rotate existing transform matrix around Z axis by angle + and store result in dest + + Parameters: + | *[in]* **m** affine transform + | *[in]* **angle** angle (radians) + | *[out]* **dest** rotated matrix + +.. c:function:: void glm_rotated(mat4 m, float angle, vec3 axis) + + rotate existing transform matrix around Z axis by angle and axis + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **angle** angle (radians) + | *[in]* **axis** axis + +.. c:function:: void glm_rotated_at(mat4 m, vec3 pivot, float angle, vec3 axis) + + rotate existing transform around given axis by angle at given pivot point (rotation center) + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **pivot** pivot, anchor point, rotation center + | *[in]* **angle** angle (radians) + | *[in]* **axis** axis + +.. c:function:: void glm_spinned(mat4 m, float angle, vec3 axis) + + | rotate existing transform matrix around given axis by angle around self (doesn't affected by position) + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **angle** angle (radians) + | *[in]* **axis** axis diff --git a/cglm/docs/source/affine-pre.rst b/cglm/docs/source/affine-pre.rst new file mode 100644 index 0000000..c313fc9 --- /dev/null +++ b/cglm/docs/source/affine-pre.rst @@ -0,0 +1,154 @@ +.. default-domain:: C + +3D Affine Transforms (pre) +================================================================================ + +Pre transform functions which are regular transform functions. + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_translate_to` +#. :c:func:`glm_translate` +#. :c:func:`glm_translate_x` +#. :c:func:`glm_translate_y` +#. :c:func:`glm_translate_z` +#. :c:func:`glm_translate_make` +#. :c:func:`glm_scale_to` +#. :c:func:`glm_scale_make` +#. :c:func:`glm_scale` +#. :c:func:`glm_scale_uni` +#. :c:func:`glm_rotate_x` +#. :c:func:`glm_rotate_y` +#. :c:func:`glm_rotate_z` +#. :c:func:`glm_rotate_make` +#. :c:func:`glm_rotate` +#. :c:func:`glm_rotate_at` +#. :c:func:`glm_rotate_atm` +#. :c:func:`glm_decompose_scalev` +#. :c:func:`glm_uniscaled` +#. :c:func:`glm_decompose_rs` +#. :c:func:`glm_decompose` +#. :c:func:`glm_spin` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_translate_to(mat4 m, vec3 v, mat4 dest) + + translate existing transform matrix by *v* vector and store result in dest + + Parameters: + | *[in]* **m** affine transform + | *[in]* **v** translate vector [x, y, z] + | *[out]* **dest** translated matrix + +.. c:function:: void glm_translate(mat4 m, vec3 v) + + translate existing transform matrix by *v* vector + and stores result in same matrix + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **v** translate vector [x, y, z] + +.. c:function:: void glm_translate_x(mat4 m, float x) + + translate existing transform matrix by x factor + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **v** x factor + +.. c:function:: void glm_translate_y(mat4 m, float y) + + translate existing transform matrix by *y* factor + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **v** y factor + +.. c:function:: void glm_translate_z(mat4 m, float z) + + translate existing transform matrix by *z* factor + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **v** z factor + +.. c:function:: void glm_rotate_x(mat4 m, float angle, mat4 dest) + + rotate existing transform matrix around X axis by angle + and store result in dest + + Parameters: + | *[in]* **m** affine transform + | *[in]* **angle** angle (radians) + | *[out]* **dest** rotated matrix + +.. c:function:: void glm_rotate_y(mat4 m, float angle, mat4 dest) + + rotate existing transform matrix around Y axis by angle + and store result in dest + + Parameters: + | *[in]* **m** affine transform + | *[in]* **angle** angle (radians) + | *[out]* **dest** rotated matrix + +.. c:function:: void glm_rotate_z(mat4 m, float angle, mat4 dest) + + rotate existing transform matrix around Z axis by angle + and store result in dest + + Parameters: + | *[in]* **m** affine transform + | *[in]* **angle** angle (radians) + | *[out]* **dest** rotated matrix + +.. c:function:: void glm_rotate(mat4 m, float angle, vec3 axis) + + rotate existing transform matrix around given axis by angle at ORIGIN (0,0,0) + + **❗️IMPORTANT ❗️** + + If you need to rotate object around itself e.g. center of object or at + some point [of object] then `glm_rotate_at()` would be better choice to do so. + + Even if object's model transform is identity, rotation may not be around + center of object if object does not lay out at ORIGIN perfectly. + + Using `glm_rotate_at()` with center of bounding shape ( AABB, Sphere ... ) + would be an easy option to rotate around object if object is not at origin. + + One another option to rotate around itself at any point is `glm_spin()` + which is perfect if only rotating around model position is desired e.g. not + specific point on model for instance center of geometry or center of mass, + again if geometry is not perfectly centered at origin at identity transform, + rotation may not be around geometry. + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **angle** angle (radians) + | *[in]* **axis** axis + +.. c:function:: void glm_rotate_at(mat4 m, vec3 pivot, float angle, vec3 axis) + + rotate existing transform around given axis by angle at given pivot point (rotation center) + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **pivot** pivot, anchor point, rotation center + | *[in]* **angle** angle (radians) + | *[in]* **axis** axis + +.. c:function:: void glm_spin(mat4 m, float angle, vec3 axis) + + | rotate existing transform matrix around given axis by angle around self (doesn't affected by position) + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **angle** angle (radians) + | *[in]* **axis** axis diff --git a/cglm/docs/source/affine.rst b/cglm/docs/source/affine.rst new file mode 100644 index 0000000..ed08ece --- /dev/null +++ b/cglm/docs/source/affine.rst @@ -0,0 +1,196 @@ +.. default-domain:: C + +3D Affine Transforms +================================================================================ + +Header: cglm/affine.h + +Before starting, **cglm** provides two kind of transform functions; pre and post. + +Pre functions (`T' = Tnew * T`) are like `glm_translate`, `glm_rotate` which means it will translate the vector first and then apply the model transformation. +Post functions (`T' = T * Tnew`) are like `glm_translated`, `glm_rotated` which means it will apply the model transformation first and then translate the vector. + +`glm_translate`, `glm_rotate` are pre functions and are similar to C++ **glm** which you are familiar with. + +In new versions of **cglm** we added `glm_translated`, `glm_rotated`... which are post functions, +they are useful in some cases, e.g. append transform to existing transform (apply/append transform as last transform T' = T * Tnew). + +Post functions are named after pre functions with `ed` suffix, e.g. `glm_translate` -> `glm_translated`. So don't mix them up. + +Initialize Transform Matrices +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ +Functions with **_make** prefix expect you don't have a matrix and they create +a matrix for you. You don't need to pass identity matrix. + +But other functions expect you have a matrix and you want to transform them. If +you didn't have any existing matrix you have to initialize matrix to identity +before sending to transform functions. + +There are also functions to decompose transform matrix. These functions can't +decompose matrix after projected. + +Rotation Center +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Rotating functions uses origin as rotation center (pivot/anchor point), +since scale factors are stored in rotation matrix, same may also true for scalling. +cglm provides some functions for rotating around at given point e.g. +**glm_rotate_at**, **glm_quat_rotate_at**. Use them or follow next section for algorithm ("Rotate or Scale around specific Point (Pivot Point / Anchor Point)"). + +Also **cglm** provides :c:func:`glm_spin` and :c:func:`glm_spinned` functions to rotate around itself. No need to give pivot. +These functions are useful for rotating around center of object. + +Rotate or Scale around specific Point (Anchor Point) +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +If you want to rotate model around arbitrary point follow these steps: + +1. Move model from pivot point to origin: **translate(-pivot.x, -pivot.y, -pivot.z)** +2. Apply rotation (or scaling maybe) +3. Move model back from origin to pivot (reverse of step-1): **translate(pivot.x, pivot.y, pivot.z)** + +**glm_rotate_at**, **glm_quat_rotate_at** and their helper functions works that way. +So if you use them you don't need to do these steps manually which are done by **cglm**. + +The implementation would be: + +.. code-block:: c + :linenos: + + glm_translate(m, pivot); + glm_rotate(m, angle, axis); + glm_translate(m, pivotInv); /* pivotInv = -pivot */ + +or just: + +.. code-block:: c + :linenos: + + glm_rotate_at(m, pivot, angle, axis); + +.. _TransformsOrder: + +Transforms Order +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +It is important to understand this part especially if you call transform +functions multiple times + +`glm_translate`, `glm_rotate`, `glm_scale` and `glm_quat_rotate` and their +helpers functions works like this (cglm provides reverse order as `ed` suffix e.g `glm_translated`, `glm_rotated` see post transforms): + +.. code-block:: c + :linenos: + + TransformMatrix = TransformMatrix * TranslateMatrix; // glm_translate() + TransformMatrix = TransformMatrix * RotateMatrix; // glm_rotate(), glm_quat_rotate() + TransformMatrix = TransformMatrix * ScaleMatrix; // glm_scale() + +As you can see it is multiplied as right matrix. For instance what will happen if you call `glm_translate` twice? + +.. code-block:: c + :linenos: + + glm_translate(transform, translate1); /* transform = transform * translate1 */ + glm_translate(transform, translate2); /* transform = transform * translate2 */ + glm_rotate(transform, angle, axis) /* transform = transform * rotation */ + +Now lets try to understand this: + +1. You call translate using `translate1` and you expect it will be first transform +because you call it first, do you? + +Result will be **`transform = transform * translate1`** + +2. Then you call translate using `translate2` and you expect it will be second transform? + +Result will be **`transform = transform * translate2`**. Now lets expand transform, +it was `transform * translate1` before second call. + +Now it is **`transform = transform * translate1 * translate2`**, now do you understand what I say? + +3. After last call transform will be: + +**`transform = transform * translate1 * translate2 * rotation`** + +The order will be; **rotation will be applied first**, then **translate2** then **translate1** + +It is all about matrix multiplication order. It is similar to MVP matrix: +`MVP = Projection * View * Model`, model will be applied first, then view then projection. + +**Confused?** + +In the end the last function call applied first in shaders. + +As alternative way, you can create transform matrices individually then combine manually, +but don't forget that `glm_translate`, `glm_rotate`, `glm_scale`... are optimized and should be faster (an smaller assembly output) than manual multiplication + +.. code-block:: c + :linenos: + + mat4 transform1, transform2, transform3, finalTransform; + + glm_translate_make(transform1, translate1); + glm_translate_make(transform2, translate2); + glm_rotate_make(transform3, angle, axis); + + /* first apply transform1, then transform2, thentransform3 */ + glm_mat4_mulN((mat4 *[]){&transform3, &transform2, &transform1}, 3, finalTransform); + + /* if you don't want to use mulN, same as above */ + glm_mat4_mul(transform3, transform2, finalTransform); + glm_mat4_mul(finalTransform, transform1, finalTransform); + +Now transform1 will be applied first, then transform2 then transform3 + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_translate_to` +#. :c:func:`glm_translate` +#. :c:func:`glm_translate_x` +#. :c:func:`glm_translate_y` +#. :c:func:`glm_translate_z` +#. :c:func:`glm_translate_make` +#. :c:func:`glm_scale_to` +#. :c:func:`glm_scale_make` +#. :c:func:`glm_scale` +#. :c:func:`glm_scale_uni` +#. :c:func:`glm_rotate_x` +#. :c:func:`glm_rotate_y` +#. :c:func:`glm_rotate_z` +#. :c:func:`glm_rotate_make` +#. :c:func:`glm_rotate` +#. :c:func:`glm_rotate_at` +#. :c:func:`glm_rotate_atm` +#. :c:func:`glm_decompose_scalev` +#. :c:func:`glm_uniscaled` +#. :c:func:`glm_decompose_rs` +#. :c:func:`glm_decompose` + +Post functions (**NEW**): + +1. :c:func:`glm_translated_to` +#. :c:func:`glm_translated` +#. :c:func:`glm_translated_x` +#. :c:func:`glm_translated_y` +#. :c:func:`glm_translated_z` +#. :c:func:`glm_rotated_x` +#. :c:func:`glm_rotated_y` +#. :c:func:`glm_rotated_z` +#. :c:func:`glm_rotated` +#. :c:func:`glm_rotated_at` +#. :c:func:`glm_spinned` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. toctree:: + :maxdepth: 1 + :caption: Affine categories: + + affine-common + affine-pre + affine-post diff --git a/cglm/docs/source/affine2d.rst b/cglm/docs/source/affine2d.rst new file mode 100644 index 0000000..7f72aa2 --- /dev/null +++ b/cglm/docs/source/affine2d.rst @@ -0,0 +1,140 @@ +.. default-domain:: C + +2D Affine Transforms +================================================================================ + +Header: cglm/affine2d.h + +2D Transforms uses `2d` suffix for naming. If there is no 2D suffix it is 3D function. + +Initialize Transform Matrices +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ +Functions with **_make** prefix expect you don't have a matrix and they create +a matrix for you. You don't need to pass identity matrix. + +But other functions expect you have a matrix and you want to transform them. If +you didn't have any existing matrix you have to initialize matrix to identity +before sending to transform functions. + +Transforms Order +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +See :ref:`TransformsOrder` to read similar section. + + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_translate2d` +#. :c:func:`glm_translate2d_to` +#. :c:func:`glm_translate2d_x` +#. :c:func:`glm_translate2d_y` +#. :c:func:`glm_translate2d_make` +#. :c:func:`glm_scale2d_to` +#. :c:func:`glm_scale2d_make` +#. :c:func:`glm_scale2d` +#. :c:func:`glm_scale2d_uni` +#. :c:func:`glm_rotate2d_make` +#. :c:func:`glm_rotate2d` +#. :c:func:`glm_rotate2d_to` + +.. c:function:: void glm_translate2d(mat3 m, vec2 v) + + translate existing 2d transform matrix by *v* vector and stores result in same matrix + + Parameters: + | *[in, out]* **m** 2d affine transform + | *[in]* **v** translate vector [x, y] + +.. c:function:: void glm_translate2d_to(mat3 m, vec2 v, mat3 dest) + + translate existing 2d transform matrix by *v* vector and store result in dest + + Parameters: + | *[in]* **m** 2d affine transform + | *[in]* **v** translate vector [x, y] + | *[out]* **dest** translated matrix + +.. c:function:: void glm_translate2d_x(mat3 m, float x) + + translate existing 2d transform matrix by x factor + + Parameters: + | *[in, out]* **m** 2d affine transform + | *[in]* **x** x factor + +.. c:function:: void glm_translate2d_y(mat3 m, float y) + + translate existing 2d transform matrix by y factor + + Parameters: + | *[in, out]* **m** 2d affine transform + | *[in]* **y** y factor + +.. c:function:: void glm_translate2d_make(mat3 m, vec2 v) + + creates NEW translate 2d transform matrix by *v* vector + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **v** translate vector [x, y] + +.. c:function:: void glm_scale2d_to(mat3 m, vec2 v, mat3 dest) + + scale existing 2d transform matrix by *v* vector and store result in dest + + Parameters: + | *[in]* **m** affine transform + | *[in]* **v** scale vector [x, y] + | *[out]* **dest** scaled matrix + +.. c:function:: void glm_scale2d_make(mat3 m, vec2 v) + + creates NEW 2d scale matrix by *v* vector + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **v** scale vector [x, y] + +.. c:function:: void glm_scale2d(mat3 m, vec2 v) + + scales existing 2d transform matrix by *v* vector and stores result in same matrix + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **v** translate vector [x, y] + +.. c:function:: void glm_scale2d_uni(mat3 m, float s) + + applies uniform scale to existing 2d transform matrix v = [s, s] and stores result in same matrix + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **s** scale factor + +.. c:function:: void glm_rotate2d_make(mat3 m, float angle) + + creates NEW rotation matrix by angle around *Z* axis + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **angle** angle (radians) + +.. c:function:: void glm_rotate2d(mat3 m, float angle) + + rotate existing 2d transform matrix around *Z* axis by angle and store result in same matrix + + Parameters: + | *[in, out]* **m** affine transform + | *[in]* **angle** angle (radians) + +.. c:function:: void glm_rotate2d_to(mat3 m, float angle, mat3 dest) + + rotate existing 2d transform matrix around *Z* axis by angle and store result in dest + + Parameters: + | *[in]* **m** affine transform + | *[in]* **angle** angle (radians) + | *[out]* **dest** rotated matrix \ No newline at end of file diff --git a/cglm/docs/source/api.rst b/cglm/docs/source/api.rst new file mode 100644 index 0000000..4712030 --- /dev/null +++ b/cglm/docs/source/api.rst @@ -0,0 +1,28 @@ +📚 API documentation +================================ + +**cglm** provides a few APIs for similar functions. + +* 📦 **Inline API**: All functions are inline. You can include **cglm/cglm.h** header + to use this API. This is the default API. `glm_` is namespace/prefix for this API. +* 📦 **Call API**: All functions are not inline. You need to build *cglm* and link it + to your project. You can include **cglm/call.h** header to use this API. `glmc_` is namespace/prefix for this API. + +And also there are also sub categories: + +* 📦 **Array API**: Types are raw arrays and functions takes array as argument. You can include **cglm/cglm.h** header + to use this API. This is the default API. `glm_` is namespace/prefix for this API. +* 📦 **Struct API**: Types are union/struct and functions takes struct as argument and return structs if needed. You can include **cglm/struct.h** header + to use this API. This also includes **cglm/cglm.h** header.`glms_` is namespace/prefix for this API but your can omit or change it, see struct api docs. +* 📦 **SIMD API**: SIMD functions and helpers. `glmm_` is namespace/prefix for this API. + +📌 Since struct api and call api are built top of inline array api, follow inline array api docs for individual functions. + +.. toctree:: + :maxdepth: 1 + :caption: API documentations: + + api_inline_array + api_struct + api_call + api_simd diff --git a/cglm/docs/source/api_call.rst b/cglm/docs/source/api_call.rst new file mode 100644 index 0000000..9e8fff3 --- /dev/null +++ b/cglm/docs/source/api_call.rst @@ -0,0 +1,11 @@ +Call API +================================ + +Call API is pre-built API for making calls from library. It is built on top of the array api. **glmc_** is the namespace for the call api. +**c** stands for call. + +You need to built cglm to use call api. See build instructions (:doc:`build`) for more details. + +The functions except namespace **glmc_** are same as inline api. See ( :doc:`api_inline_array` ) for more details. + +📌 In the future we can define option to forward inline functions or struct api to call api. \ No newline at end of file diff --git a/cglm/docs/source/api_inline_array.rst b/cglm/docs/source/api_inline_array.rst new file mode 100644 index 0000000..f989bfe --- /dev/null +++ b/cglm/docs/source/api_inline_array.rst @@ -0,0 +1,78 @@ +Array API - Inline (Default) +======================================== + +This is the default API of *cglm*. All functions are forced to be inlined +and struct api, call api uses this inline api to share implementation. + +📌 Call api is also array api but it is not inlined. +In the future there may be option to forward struct api to call api instead of inline api to reduce binary size if needed. + +📌 **USE this API docs for similar functions in struct and call api** + +📌 In struct api you can omit namespace e.g :code:`glms_vec3_dot` can be called as :code:`vec3_dot` in struct api, see :doc:`api_struct` to configure struct api for more details. +📌 In struct api functions can return struct/union +📌 In struct api you can access items like **.x**, **.y**, **.z**, **.w**, **.r**, **.g**, **.b**, **.a**, **.m00**, **m01**... + +Some functions may exist twice, once for their namespace and once for global namespace +to make easier to write very common functions + +For instance, in general we use :code:`glm_vec3_dot` to get dot product +of two **vec3**. Now we can also do this with :code:`glm_dot`, +same for *_cross* and so on... + +The original function stays where it is, the function in global namespace +of same name is just an alias, so there is no call version of those functions. +e.g there is no func like :code:`glmc_dot` because *glm_dot* is just alias for +:code:`glm_vec3_dot` + +By including **cglm/cglm.h** header you will include all inline version +of functions. Since functions in this header[s] are inline you don't need to +build or link *cglm* against your project. + +But by including **cglm/call.h** header you will include all *non-inline* +version of functions. You need to build *cglm* and link it. +Follow the :doc:`build` documentation for this + +.. toctree:: + :maxdepth: 1 + :caption: API categories: + + affine + affine-mat + affine2d + cam + frustum + box + aabb2d + quat + euler + mat2 + mat2x3 + mat2x4 + mat3 + mat3x2 + mat3x4 + mat4 + mat4x2 + mat4x3 + vec2 + vec2-ext + vec3 + vec3-ext + vec4 + vec4-ext + ivec2 + ivec3 + ivec4 + color + plane + noise + project + util + io + call + sphere + curve + bezier + version + ray diff --git a/cglm/docs/source/api_simd.rst b/cglm/docs/source/api_simd.rst new file mode 100644 index 0000000..3a5024d --- /dev/null +++ b/cglm/docs/source/api_simd.rst @@ -0,0 +1,12 @@ +SIMD API +================================ + +SIMD api is special api for SIMD operations. **glmm_** prefix is used for SIMD operations in cglm. It is used in many places in cglm. +You can use it for your own SIMD operations too. In the future the api may be extended by time. + +Supported SIMD architectures ( may vary by time ) + +* SSE / SSE2 +* AVX +* NEON +* WASM 128 diff --git a/cglm/docs/source/api_struct.rst b/cglm/docs/source/api_struct.rst new file mode 100644 index 0000000..f3e4e78 --- /dev/null +++ b/cglm/docs/source/api_struct.rst @@ -0,0 +1,101 @@ +Struct API +================================ + +Struct API is alternative API to array api to use **cglm** with improved type safety and easy to use. +Since struct api is built top of array api, every struct API is not documented here. +See array api documentation for more information for individual functions. + +By default struct api adds `s` suffix to every type name e.g. vec3s, mat4s, versors etc. +Also struct api `s` suffix to namespace e.g. `glms_vec3_add`, `glms_mat4_mul` etc. + +By starting v0.9.0, struct api namespace is configurable. We can omit **glms_** namespace or +even change it with custom name to move existing api integrations to **cglm** more easily... +We can also add **s** to function names if we want e.g. `glms_vec3_add()` -> `vec3_add()` or `vec3s_add()`. + +By including **cglm/struct.h** header you will include all struct api. It will also include **cglm/cglm.h** too. +Since struct apis are inline you don't need to build or link *cglm* against +your project unless if you want to use pre-built call-api too. + +Struct API is built top of array api. So you can mix them. +Use **.raw** union member to access raw array data to use it with array api. + +Unlike array api ([0], [1], [0][0] ...), it is also possible to use struct api +with **.x**, **.y**, **.z**, **.w**, **.r**, **.g**, **.b**, **.a**, **.m00**, **m01**... +accessors to access individual elements/properties of vectors and matrices. + +Struct API usage: +----------------- + +.. code-block:: c + + #include + + mat4s m1 = glms_mat4_identity(); /* init... */ + mat4s m2 = glms_mat4_identity(); /* init... */ + mat4s m3 = glms_mat4_mul(glms_mat4_mul(m1, m2), glms_mat4_mul(m3, m4)); + + vec3s v1 = { 1.0f, 0.0f, 0.0f }; + vec3s v2 = { 0.0f, 1.0f, 0.0f }; + vec4s v4 = { 0.0f, 1.0f, 0.0f, 0.0f }; + vec4 v5a = { 0.0f, 1.0f, 0.0f, 0.0f }; + + mat4s m4 = glms_rotate(m3, M_PI_2, + glms_vec3_cross(glms_vec3_add(v1, v6) + glms_vec3_add(v1, v7))); + + v1.x = 1.0f; v1.y = 0.0f; v1.z = 0.0f; + // or + v1.raw[0] = 1.0f; v1.raw[1] = 0.0f; v1.raw[2] = 0.0f; + + /* use struct api with array api (mix them). */ + /* use .raw to access array and use it with array api */ + + glm_vec4_add(m4.col[0].raw, v5a, m4.col[0].raw); + glm_mat4_mulv(m4.raw, v4.raw, v5a); + +or omit `glms_` namespace completely (see options below): + +.. code-block:: c + + #define CGLM_OMIT_NS_FROM_STRUCT_API + + #include + + mat4s m1 = mat4_identity(); /* init... */ + mat4s m2 = mat4_identity(); /* init... */ + mat4s m3 = mat4_mul(mat4_mul(m1, m2), mat4_mul(m3, m4)); + + vec3s v1 = { 1.0f, 0.0f, 0.0f }; + vec3s v2 = { 0.0f, 1.0f, 0.0f }; + vec4s v4 = { 0.0f, 1.0f, 0.0f, 0.0f }; + vec4 v5a = { 0.0f, 1.0f, 0.0f, 0.0f }; + + mat4s m4 = glms_rotate(m3, M_PI_2, + vec3_cross(vec3_add(v1, v6) + vec3_add(v1, v7))); + + v1.x = 1.0f; v1.y = 0.0f; v1.z = 0.0f; + // or + v1.raw[0] = 1.0f; v1.raw[1] = 0.0f; v1.raw[2] = 0.0f; + + /* use struct api with array api (mix them) */ + glm_vec4_add(m4.col[0].raw, v5a, m4.col[0].raw); + glm_mat4_mulv(m4.raw, v4.raw, v5a); + +Configuring the Struct API: +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +To configure the Struct API namespace, you can define the following macros before including the cglm/struct.h header: + +- **CGLM_OMIT_NS_FROM_STRUCT_API**: omits CGLM_STRUCT_API_NS (`glms_`) namespace completely if there is sub namespace e.g `mat4_`, `vec4_` ... DEFAULT is not defined +- **CGLM_STRUCT_API_NS**: define name space for struct api, DEFAULT is **glms** +- **CGLM_STRUCT_API_NAME_SUFFIX**: define name suffix, DEFAULT is **empty** e.g defining it as #define CGLM_STRUCT_API_NAME_SUFFIX s will add s suffix to mat4_mul -> mat4s_mul + +❗️ IMPORTANT ❗️ + +It's a good idea to set up your config macros in build settings like CMake, Xcode, or Visual Studio. This is especially important if you're using features like Modules in Xcode, where adding macros directly before the **cglm** headers might not work. + +Detailed documentation for Struct API: +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Since struct api if built top of array api, see array api functions for more information about individual functions. diff --git a/cglm/docs/source/bezier.rst b/cglm/docs/source/bezier.rst new file mode 100644 index 0000000..8b29751 --- /dev/null +++ b/cglm/docs/source/bezier.rst @@ -0,0 +1,89 @@ +.. default-domain:: C + +Bezier +================================================================================ + +Header: cglm/bezier.h + +Common helpers for cubic bezier and similar curves. + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_bezier` +2. :c:func:`glm_hermite` +3. :c:func:`glm_decasteljau` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: float glm_bezier(float s, float p0, float c0, float c1, float p1) + + | cubic bezier interpolation + | formula: + + .. code-block:: text + + B(s) = P0*(1-s)^3 + 3*C0*s*(1-s)^2 + 3*C1*s^2*(1-s) + P1*s^3 + + | similar result using matrix: + + .. code-block:: text + + B(s) = glm_smc(t, GLM_BEZIER_MAT, (vec4){p0, c0, c1, p1}) + + | glm_eq(glm_smc(...), glm_bezier(...)) should return TRUE + + Parameters: + | *[in]* **s** parameter between 0 and 1 + | *[in]* **p0** begin point + | *[in]* **c0** control point 1 + | *[in]* **c1** control point 2 + | *[in]* **p1** end point + + Returns: + B(s) + +.. c:function:: float glm_hermite(float s, float p0, float t0, float t1, float p1) + + | cubic hermite interpolation + | formula: + + .. code-block:: text + + H(s) = P0*(2*s^3 - 3*s^2 + 1) + T0*(s^3 - 2*s^2 + s) + P1*(-2*s^3 + 3*s^2) + T1*(s^3 - s^2) + + | similar result using matrix: + + .. code-block:: text + + H(s) = glm_smc(t, GLM_HERMITE_MAT, (vec4){p0, p1, c0, c1}) + + | glm_eq(glm_smc(...), glm_hermite(...)) should return TRUE + + + Parameters: + | *[in]* **s** parameter between 0 and 1 + | *[in]* **p0** begin point + | *[in]* **t0** tangent 1 + | *[in]* **t1** tangent 2 + | *[in]* **p1** end point + + Returns: + B(s) + +.. c:function:: float glm_decasteljau(float prm, float p0, float c0, float c1, float p1) + + | iterative way to solve cubic equation + + Parameters: + | *[in]* **prm** parameter between 0 and 1 + | *[in]* **p0** begin point + | *[in]* **c0** control point 1 + | *[in]* **c1** control point 2 + | *[in]* **p1** end point + + Returns: + parameter to use in cubic equation diff --git a/cglm/docs/source/box.rst b/cglm/docs/source/box.rst new file mode 100644 index 0000000..57632d9 --- /dev/null +++ b/cglm/docs/source/box.rst @@ -0,0 +1,181 @@ +.. default-domain:: C + +axis aligned bounding box (AABB) +================================================================================ + +Header: cglm/box.h + +Some convenient functions provided for AABB. + +**Definition of box:** + +cglm defines box as two dimensional array of vec3. +The first element is **min** point and the second one is **max** point. +If you have another type e.g. struct or even another representation then you must +convert it before and after call cglm box function. + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_aabb_transform` +#. :c:func:`glm_aabb_merge` +#. :c:func:`glm_aabb_crop` +#. :c:func:`glm_aabb_crop_until` +#. :c:func:`glm_aabb_frustum` +#. :c:func:`glm_aabb_invalidate` +#. :c:func:`glm_aabb_isvalid` +#. :c:func:`glm_aabb_size` +#. :c:func:`glm_aabb_radius` +#. :c:func:`glm_aabb_center` +#. :c:func:`glm_aabb_aabb` +#. :c:func:`glm_aabb_sphere` +#. :c:func:`glm_aabb_point` +#. :c:func:`glm_aabb_contains` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_aabb_transform(vec3 box[2], mat4 m, vec3 dest[2]) + + | apply transform to Axis-Aligned Bounding Box + + Parameters: + | *[in]* **box** bounding box + | *[in]* **m** transform matrix + | *[out]* **dest** transformed bounding box + +.. c:function:: void glm_aabb_merge(vec3 box1[2], vec3 box2[2], vec3 dest[2]) + + | merges two AABB bounding box and creates new one + + two box must be in same space, if one of box is in different space then + you should consider to convert it's space by glm_box_space + + Parameters: + | *[in]* **box1** bounding box 1 + | *[in]* **box2** bounding box 2 + | *[out]* **dest** merged bounding box + +.. c:function:: void glm_aabb_crop(vec3 box[2], vec3 cropBox[2], vec3 dest[2]) + + | crops a bounding box with another one. + + this could be useful for getting a bbox which fits with view frustum and + object bounding boxes. In this case you crop view frustum box with objects + box + + Parameters: + | *[in]* **box** bounding box 1 + | *[in]* **cropBox** crop box + | *[out]* **dest** cropped bounding box + +.. c:function:: void glm_aabb_crop_until(vec3 box[2], vec3 cropBox[2], vec3 clampBox[2], vec3 dest[2]) + + | crops a bounding box with another one. + + this could be useful for getting a bbox which fits with view frustum and + object bounding boxes. In this case you crop view frustum box with objects + box + + Parameters: + | *[in]* **box** bounding box + | *[in]* **cropBox** crop box + | *[in]* **clampBox** minimum box + | *[out]* **dest** cropped bounding box + +.. c:function:: bool glm_aabb_frustum(vec3 box[2], vec4 planes[6]) + + | check if AABB intersects with frustum planes + + this could be useful for frustum culling using AABB. + + OPTIMIZATION HINT: + if planes order is similar to LEFT, RIGHT, BOTTOM, TOP, NEAR, FAR + then this method should run even faster because it would only use two + planes if object is not inside the two planes + fortunately cglm extracts planes as this order! just pass what you got! + + Parameters: + | *[in]* **box** bounding box + | *[out]* **planes** frustum planes + +.. c:function:: void glm_aabb_invalidate(vec3 box[2]) + + | invalidate AABB min and max values + + | It fills *max* values with -FLT_MAX and *min* values with +FLT_MAX + + Parameters: + | *[in, out]* **box** bounding box + +.. c:function:: bool glm_aabb_isvalid(vec3 box[2]) + + | check if AABB is valid or not + + Parameters: + | *[in]* **box** bounding box + + Returns: + returns true if aabb is valid otherwise false + +.. c:function:: float glm_aabb_size(vec3 box[2]) + + | distance between of min and max + + Parameters: + | *[in]* **box** bounding box + + Returns: + distance between min - max + +.. c:function:: float glm_aabb_radius(vec3 box[2]) + + | radius of sphere which surrounds AABB + + Parameters: + | *[in]* **box** bounding box + +.. c:function:: void glm_aabb_center(vec3 box[2], vec3 dest) + + | computes center point of AABB + + Parameters: + | *[in]* **box** bounding box + | *[out]* **dest** center of bounding box + +.. c:function:: bool glm_aabb_aabb(vec3 box[2], vec3 other[2]) + + | check if two AABB intersects + + Parameters: + | *[in]* **box** bounding box + | *[out]* **other** other bounding box + +.. c:function:: bool glm_aabb_sphere(vec3 box[2], vec4 s) + + | check if AABB intersects with sphere + + | https://github.com/erich666/GraphicsGems/blob/master/gems/BoxSphere.c + | Solid Box - Solid Sphere test. + + Parameters: + | *[in]* **box** solid bounding box + | *[out]* **s** solid sphere + +.. c:function:: bool glm_aabb_point(vec3 box[2], vec3 point) + + | check if point is inside of AABB + + Parameters: + | *[in]* **box** bounding box + | *[out]* **point** point + +.. c:function:: bool glm_aabb_contains(vec3 box[2], vec3 other[2]) + + | check if AABB contains other AABB + + Parameters: + | *[in]* **box** bounding box + | *[out]* **other** other bounding box diff --git a/cglm/docs/source/build.rst b/cglm/docs/source/build.rst new file mode 100644 index 0000000..ad09a04 --- /dev/null +++ b/cglm/docs/source/build.rst @@ -0,0 +1,153 @@ +Build cglm +================================ + +| **cglm** does not have any external dependencies. + +.. note:: + If you only need to inline versions, you don't need to build **cglm**, you don't need to link it to your program. + Just import cglm to your project as dependency / external lib by copy-paste then use it as usual + +CMake (All platforms): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. code-block:: bash + :linenos: + + $ mkdir build + $ cd build + $ cmake .. # [Optional] -DCGLM_SHARED=ON + $ make + $ sudo make install # [Optional] + +**make** will build cglm to **build** folder. +If you don't want to install **cglm** to your system's folder you can get static and dynamic libs in this folder. + +**CMake Options:** + +.. code-block:: CMake + :linenos: + + option(CGLM_SHARED "Shared build" ON) + option(CGLM_STATIC "Static build" OFF) + option(CGLM_USE_C99 "" OFF) # C11 + option(CGLM_USE_TEST "Enable Tests" OFF) # for make check - make test + +**Use as header-only library with your CMake project example** +This requires no building or installation of cglm. + +.. code-block:: CMake + :linenos: + + cmake_minimum_required(VERSION 3.8.2) + + project() + + add_executable(${PROJECT_NAME} src/main.c) + target_link_libraries(${LIBRARY_NAME} PRIVATE + cglm_headers) + + add_subdirectory(external/cglm/ EXCLUDE_FROM_ALL) + +**Use with your CMake project example** + +.. code-block:: CMake + :linenos: + + cmake_minimum_required(VERSION 3.8.2) + + project() + + add_executable(${PROJECT_NAME} src/main.c) + target_link_libraries(${LIBRARY_NAME} PRIVATE + cglm) + + add_subdirectory(external/cglm/) + +Meson (All platforms): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. code-block:: + :linenos: + + $ meson build # [Optional] --default-library=static + $ cd build + $ ninja + $ sudo ninja install # [Optional] + +**Meson Options:** + +.. code-block:: + :linenos: + + c_std=c11 + buildtype=release + default_library=shared + enable_tests=false # to run tests: ninja test + + +**Use with your Meson project** + +.. code-block:: + :linenos: + + # Clone cglm or create a cglm.wrap under /subprojects + project('name', 'c') + + cglm_dep = dependency('cglm', fallback : 'cglm', 'cglm_dep') + + executable('exe', 'src/main.c', dependencies : cglm_dep) + + +Unix (Autotools): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. code-block:: bash + :linenos: + + $ sh autogen.sh + $ ./configure + $ make + $ make check # run tests (optional) + $ [sudo] make install # install to system (optional) + +**make** will build cglm to **.libs** sub folder in project folder. +If you don't want to install **cglm** to your system's folder you can get static and dynamic libs in this folder. + +Windows (MSBuild): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Windows related build files, project files are located in `win` folder, +make sure you are inside in cglm/win folder. + +Code Analysis are enabled, it may take awhile to build. + +.. code-block:: bash + :linenos: + + $ cd win + $ .\build.bat + +if *msbuild* is not worked (because of multi versions of Visual Studio) +then try to build with *devenv*: + +.. code-block:: bash + :linenos: + + $ devenv cglm.sln /Build Release + +Currently tests are not available on Windows. + +Documentation (Sphinx): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +**cglm** uses sphinx framework for documentation, it allows lot of formats for documentation. To see all options see sphinx build page: + +https://www.sphinx-doc.org/en/master/man/sphinx-build.html + +Example build: + +.. code-block:: bash + :linenos: + + $ cd cglm/docs + $ sphinx-build source build diff --git a/cglm/docs/source/call.rst b/cglm/docs/source/call.rst new file mode 100644 index 0000000..9b473fd --- /dev/null +++ b/cglm/docs/source/call.rst @@ -0,0 +1,19 @@ +.. default-domain:: C + +precompiled functions (call) +================================================================================ + +All functions in **glm_** namespace are forced to **inline**. +Most functions also have pre-compiled version. + +Precompiled versions are in **glmc_** namespace. *c* in the namespace stands for +"call". + +Since precompiled functions are just wrapper for inline versions, +these functions are not documented individually. +It would be duplicate documentation also it +would be hard to sync documentation between inline and call version for me. + +By including **clgm/cglm.h** you include all inline versions. To get precompiled +versions you need to include **cglm/call.h** header it also includes all +call versions plus *clgm/cglm.h* (inline versions) diff --git a/cglm/docs/source/cam.rst b/cglm/docs/source/cam.rst new file mode 100644 index 0000000..29e9b02 --- /dev/null +++ b/cglm/docs/source/cam.rst @@ -0,0 +1,313 @@ +.. default-domain:: C + +camera +====== + +Header: cglm/cam.h + +There are many convenient functions for camera. For instance :c:func:`glm_look` +is just wrapper for :c:func:`glm_lookat`. Sometimes you only have direction +instead of target, so that makes easy to build view matrix using direction. +There is also :c:func:`glm_look_anyup` function which can help build view matrix +without providing UP axis. It uses :c:func:`glm_vec3_ortho` to get a UP axis and +builds view matrix. + +You can also *_default* versions of ortho and perspective to build projection +fast if you don't care specific projection values. + +*_decomp* means decompose; these function can help to decompose projection +matrices. + +.. note:: Be careful when working with high range (very small near, very large + far) projection matrices. You may not get exact value you gave. + **float** type cannot store very high precision so you will lose precision. + Also your projection matrix will be inaccurate due to losing precision + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_frustum` +#. :c:func:`glm_ortho` +#. :c:func:`glm_ortho_aabb` +#. :c:func:`glm_ortho_aabb_p` +#. :c:func:`glm_ortho_aabb_pz` +#. :c:func:`glm_ortho_default` +#. :c:func:`glm_ortho_default_s` +#. :c:func:`glm_perspective` +#. :c:func:`glm_persp_move_far` +#. :c:func:`glm_perspective_default` +#. :c:func:`glm_perspective_resize` +#. :c:func:`glm_lookat` +#. :c:func:`glm_look` +#. :c:func:`glm_look_anyup` +#. :c:func:`glm_persp_decomp` +#. :c:func:`glm_persp_decompv` +#. :c:func:`glm_persp_decomp_x` +#. :c:func:`glm_persp_decomp_y` +#. :c:func:`glm_persp_decomp_z` +#. :c:func:`glm_persp_decomp_far` +#. :c:func:`glm_persp_decomp_near` +#. :c:func:`glm_persp_fovy` +#. :c:func:`glm_persp_aspect` +#. :c:func:`glm_persp_sizes` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_frustum(float left, float right, float bottom, float top, float nearVal, float farVal, mat4 dest) + + | set up perspective peprojection matrix + + Parameters: + | *[in]* **left** viewport.left + | *[in]* **right** viewport.right + | *[in]* **bottom** viewport.bottom + | *[in]* **top** viewport.top + | *[in]* **nearVal** near clipping plane + | *[in]* **farVal** far clipping plane + | *[out]* **dest** result matrix + +.. c:function:: void glm_ortho(float left, float right, float bottom, float top, float nearVal, float farVal, mat4 dest) + + | set up orthographic projection matrix + + Parameters: + | *[in]* **left** viewport.left + | *[in]* **right** viewport.right + | *[in]* **bottom** viewport.bottom + | *[in]* **top** viewport.top + | *[in]* **nearVal** near clipping plane + | *[in]* **farVal** far clipping plane + | *[out]* **dest** result matrix + +.. c:function:: void glm_ortho_aabb(vec3 box[2], mat4 dest) + + | set up orthographic projection matrix using bounding box + | bounding box (AABB) must be in view space + + Parameters: + | *[in]* **box** AABB + | *[in]* **dest** result matrix + +.. c:function:: void glm_ortho_aabb_p(vec3 box[2], float padding, mat4 dest) + + | set up orthographic projection matrix using bounding box + | bounding box (AABB) must be in view space + + this version adds padding to box + + Parameters: + | *[in]* **box** AABB + | *[in]* **padding** padding + | *[out]* **dest** result matrix + +.. c:function:: void glm_ortho_aabb_pz(vec3 box[2], float padding, mat4 dest) + + | set up orthographic projection matrix using bounding box + | bounding box (AABB) must be in view space + + this version adds Z padding to box + + Parameters: + | *[in]* **box** AABB + | *[in]* **padding** padding for near and far + | *[out]* **dest** result matrix + + Returns: + square of norm / magnitude + +.. c:function:: void glm_ortho_default(float aspect, mat4 dest) + + | set up unit orthographic projection matrix + + Parameters: + | *[in]* **aspect** aspect ration ( width / height ) + | *[out]* **dest** result matrix + +.. c:function:: void glm_ortho_default_s(float aspect, float size, mat4 dest) + + | set up orthographic projection matrix with given CUBE size + + Parameters: + | *[in]* **aspect** aspect ration ( width / height ) + | *[in]* **size** cube size + | *[out]* **dest** result matrix + +.. c:function:: void glm_perspective(float fovy, float aspect, float nearVal, float farVal, mat4 dest) + + | set up perspective projection matrix + + Parameters: + | *[in]* **fovy** field of view angle (in radians) + | *[in]* **aspect** aspect ratio ( width / height ) + | *[in]* **nearVal** near clipping plane + | *[in]* **farVal** far clipping planes + | *[out]* **dest** result matrix + +.. c:function:: void glm_persp_move_far(mat4 proj, float deltaFar) + + | extend perspective projection matrix's far distance + + | this function does not guarantee far >= near, be aware of that! + + Parameters: + | *[in, out]* **proj** projection matrix to extend + | *[in]* **deltaFar** distance from existing far (negative to shink) + +.. c:function:: void glm_perspective_default(float aspect, mat4 dest) + + | set up perspective projection matrix with default near/far + and angle values + + Parameters: + | *[in]* **aspect** aspect aspect ratio ( width / height ) + | *[out]* **dest** result matrix + +.. c:function:: void glm_perspective_resize(float aspect, mat4 proj) + + | resize perspective matrix by aspect ratio ( width / height ) + this makes very easy to resize proj matrix when window / viewport reized + + Parameters: + | *[in]* **aspect** aspect ratio ( width / height ) + | *[in, out]* **proj** perspective projection matrix + +.. c:function:: void glm_lookat(vec3 eye, vec3 center, vec3 up, mat4 dest) + + | set up view matrix + + .. note:: The UP vector must not be parallel to the line of sight from the eye point to the reference point. + + Parameters: + | *[in]* **eye** eye vector + | *[in]* **center** center vector + | *[in]* **up** up vector + | *[out]* **dest** result matrix + +.. c:function:: void glm_look(vec3 eye, vec3 dir, vec3 up, mat4 dest) + + | set up view matrix + + convenient wrapper for :c:func:`glm_lookat`: if you only have direction not + target self then this might be useful. Because you need to get target + from direction. + + .. note:: The UP vector must not be parallel to the line of sight from the eye point to the reference point. + + Parameters: + | *[in]* **eye** eye vector + | *[in]* **dir** direction vector + | *[in]* **up** up vector + | *[out]* **dest** result matrix + +.. c:function:: void glm_look_anyup(vec3 eye, vec3 dir, mat4 dest) + + | set up view matrix + + convenient wrapper for :c:func:`glm_look` if you only have direction + and if you don't care what UP vector is then this might be useful + to create view matrix + + Parameters: + | *[in]* **eye** eye vector + | *[in]* **dir** direction vector + | *[out]* **dest** result matrix + +.. c:function:: void glm_persp_decomp(mat4 proj, float *nearVal, float *farVal, float *top, float *bottom, float *left, float *right) + + | decomposes frustum values of perspective projection. + + Parameters: + | *[in]* **eye** perspective projection matrix + | *[out]* **nearVal** near + | *[out]* **farVal** far + | *[out]* **top** top + | *[out]* **bottom** bottom + | *[out]* **left** left + | *[out]* **right** right + +.. c:function:: void glm_persp_decompv(mat4 proj, float dest[6]) + + | decomposes frustum values of perspective projection. + | this makes easy to get all values at once + + Parameters: + | *[in]* **proj** perspective projection matrix + | *[out]* **dest** array + +.. c:function:: void glm_persp_decomp_x(mat4 proj, float *left, float *right) + + | decomposes left and right values of perspective projection. + | x stands for x axis (left / right axis) + + Parameters: + | *[in]* **proj** perspective projection matrix + | *[out]* **left** left + | *[out]* **right** right + +.. c:function:: void glm_persp_decomp_y(mat4 proj, float *top, float *bottom) + + | decomposes top and bottom values of perspective projection. + | y stands for y axis (top / bottom axis) + + Parameters: + | *[in]* **proj** perspective projection matrix + | *[out]* **top** top + | *[out]* **bottom** bottom + +.. c:function:: void glm_persp_decomp_z(mat4 proj, float *nearVal, float *farVal) + + | decomposes near and far values of perspective projection. + | z stands for z axis (near / far axis) + + Parameters: + | *[in]* **proj** perspective projection matrix + | *[out]* **nearVal** near + | *[out]* **farVal** far + +.. c:function:: void glm_persp_decomp_far(mat4 proj, float * __restrict farVal) + + | decomposes far value of perspective projection. + + Parameters: + | *[in]* **proj** perspective projection matrix + | *[out]* **farVal** far + +.. c:function:: void glm_persp_decomp_near(mat4 proj, float * __restrict nearVal) + + | decomposes near value of perspective projection. + + Parameters: + | *[in]* **proj** perspective projection matrix + | *[out]* **nearVal** near + +.. c:function:: float glm_persp_fovy(mat4 proj) + + | returns field of view angle along the Y-axis (in radians) + + if you need to degrees, use glm_deg to convert it or use this: + fovy_deg = glm_deg(glm_persp_fovy(projMatrix)) + + Parameters: + | *[in]* **proj** perspective projection matrix + + Returns: + | fovy in radians + +.. c:function:: float glm_persp_aspect(mat4 proj) + + | returns aspect ratio of perspective projection + + Parameters: + | *[in]* **proj** perspective projection matrix + +.. c:function:: void glm_persp_sizes(mat4 proj, float fovy, vec4 dest) + + | returns sizes of near and far planes of perspective projection + + Parameters: + | *[in]* **proj** perspective projection matrix + | *[in]* **fovy** fovy (see brief) + | *[out]* **dest** sizes order: [Wnear, Hnear, Wfar, Hfar] diff --git a/cglm/docs/source/cglm-intro.png b/cglm/docs/source/cglm-intro.png new file mode 100644 index 0000000000000000000000000000000000000000..74ac74ec4a588606b06cc65f5ffc8dbed3d319c8 GIT binary patch literal 74053 zcmeFZWmH^C*Di_$3mOOncMC2-8cnd^&;<7Yjk`Mp*C36%ySuvv3GVLh+Q=pQ{mysx ze%~|BuRF&5alallMpxIWRkLc&xk~2qtPYl!6-PzJM}~reLY4R`0)&Esor8jc0V2Y` zmT)^-xV?Ts+X2PDK$VRW?!V^MOunhwtI9|N^sOwI^bD-N8!|atSijbSg5q-qyk;#7 z?e$2VEzB+L0M7j6f7JlI=KmBklau~c#ommcTvbM%RM^Vakn|H12NMgq05T~lDW9zY z2mllj`>)$yfANzW+uK_Mn32Q11! zhNcfgt^2oReTVt=;6OC#li7@dp`jtWt(_hH!_9GWyUXQHQJ+B?r^UkE#>U3xM|%3= zpMuYKu=L6$#vJDQCWNvlKtiLjv{c8V%6{sOtZsM$b)fjj`r`jh($Rq^vj3NUB_$=T zn3NPg9(sCuwF}dL%Qy|oeUGQ>*;@l;Uh5b)2Z!r#T{@wOePh^-M4c)H;y;OF@KU&A zpaT`G4=Osr0S5()OT5j*iyF^$;AI<`4mjz*rHyzYKH51C1#dAbYU&$l85z{A(t?5y zuot{QAh2w{$>ER%WD2_-W!-8$_T}?owb#otlTKZPIp^(?og9F0)O*mMcCP#OWO*?P z*|8O1Dm4#iCLS!;+P0FKCz}0-C}~npq@LV&AsgHua`&7o#ta;rbv?d$!@eHrjd72On85U~*;CinV=$#$x$<^0}eLh)2Hp@#d{#1v=!orlw{Hf!H|% zg_g+CTg46bH0+Ao#sl%!o)H^VRH#~Ndb*CTSnB5$mw%g?2?yaefSn*t&YZ~}HF&L3 zH);Mi$*^r-0HSUk-q3}jqN0OGt7SWW=OJNXVO0SF>z|B9Qw)f*#2SV&g#(q#)N4P6 z=k6A|Ulb?#c1dcx?&sR#OV*;PcFT71C-Sca4`-;@G`Z|Q-LC2cuxUs|^i6#8)rW7p zSnKqi4j;^Bi6+7OCcce2Si*v;Nd~&hr}*^&R3_M2B6v`c{V}rLC*wL*HxY618>mfD z)ia&a65Ni>X~=?fj#q=tYU;y>BpJiMMmG@qV}#6F0(cvyP&_t0gxc$n5tXJB1_U`91S>4-0-bDcl~BUUVX$uSkF?@ zm+E<&*YW6U7Ha+k=!&k%w3fFKtL!78<7%6SNd;=5M`sgl=*LBx$8?0^dfDaH0*s~YEGiSo zXxw4fE%bu^=$BJg{v^DP=5+UFwc@2lYR_$d_iO14Iv66yGbKelu34{#tNnZoM{?v% zWWcW3-O@Ln)HQ*I5d-*UD_PmmPOJ7OR9?%o4G^+Dg7 zfGPJpCr0)o9>1&o{lRoGMXE7)Jt{{~QIZrH2tPCt+`~`MyJ>v#HgXe+8tWzd>t;_* zPEO3=p~2%dUgec&D9(Ej2oxuI9?#H(FNNh2@kK(x6Due3Os%M;y#2W;ZbO_#M{<$F zAVC&r7_Vt7Auyyzc}(gH|Ga15Fwu%!?lTvfFrt0U{BpqzgiCJ8`5hk@hvqw3`CZI! zeoj#$9~apWFQgVND%r;i-XRjbU1%rZfHoy@f`OLyg9u=9%^vxNu7Vfdjn;%ZBPCWk ziKCFbiD^OpCz))#oKc@A-2Ir}s>Dawi_tFPnP#Wc5G+~l%t+2}p$-i9-P=KR`Ld}1 zXeuYb1x?m(kbIiHo;|f5cvWOUYiKx;{cT@3feMZmlPcx;eS`L^C(vfL8k?~zJO%{4 z-yQ%qcUz#{%j8=atU1YBsD<70!`|{#C@m7n2p~mo7MHgbsk}HGCUMh5p1>WnBMiCW zU2FU}inhOUT-R%F#tfUsan7YbdaF|(9=+`UM zl_IhLu$J4UmhLzSZ;1thg0Wg1GL^1r%8b%nKJ`VAM|V|ZS$lUEox zlbDuqo>m{B>l_<*TNrs_lfYHIz+)tQ{qhE90?!1WLrxzUi8L ziu)_8C>|g#ohx!ybWS{`kLBomfHYS^NLcIQ^0rOE&l0@wkESKY^D! zrv|C+J{6-X#0wWd4iU@8#tK_UK0ljmv8JsS?dqS;ehi}YOb>+IH^_7p9EWx*G;F84 zUkvibGiu-{RGqd#PB^-x`pnK%lGFU~A{Hsb7OJ*}vr{15u!8!1VR&*bmZXnVet4mS zlTJAPNO=o?E{Zx{9;#V&#ly-;ok=w5|4HRxgQ7Vq!qDR)|J%6RS zn$fUA70yI&h^~Wg6N~zp=2V#^OQdvrG;cTSkE>nK^fiM)IP&((V|k~syvVV-%~gwB zFLMv+Vp=G38#mDD4#M`2m4+G#?w3(A%5*PG#y5lCcyV1(Uw3&J)%gGeg?A&}j6WgM zbD?*s-5x;@2EdbDWw{EzeU46^Vmp?vtU|0r1M8gx-J;ChDI$dE#u4;omhJ6q3}ie3 z*6$v%-D1hi`-=_J_A~y*@dp9-J91Bi5Tgif-hZI2S3r*s3#r zTh2pl>k-N5U8I31i59EG@x@-?_5cH=C8{=9kd>F$4L^-bF`{rzC#r8Zfp9{qZ${{c zEaj~E8TS%r)^Ey7jq0NCyhf8rYPvefOJf+1aJ9bH8~@^E6KX}|P4@$NsUV$xD`v~i zHMm>GGVwThGo*{J6EnzxhB?>#`^GOWg2Veu>E5Y<@7iFM3G}db!tE}ZC&s1Q76uK> zYb%)>UZ9;Sk5?ekkkEQND2Z$5!*YO{3dZ>4mvjTXen zZh@-h9jEy4VWT-q2#q>F=Dj$`b50fi7c66G|{i9S*ToP`udC{xD zl(CLuRd`qBm}k#8f2ZtY(~8}91}nZ#02?kl9nprU52eoI-txV1j2bJwI@>4^KpRWfH~doZM) z*uTeWN8X^Es62^i2WQd9yS@mu^o|KYj%4FTPkNrvX`9;(Zy>jFeCIvnvdmuW{QOUN zU2Q+{zAgpl%!fmS4#Q%pks;}Wr}SNDqIyYP9zU)5zyT*qW&uFS7fEn3C>3UuVfky9 zl){Qw(kewe_7in1?1!LukkF^%dsbV)F5rAT-DWfRMaEMweD#-K)Q_pR&4Z46JsbL$!2(c&kRlGpC4E^WDd5phVJ>ELnY*fN|f>3{a>V_H_!V? zypmheyJR$|YEWD`G;n`hR1G(yTL@ogsaewv$wkhXNyY%Yd^w0awIr_w6VzxE@dlP- zuBfq;eHo>URZb5QvrrFR9=L`H?oRcj{0vZax*Nv52j8MFFbI`#ff2?w%?2(>On1NauE$u?Yw=t{5Q@>>4m5V+w zOdfzQS(=d`C_8Wbb4%N^&%iG(6AG!RJ~}wCDEM1vhJQPh=)#PU|IW#pD*4tVPYRd{ zwoi+nDD)(Hql;P%#-xgoz54tp*!{vUaw3DEK4i>1a8$1vhu*7F3rVY%aXp}V8d z>vqz(d$AwnB~0T^;UFl3~?P*mEW7v-jE6=LZqJ3|jh-c*Ogj z`9Kfs8${473{?K^eOj`3Ip0$vi?>Z>RVH9QH z!;u6og$z?0VZFt`=#L-mE3l6qoBr(avM6nt^t#N4Vq!ygKf%fP5WGk z#4Pmrccd0h+VFX5aMgX0k_M+sYQP6V3OQO~-CDoq>am{f9xt#1>SjZg;1ci5w)?sK z#-*=BPTVGmzj5-v<-g zOhY-OhePQ7=0!eO_y&v~~0ig*5LQ^5O5pwm5oP!ApnUEod^nlLVN$t%xQ zkEbb3Y9#Hp-(5m&c3if|>v5Ch_S^k>D%@slPX4YeJ$9cT0yX04uYVl3k|;4~?PUT& zvZ3rlK2ba{rtXHL(j^RM#a67|xpf`&8{agLK-g|Gpd`;@=7Z+T=H-tDdXF?O#=M>{ zwP}46#}jLQehVpy3pdMC`sIwE&(8sqYhPOf0$hoxXh(2rt?Br=Zlh2BSFK9Mg9AJ_mo9EK{ zm8OU_o&Q+^OSm!8TLDvkf;PO_D@R2y(5fK9QeGs1T`Niq*-mcryn zZ{KO5&SJ5uU{*hYhe#Okq>xfmB*Z?-&Z@qU_Ja)%<;PTPVPm_5IPw(;B;r&&deAe8 z2PJx!Pwtz>x~;T4!6`x%5HrnonEHqYKCGZ{A?(0jt4CdFOv*%sNJROU!rN?78uBzC z3Pux5I|Om(pQr3e|7bd<47FD0XT@mdItEG=#_QFrKA0~A-RPF<>+ZM>wZEfsqhAWm zieJLA!~!vTMrT?+AknrFs3BGP)KFtuT5D^kamPl>ioHE3$F&Lgei2Jj2j6JAO_j*P z%|21>P@FcqzGPZEV6YGbKXV|U0h7mp@crRkAN!p5Kc4m9d@zBjKV$+M!Ln`wlD^`!*eVQJ(JdK1KBG}lv+?4mf>PbKdERGF z&a_jHIH1xEI&6r%fMJS!evo5#Z8RC8BY*{Ospe|u{GPHceq4#fp)4U|msfHiW&1!K z)15?+r)72`f!S=#ZTL-j*d+^Of6si@%>Jrqa+=F;tXzi@jO zJ)Rl_^=mdLH#Q4VO>90}Oy>7oBi^SymC>ab?`wXc4@fE=*u6*B@Wsv7x1t`_uzraI zjOz8pr*~$^5bsaL@SJU-+s3g%WxS^XJ~u1uk`cA_t$(jh*6)DVpW@ zxhrvcj?d}=*?YkiP-T`u7JeZPPu10VPTy~=Bt+~haQ)c5Zl3)25U!BcU*>|&CA<<@ z|2=UGAFuq2M9`t8_Np#u{=gLE-H3|qYYLdI1su?#J(OLTw=nmcXI(x~-@rfYxB$)N z5pzh*l;x_EYB%07B@C7&=Pf1bF&lAu@YEYoOe)YO=u1dR8P;+*p&}3AdklTm)7N6K z8qxI@lG6FKcta#9Vjy8z;;bil5d-s4w1K%gwlzU?cyd^0PN(ieM*)`}M|I)$`1lIy zvXslmJQU>B@H>};rl_2&H^?M}g<%JYuaAyDx{?i0X`FP83N_eZc6W z#K*;I1CU*_O`BdmvlT3~36uC{<%? z>E-Cw7Z9)KJ~i~~+9N;wI%?IDb4}`%X*jN1?D^))v{1j8F_CXg$ zK8D-qP<3mi$&U1rA(~o9FTLprYk-gG!>~jj>tLDyE~xvAAH9Z0=zsD#z1ua|h}aQo zl%AkpcQPRT64CK&VArR^iB2-i;WvsmQOQ}lLgSSXk$mIYAi%DtwQ~8KL03;_p=Si4F6lRw5x!4DX4qSx*>q*^vn3`f zH@dm}0&_yun>QBqQ1?Z*$BX@&33-Z}#g3;dt(LztJ|iIJ%N3>id3*#>SOC{QCFD?D z)A8+MvF2leU}e5lFD)RyQF#pfk8pTx zMx3^|l64^WTttcU*=SdyhTQG(B~DKp;FTvhoePO8T0UR@wohA@=%_8a`A z;&OS%S~CFBr=j^11=bb@t%Rv4vdE$yD(ti=fUR$9RS7 zAHYULF~Go)w5qX22ip<9!3?eja-lLzNjAUnZz`WoAL)h8%B6$IhEw?>6%296n0S}+ zkOkBBue>Kw=K2*$fchtEim@Jp8}afirYTNH^a{g<#Ne{G3+}dSU>#+}P)6^yc-yam zTZ^rx1P-fCX`z!-mz`qgoBpo?kB%1*eQ^dS6-dZw zy&hXJEHQYI(wu0^?|t;eN6`(JBd7b6Nv08PdZy>xhGPnMGUU&tRGY)#_p3JjL{^&u zLDoPH%RXbTdQeENwWoB8Dc}&m%@{SAg~gr72TJM*)g&aZjiKtH59STL^35$ysaQ(s z9s&0`Ul0kkY|kCmV&fQ6MLXJA$=hg7{xiY&#ZO+?)^yuU=iuI+Vypu2FP>Z&gyP7Kx*T`1Qr zBWFfcE+7OJOlwb3qeb3GKayL(drd0}U;HZWl)8wDLzAQ55NdIyK+VCep(4j+d5d3@ zJ5xh;(^BlQpJVM)=C)7^7Cc6u3Eo~1)alhWu~6&SXO7sWntJaZO`i^!Fh9{94)#kNtj=IFMX%jx{URX6 z)7!#fDlas2X*-PAl_IH*FOBJhTCkVmW7y;7`#GiDzDn>!9e-luLBB(nB;>w$z%G(w zSFh&#_p*u23G;@u+vx;C4?HLV1lqB~7OtzuG+&h6?>pAzSiO_9P6D@9QKK|dr6-5e z%$ENa(fkafrQMlF@kmT`p=9sL1%#o)ZmRitrC>gSe=3%V0!9GLq;Hn-Cu#)IQ0=jY z-Tb)eFTiRKJWDds5PXf5Kk03&BTbCY1~1=Y%2wAo82J>dp*6pvYAfv&WuQAT3{jV& z%V*dN6J_P<-_tqSR^JgWIHIUKpKZy+-&D6gSXQ*LmgY{cO8Fo``l}S|>oISXhMLg~ z%VxA^RnHF|OyDu=!)XhF1NL{5LPHCv8X;5K3Fc(wlaY*3tJ@VJw1s{q1rhIZxGOpC z=)ui|282>g-Gjo_IUkeb_N*vR`T|Kjxziy*q)1k~biK}bpcwPYALbd(GP@R*uZycAGgy|w zACrfto11qKBnl)nQ$Y0c9W-=QrCFGbj(1Mqs6%}j8Q;z=a{xZZqa19lvq&To^oLPcC>o0Iu=~D&_OH~)j%gTY1b)xtU=fKU&}U*lqtu%z zAZrU%x(3+SihRnl7YqE^rlJ|w4#H?B^cV6)l<5nOAZb$pG?gcAzr~KKNH;&e7E?B&>_QUY1mr-6<{oBten~eZe0yg;#;JlMY*NU!(W2tsx z61|gK;9(Hw6g-|VZLbLme6ER?yJATXCk>6b+p^-CMv>a#o-1W3gF$1wV@b?_SLj8N z4~-UniG&Celjp@C%_rkYV_plXr^3_=bH{9k%%ok#F}G}&{l%^B0Mx@i+Y3{8X~~66 z7c|L3!M|%qoYJRTt1}{0(s1wq4jlCoF_G1sPAtT8S1~#`N~DfPN43TCK9?r^k*nHX zn|(dXCkTJrmG;Xckr+*d;6T_mIW|4x_KhJ7CJZXwoU%HFv-{rxW$Og2y$h)xbd~3% zk!iYq+L`^~l+@mdsmi7H{--2BpoarD3@8s0{W5Qm(^{KeW$(;NBYFx7s7l$cdbmos z(MqTB0w3o3J*DC8agHrHId`V`{5x`PPYz8#mx{;>ppSEqGo-s?z@8bWEdkW3rAs~p z<<9b?wx-}wsna{nviUN(*f(8B&^|0i83|VHFODx{C;O{)pU$euw)hJ#)fg@QbFKe_ zhzgURL_<7eId@b9Ss?o`7DBtk2MOK2HkZYFnRrtIgD zj23y(J|BoY^Dl{(ls7Ub7_$GSC!uD{Gcu#&fC;2TZ=}cN)h8{dQ5W*_kK8 zv_HlF8~*KBm+nBmS=Vf4N=C95@WIF*MbMDrHq=fRJGdf7s~g=!uKVOS|2%bR9`U zC6dwiJ5~SC{a<#lmw`#T4_}n3o<;O>H8k9QzrVllt5CPu#Al;{iL>+{$|oh`NyC#) zWXmPF&ki3x+5@_Q{$kXtvL z=F5c$yMJt@{sMb~$T8}X9i4BIU`GRhX+1?nMQ;;&1(MW`>^=LRRxSzv7o~r5o_{%# z5Ftj8)Ei#co210U6YDN%U=2Mk&NUY9W756_^C*NvzQckFwh#2n|}A*R3b zd*(1U`MO*jZbSh)YDX+ci+`WM{*_OaAw9u9>I0Q`uknQvQR^-Khmtbbk$Pf(5SiTo zP)4ff^xoC|8%yNXTLCdbnsjqABiezTE;zi>U3LH7*90LW%pig{%{dUoTn`WL512*PI{=*~wznH3;_u;k52!xW9rk?#O5TzWIme#f(B~DCtnv&A&$pk!uM<-5c zF`F$5iPv^5tjeZpdww`{%gf0L+lX=Srx1@=`a)Cl#f5WcW&DD0=t?g=DJqKf12KP0 zbab7-<5~B;dt3Q0`uW$T9wi7xc*meD+%pZ$kls{r1%FLq9V>dUx3`yUA!|XK=ep3S zD$}vxEO@BW|1@OJ`GLbw*o1!GRfS+tJ(77_YN_4J3$fXC?{{Wwi(S$KUmm1dL9w@sEGMwGj+)QLt$1031n=_3wf9;cI~9B5ehZ!u1G4G7yS zmHOhQ4g4K4FSu7PY@pQI%Cs&+3ua!R*GN=rE1Lu0k8&`^UYBB`vD#UrDk(j_-4wft zpS&jxIk{5t@I-Vc;u*iS0x#I`bJrH%(<_4=f6!_PysgYqZ}M`hC+W;)r?3G@2y<(y$9z=O;Z%fx9x7OQdU`4!*cgz~pzIt@+Fh}4OSkB`Sg z!q|j23^Xn@rRR#7+!u9-W=1pxF5#wMz^5i4 z1{9jU&ROyH{CV$fb9L0Z(bfyaGYMKAvHnnLC&ETpo@PA8sq(NO*~|1R%WqjfR>`xp zFh6g?yj0MaEOF|RFkv-9bF;8q{kXAqACUHEkt7_2`~K{^H}qy!(C}Gr zs25_0W9ZAi<2DCb9v^W$6Di38Zu#MM%@pi+Q(fU{J4^#ZQsSDl5k+_QTgp7s zvP;ti`@~9yzK!u|OHvGRh*gkdT*%<&y_b*EZuYPvlW-!Max5)sYyPJJ-)KkH^=J4? z_7vv1dMlk6NnAZ)hEK#HS$0%5OuJ5ACsKu%3BST)tq98Gm6esf(vqbpnfFut_eVXT zK`giNig7xmvqvN^cl~(l?vZqO@cMTi^gC9Fc7%0zN8PRHxAFR`u-T2n5vJaS!ps*I zpLkKRNncRlrq;Ty=WshcGWrygF`B~K2J8LqQFq=9Lrb&yPFz3BZLSzH56Mg1f9ha` zCL3E@iUpyDQ+lQ9!dJ84Z-(81Zsv} zD(HZuNvT(=lh|N!J~zkqD$0Y)qpueHVQqqN6GsWr&_viq9TTFYh7N!U z(rH_AC7Isywt3Y8{q%hqo*zNzYHb1m(kw#Em`C-(rbYd7T<~`%@~V`G`#9%vi@6H2 zbO@SaMHCMvl{2D?3{ATEa684NSNF5VkQ4JQvz;I|L=t?~iNCh))HPG8;r7l5Z3pJj zQTJqb9^!T&E%_j*tKEgt_hJf9EgmTsh!#;FmpUYTAE-Ynw4x2x=XI=pMlV_+aXYG9 z{zDt6m?nEVQGn?6S4kKW_`N-gqA+e1tPt0 z;`)vbVmz-ee>K{b1c@DlGjMH_NI~87DOJqM5H)(8thT@C+hcRUcny@*fpaPR%l7@k zDSu+^dz*JL?6fsdT=Ef4oRr*%(r^VjKJx9Xa?9x=5v>k{edn_cQ%Wm)2(yRVRV4(F zx7X#(7ppK^kTrUDuKoH-pL~08%6jOc!@+}*Z@55flJvPH=qw5EiDKvgo?qA^mR{LC zRZYm#E%N^T#UtuP+f5IaocWelC-KqP{S*0ldmnN5;CBtN*GSUmmdv{+Z;0K%^HBCr zfu;!9RWuBzPy38LR(K%4q3Nh%$%U?nd399ml(6xt$KHYFGM;PEhVFx`VGY*PZe1t` zu0Wp_Y86dFWhl)Le>~2Z%;!w78%NCU42=>)VEbFXLSOhy4fQh?DHa6K6CJV?cA zSBe$We>TSa)5~*T{l@@O>9oClQLSGotJelXV2N>J*+h1>a^XC3s&^^hev8t8v({1oANj;!|pCL(+8B`HlJV@dr+Fk?NYpLT7xgDs3B6toR za`IySNIJ;BY=0SmeBL3T;$t(=pv)rZk#C~}ol3+=pJFQ3rEkT$k=$|JCmOLB&+4?n zCPHkvQ(UwaQ+vF!m7l0nW`VO*AqF2$}hnX(& zVFW&2B7qr6+oQjn_NJi3bGhi? zOg#xYiF~%v`*S!Vg@>k}V_mbaiw6t>XXGgnF3>%oLQK&CdUVT84)i%#)!|;oz$HmU z+%JtMzqHGu?#zDLGaJbC%w+0A@En-GbMmIkO1}g=a#<{V>&WDoe~c-~($D7iAQL#O zNzUDQl4lVgkNPUCFeiqfpNgjS9eo{ff<9}rDOJ|dpwG{`323j2?h?M%mYzUn1x<9w zU%%pdC_6zP@=Q~LWj!CyjbhjmeO3QkqNH|77P35GhbXD_(e0&5PhM6H!l684tw!vA z*7;={HJK7TxdOKMU^pA5@KFN+xx#M1Eg#_O7fh?hGCjBo$S{I)xz0zC*kW|#<-s=E zg+=Gfm>-3Y72eJdl^ffGO3`mn%C|upQnoi@lgExM#5D=2jq}gL$_sM0z4N?A6DU_w zrHBFC(^*dfdop1W(7v@g3g3JjN_U%-;WDhAWo1svzpxq$1<0oHx!9sNpyiH2$n4-9 zR3+OW##qgfa`(lf6Ng0=nSMto1j=jbk!3Ki&jVq^ zIFuOMbz-@qUt#+4NFm9DRKU7W+~iHmdH_L9&t=9I`mPK=xXT@VxK>5um^;QdXo&8T zcmtFjNfc(u9-X=xoRiO=4~taZT98;n!cPFPURCQZ6b20)#s#(@^IqU#(j0uJ9FxVP z(b+wFyz!ZoUl3zn5JMV+D5otRw>W)7oQ(0w%)CQz=(S4|f3p0%;}Kp9oiN=*EBBXY zf88RrnZ#2ilx$n58ec-4E!O)*iT)%79yvJUUp z-v&2_U&Up$FlA#kz+%zv(2DuW<2bOSPPpLPL1B4F6avMoNKz~MD{uQhnl&(3&kqZ_ z(?PP4yh4V2mecP{H{T3Y`L_sWD>f6V-Fz>a`Ya+qw{aPH>)iE5655cF^4Szn=|w*L z5ql^PJ9njrNG3&dVj+wi6a1blRW~()aKXvtomkj|UjFxZ(n;l`vArd)0KP_A+Cs1q z;d3=>e`MUmv226=5mCi+q+#A^6sQ%?_HofYvm!5(C0ry*Ah_kHC_LTIq2@5QBhk83 z-ABBk!G#LjENLESwR}jQxj0<$r^qblCV7Lv81d2@QH|jY0a?-!EZR%9p;ceVz8^t| zraGhiR|5552)V&4H%G&wdHii7`+mn5btzqs7t)Lw%loa;`2sUmAX-~j+UJF&%6GRI zAR3p)M9xd-bze?j4VYW?%JKLcj+k^BED%Z^n&-#xu04=2AhGzMV=Z8dPJWgT&i(N@f?+7BpR^l#WW#AI zL!-vKhyI!SP#mf*lGH!fyUtJH6e%ksE0k+^)@HdX#VB}L0%uIRalvvNmdv;JffY=I~wJ%C!)HWdXwODsz=}U@cHI(S|N!81I#+a+wWZk zlOGlR+%lI~aue3t$^`X7Sd|Vk-ie1y(=y~5)9i8dp_ooy^IZPB7;{3*138nOO6Kb#x_XoDfyRvNU>pxh3pN| zigp%mZ;czPoxdH*%02EO5YFENZLD}hpf2VHo@~t>jPo}gX>M9bJnuelqh&If`gic1 zoOYn}d41ATa*RFZyJy|R_UkPjf7>eqxdN{`N@GI%s!;*j>1ca4B~)t(y9~1vuz*dzGfb&^)zDOS#CWe zhhaaJvzqERwWH0K87W(5045$Sud+1vd(!=)*jT)e(LOsLa2bb+v>-cev!_Ci2Pplm z3EGV!U&ih{qTvNi*$|IcTAOiQ#(Ga>yHY~h(U^hFYN5-pxAcwi=K4{k%LuHm6Sxo8 zNS(s~dlp+Dj{;}xA^tGslD&0onnUev=)DK7blDd))EWcbwtkQI?lncC$NS_T4%9+i zYPS+geto_k^3F)(M`gnymzpz%5|zvZpG;N;UJ$-fU&}1}mB4bg)fzeI_Sx6Y_2gR9 zSoxZI$x~X*DvR*)E}iAuX;R*VDZ1Bz5!Ai5cOx;+NtdQYp&cCdsIvSD%mWnev4CSU!{u@Omt)$J z3*P`hr*#2eFY_d#g_0PF``vbMZfTOtgVECMCHiT*O{6s1?K_O!gLwV>>Jqzm>mQK> zgGLKLcEq``*U#S2us7Ksj6p@?YB&?hi-TPS><|XY4FZPTXsoz0(H<=OE}d_XFToSaYJ*VDs$8hd6U+b<2P;> z;HeY!U1@-)a}T~w2wa(vqfRYFBnIAtt*Kmz3$wEuUogdph696ciD{0By+woSSbkLC z+XLb`Zc8-kWyhvHr31BwpJQ2l=gG!`7K*s_C|`7ka@byYuP+&ot3#k^OLgzZ8qx#Z zaCXrBp7`B3?Y#cHKv!k{{LT5%=~1r#5lKvafv`mbQR8&u-guH-g5MbDsnEJBr57VQ z%b##Td=bHk2;-(|MZ#37<+SOWR={=~YJLx+MqOAyy1-NZ{N&TzW+gfh2labOKsvmd z&bHPin}wYj&9H&lTYvbtnh7yj>iNfKEIuw9l9v7of|f|RMF$RfXH=VA-8dkw)mhk{ z)=ZfS!Q7kGmOQ?Pb{6jryqTv>XZ2)*ko&5m_3$dEKeGp)$CkzM7zt|hyvy# zDx?4VUohZRZ)i)hiEmn|{t)VubmC8@Q<8KHlDmmUX>=k!Pl)v|{kntVz=IF+bZgJT z2QO;i_)<72!xtN_4XHu3(7`?#Nh2LVaVbdRu}LMUjy=gehth&3oo4-->iRux>49ce zLrJFL9aQv{-*q*b9*id>-+_%J*+za4 z9GxkNJDx$Z3$v|e@|K)r*CEjWIlU~k#;6+R5$~eQQngd8S9bL?&4e>aZi_1WLk$;W z@^Uz2kp=ju7*#rm=Qv>+AL_2(imrYg_O$|E_IobXrI!Kd*{xQe&hC@Ng}(Xnp~o4u zg=@J1rc^^IMVj*ZRgYB=SUq=sZkl*3=K_sj8foZSVZh^!i(~(jlEFRrMsp(e5a=#n zcC9@wNy?bJ61Mc#8RwbC^2Pg}YDZzI*7ZyA&(S<0$Gh8{K{MPPs^OTe7vvj(?Ql(_ zYI{vd6~tTDc(B<@xc+%VTvJ1&b#V{`<;Bi(FKGU7rfL$8bbxxrj&+sd5aNJtT&&VS zF#yfET_cWKWF)_Pd@s0%n6;Oc^cuTEg8T017y$>RRTu^onh#v|otN}*Qv4O{rf-(_ zMRJMTKxS9_m*S$~R)YJxJt#9rQ!b+5T*9L1=%;T$N-m1k<`i;GEBG&|hl`fY7DW4H z)Jx76A3pE8@`gCY3?MD)(v(&P*{=ZC*uc)+74=w_*hCH^LSk98GE8)nK-GS|9PqZD z@&+do&!MuuLKq!(Iz)wB>{!2T|g;4N#=pmS=`8KDC-EqzU=(^s^V}XzHoMkLXN}IT?ReG zq!rIHt-Bx{=FGgs6MI5EWbhR3u|2eHH#n*-8W*rs_Y33Oh*4y=jxYN%W z=OOYIZ_ACo-D)efG4^()e`(e=B2v+}2S6ZPEKtMYj=E#B&=q~=jdODBS{p!goEv1M zJ$A%X6MIsVe9L;z*KTsRKQEh2xDe7~k%hJ?oo$YU!nxl{JSEYLll7Jwe)i*#HgvlE z=e-}~>Xd&)FWMqQ&H{H7G0Z-QRfB-UcV0LQR#R+SICSYf78m60 zz-KGdbXySx%Z@hItnFh##^uNB3krV=#L47R5kQJ@X; z9e|ggoz38l$j{5EJsAy^6>KK8^#%#=#y>9zdwKyimufUMq)0wp-obkQ+LLA!UDv`Y z#u`{T{fN!r;GJ8TTh*eiMwMY-R0g)h+ds+E*O5yj)tw?KRTWB13!5a|kbdQQRs2x< zZUEHy;1AJ05vKJW*};kx^{w9s41|mCYA9w^!{ejPQ!K$Np&W+lns)VCo7r%j$`JY# zk(2L}e9ThVE1T~B1&vV&9Q{oyAgN%aPPYVJWq`q*$rrg6o}s{NyZEUYK*yu@D_k$# zCpk^AmWweSeXAfQ6W%JP#pDH~t|)1k+CX$sT|4$w^}`Su=NsB#NOro@a<=h4h8Ud^ zf^(W_tMHz$N*C>i28wV4c(Lzwi#N{HW(G5OZ%rrFV{PCs82eiqT(1u_w-CXZW!mwy zQ;BDUK98)OVgKhB7RefMq1vmI4vi+7`vXOZ5V!7bDq2TjctUf0uGqrH$&+ z)Q5pCdQm;i)|#2uBX)_(FL5{(`bcWBC=l%vUZQ|9_n!vjW#`TK4+CAswfO!DK4SWC zI>>5(V_^F!Gsf)B+Yhmyjh_6x55S&%r>88JuK<>J4Cw9c#(2*wT+S^qCh5YG1}GQ{c31T_qkBU2K<3u z&#Sul$>Ayr^^tPH0MTLT5(AU%_IJ13!{a@Z!K+wEb}+Z66MB!R)n7ESjJ6T~AL_n4 zDyn69R{;r15Kuuf0)morPND?Kpaem3&Pa{}f)XWX7(fs~0m(TGaY#dyoO8}$2s1Em zbI!f@=sEZO)?4ene;$iJw$i)z-rZHz)n8S8O;U3v=3Mht{q^&fQv)bVRBEYZSM@!f zc+QYMt%g!*qVt|l36=!y(+c&m4i`j6obODIyu|Cvk!oHf9Z#z@jJxe3nPHLM&Kn3Q zUiWuxr{$kq7AtSRxeuPx=~z#xI&(#P#W3fl-|no^kcrsS)+`&-X55fzsa%k@mClVB zQ(KIu^1ynrs7&v{OErC8qZ{=mJRu6jD?x9PiCp{4VQ1pu)li=O-lf~NI-;xKMZavE z!&50jbi$MFEJtioj2%kU8y1!A{D(B(RvyHgHB2Rl!~#?lMI;cYB(tRF{&srzo;;^; zUfEn3g@@1ayJdkq`34~ z-C+FuI!VR9H9}%J!%!q4oTNU%@Bv+&kdNbKX)d>bUth;YoK%qhB{^j*f3m7NqnIS+ z{kRaI_-Iu{2~Q92Nx|UF)Wb;psc^4e1q;C$EqI|NyeQ$BybMS5Ll2)IFN%d@%1zd7 z)*tiF9eYW?|0sW@!=3w7Wxz5&Bs(B@IAk#9r*zSJDUi@m;(ok7I4NAP5<8JJ$6c3| ze6-$YP>dw|z)1M8JKgufvg_`5ZL5Z$h)e1&l#7`qf`iWKspDGg?6=NK!j%CUnZm;Z z;p$-Z_67bAfg_)FFI&rM<0_KeVmUUEO-2@`+DR_b|H9Fl|C-XXD3OS7 zKmR-<=RW@%MQ1=bY7*aFS~4|nzG>nla#DJ%OH&@_|!{4Hdt~Eij=BH)irBm&`)D$ zSMNG~xMTB2a*)vb>+lcLtrX7lw8VNboSZ=fMX!|$#+N=>;#&%gs*ztP)V_~x8L(Xq zZlq?1z<%IC_ehb$$=`XU{8PINe1;X?eaU$mu{^Tzv#q(t5hNt}2L5%vu)%pCwlSn8 z)w}FG_idlJ+oihlvzl%B8Hz~#2ZoIXX2!y(0cOGSdhQ0y>}Xm|FZ~d5qYZog z>92oyQs7yxYrT#We(i1D%4~R z(ZJ(~;{3Qz8@K!^kQvGv=?%%u=EbQgf+~9n@RF;^hb+1syF^i^bWupZ{bu}CeVmyo z-=rJrLv-FJ@wKH`;k)MHHaI?hgy8gT;7?lBH=YHR^&gNGr|djI{pKY+Ja{P8&POYJ z$>ozSA=PFlDyvn|rvzV$-t{kP#}r_-E4#EdI=Ey-N`9Oee^Yut|$4HVV&4`dJFf~$ZI~_F)NMAG8$TS>ufvXP1*?+g}(>3=}}LgcjXq&s}9NOva&KnY3S$Ew{A1>!WIKGJ`)mq z0rQJu(QEQ+G{=m(R12ps>`!T;fKVkhSMaG1;er!KwCCs9L4bxq~yFo`rpJ?hHE*if=74bT(7U zF^-kM+N1%;=$*-J3=^3>F|4>}GyUbvQbRHWDnt$3s3g3`m7JAhQMXJ)1 zXW%oc6@-gu9_2axlI+@pf$6P}; zIvX^uz2{Hl=+fd6oE#HJ`)wQg{dnFy5W<_1-Qr!GmJ=F_g-B0~tp}RhunJ}b`aGr_ zLK6Z_Fco>6S?ZsyB;8D^iZ&!JU0*i#LHES9J5SnC>{W@*L6wl}xJ5bOv2pR-v#a>!Y;DBt7R6@Jg9X0D2)9LtmfmJ%FSL`H zJy($3w?x)2h}-ALB#|HOyO{lZT__qbYww$I=%omZ26=U(Bi^|$Lf}@aXYZPmXGV*3 zfK09m|D^flb?VML3-QU+Jf9TCR{0$S!EpBE3v<`!Qzs84;eGYbV19gvj5080?n%jAB)v-uo+mL# zJcu|W6VA{!1`Bhhd&To6yPn;=?XaKRmKn8k93=8ZV9Upe<=o=Fl6@bz6I$cvCT+h> z>?nK(glA`EP#V}=>ue(KKgU0x`&!f5Xf?n&h721H{E?T z*^o?h>i$etSY|J%8&!SUsm+XhA9h+zb)zDuCMo_n`3k-62C=SM=7zjt&JVU`!y|y& z2B@m#5J|$xO}|jlk3C?|-W@v-&*Yx7_7A`MBa{2!S>}xJCRp3S44Ze#OH0@JhSR!V zy}TjM5`yl^LvfXQMQry(Ssjr+x|GW>^4fWqE|e3s4L59Td&_NBHPAfqe7b=t)ZT$9 zHsP3ev!7_lYG|ai%gD^*5#0^TL190#5~?Vg6JNdMN9j{3$0%=|T$#X& zzffgX#cQxn?Tmbs+kQykO&z*pG(BP{jORtOGQ4d`1{ta8Yt18m_^i>SckxiTdf&dB z1txpx_x_01@Phb6_Yi`9gic7X$cJ4;J1uYM8-|#H9lzoZEy90Bc$HGoyXXr@mEJsinp??9_4UhT;PV+~tZjg>R5-e^|=IoFa5usc(R@Xg(Ju1oz4|ZqerhG%{b96Hl z>QLm{w)Kq1PxvU0b8j*<VY#!*}E*&8m%b z`6jyAaMz=f{LVDtwm~NH1f_@~U_Q?vE@S0?tm8DT^G5k4ak8hOrNf@W?n!|UY%loz zr@DLF4@sX4_^7nZK_i&PMT9-|9xu2)Jb)MrqI6&e*1QP`v5Z_yIjF5k&ypaAEsxRE z@9Z&$f5Z@F9{bmpeU^FrflXy|E!~SP1N0pv9XDr~V8|n!*;jid$=0&xxUG49$xucT ztt4D_%WDG(`7oDCN*TfTp5q%6Q#@1dNC|R-WrJHbTow&yres;G;3h@Q6Ji&2HDfZI5^?-R@@SS1B4p!TFAe7 zNYLlq$G$D~v#^~v;bygE94YbMN-#lLOJ;DG#9&iHb69O`={c|ER;*x$;YX{QbTrhi z8)Qn!gF0a8@=Hw|#~Ug^qU^pCucWyFRS$iy@q@YNZ^Z)Vc{Iey6xaBGWsyPTuMR zw%t0d7Ag)a7??xt)i9gb3&UU$b;E}T>Xa-CRhV*zBZF}no1r(ObCyp)I7cGVjQs@A zQQUsd_bi#Z1SLq&eMhUVA7^;FRvF|Wb@)F`aw|#>nP%YL(z0w!?4|9?&joBBo)N=e z;)NT?{77*dA)}c6*!939vOHBb6Egd%YB|{^7`=2;8;N%GRC-SpPwl&pG_K3*;BT34 zH^76lE^-OIb#3ucdq%S|>6r0LYZ)ND0JoynU45KWGHJ(hyc~=poZZs{HupHU$sb>5 z)dR;yvML{cNM4zrYf;_{lnE|clPtC3jhN*GH6h6lQ6a^W53zGa%F8cJcjub*8H4OC zI`7(-3G5@=y*r2!P2M+;y$+mfMbK&ptBnW3%{shvZP(QgL5Y=#`6xk2Y09JA7boms0}SDkM5lR9yJ3`ygk_| ztvBXvbur4CH*>;vOcB#5>JOe&^UAOCq$YKIHosM}?Cw3_O8CcR4NMeJno? zPW~!3Ji@!|$ZOZ1ZNV~}DF-kFm>9`RDaQ>f2tC14ZTeC^vZYn3tA@qB@sodJ*p$la z-WHX>1F%idbUtONOa)dcJWKJugDUWZ%XQHbszt`f-L2AehN#XQk}#{ zc;b~DD!Swsg<_ym?*&b4J?XrxIK^s3J_;w;X%&gE?52GV{GbiEQ9mSwgTyi4IJ>4TkD7OjY-gP44|u6sHvx3CjiuFeT$LvCWY!+Db)6ugm) z1_I|(rZ8Qq-h#5mXY*UwbeWzq%~U}~7j&Uwk$lazF23qL3w30VCmkQ7`Y~z8$Y9pE zr0Z$-n8)50X0b#9(U`Splo`^=C>EDZK`~?8d zbQ-$y;TgEikXGDWe}Lt7RLfP=|l4lo!))uGlpEXIDiq|_BJN0jr6aS^~Vlvt9{GtiP1v>wq+$Ql(ICpHKD_&DmGZCPi)!5kFhJpXkt^Z8+pBI5RZH4cB zfHZZh+a*u$-Db!#VKm;O7I4Un-*)Jo{QKMilK%?rvr=y%CAZW!{08J<53y>27e@eP zHw>mLY(HD??JIu(lww|QApZ}42h7l}t56aewra-ONKje`00Z z)J4=vG-zS`UA!zpC8pCK&Aw`0907I5J-Lj!(lM&tFpl(g%=f=j$N@6B z{vAU^c=ChQV-c@UvfP}pf`pg=mzm~FP zS`!EB+^Pd;l4;FKlyB^LBpuaIa~QP+SFaecT62v0=V--6) zJ7w=+mHx9@&tV;hp4yWzLIFo528)~S?fVua14hf-9=@YEIv}x#Jq`rt;7d0&$BVoC zcy*+8WT4+`%m2``prk)u;burkNDWg5r1{Io7xQ8@rKN{e$b^`}T95Y*J;gm7(P0i) z@a+k@6`{q|KYgQLWz=6v9w@q{T?CzsXztNkCwEqOpC873BIl=SqfLsJ_MaUn)Tt`! z|KF9&4Sc|9t9$wEJZ78lcgnxIom7k}(j3{bZ4{TQI(RAOai-I1&>KR!Nwy3N(C z_dJcLWw=MmG7YHzvJAk)>sApn3iXapYKf+kc>rZnUe6W3v^)(B4RuzqQcQ~h>U!lH zKlb5cx6sLBuE^%-bvJ8p`6AEWlX}RC{4U7@D}c)G9=lhR87(ILD2{icNJEwfskPal13P(CWET`T_|@WTXfRO*=GN!4jE?>%Z4X0OBj)HjblS9u69}p> z@K}yFN32OpzfpZP{77idZ>l3dF~`ErK9|{ibDEsuM$^pi)B(wWmWZR*d{_`9=0WV zga1i8Kc{is2M+5HVx~5Lx?6yZVP9W`gBRV4!Hnv(ef(R}5~p|z2=DeX{WOXM(qRNQ z#x(w7lm6k{V!GPVOS4FiQuRSH&YsK}DsWCb8fCKWZPV*4A7LGxOHj6-&(?4}^d9gO znf2BKiIu9;1t2TF_skf-60GtpzWqp4w_UvyG4*l!b#bypwSKHLPG#Id9fz8MQ^|>- z7~WkA2P2BI199DVT?{qLgs@V8hBS-Z7ZrT&)grf1M4#6E198Vt2=CU}&mOkAZHZ61 z&EXzqW~;h~br~EagpKS6@rk6c=F4}B1z=GxSNKqlJhSSL*u2Dvz;jSi81~Nq0WS`z zb05Bi{u~jVT*cG`;4}bznK)3(AHl!>SWf?H+9*=~7(SXjvso>oGAcX!DW^ijd?YoEjC-s{|4)q|mEVRS5e zV#JVq9PY$oHJD5$hhEUL3A(heX736Qn1X7ECNsB-|e^DVaNR}l^*fzeil092>2Jv*KD?SCOdXKvVGR=Og$AI z9&Yln70dq6>XT6&^Uvj%Se9Q6*q+NFxI8BgS*w6GX;8+JJhIdqF~%||cDGM|CxDIID9g*U zz)^LwNE{PKY?r0D0D0t=o%>{~bc;w^XNvRuS$_LjBt;b?9l`_H6KMw!iA3|vO5VIA zS~=Rib?1RiySN3xcF*nyU98r3A55VT0*|@A*+Wz&3t3d2OK*la3iPOzUG@l6{OYm` z2)q_c+%(!3j5yZwM|8BED>Jow-Ah2Gw>6SN>WcNHN=7rEFB1g;E6IbAPLfA+a|<3~ zmeVbz7m?A0qm6DusftwWQL|`62+g6MF#5kO{lUa&Z-mGaNhqBu)m(5wJ1m5Da{%+{VP41xV4r)$Nx6Zy8T9yPQ*g+`@XfYG30=%_s1?aH}z^ zDagf0(X(yP?x#mHc(EV35(B{sh>7QET=ymnafiGWJqIm+(C8{@g?#F^W<516uX76~ z)(&L@Jt<(CR|^cvPLnaAi7ev~joyj;lB|@}e4Dya!kNU%IRqN3L!6a{)1@UkZ@`bY4;*9`xgJF#f?_)g3*xglCBq(CA zA;|F)h&+g8srn&DXtg=40@WIII3X})mL1`AoLNCe z1@=vMFDVR*+V<=@o}mISPlx*V=EK&jREkV9&SB<8(+Omgr>*Af0acwLJer)TNS3J% zwX4dGb-B8QtT48ewx!jgfH}c4>W(Wqtq2qSYK1J^ezmWaCI;_tr#yHrIg|EA@NNBH z`pPbm@)?(}4W2`2cJ&lEn~3QwsWeCr4+i9}2gvWwvC2%qj^lo6^$qFrdUTp6^I47M ze#g49QIdhextLY122|DyAKjC^)fIVmaKOyVH)V9VewxZ*1-aFKx>ypi!b|F*X%wv2 z;wnayK#;Suf?ZXsE4G|b++{8tX}K3qZMnB1Gp?2Ep(M(G!^i;2F(Y-x{$dW=>GM>d z$sBc;D*lQqENl>>n*L&)VI8IR(f%Osgq*XOG!~Zc1!Mv+fKL4uf0JY4c(vCflXMjG49RbK?|B%eeK*5ADp1QNXC?*!>j zT(}Wx*#9Z(e@-Q?09u~`LG|MjiFo=ZL?gHC9w}~{FelULnf)f|+;$I&GqOnq3BIM> zg@$A+ByAyg|8haeLh!T^wPt)_z798QI#x;5`L%i5(rfij85-k@7JQWT4*j5xuk z%V1BTR!&(u*m93~L~TKL70cm;gNU$R8&8lqkP?MfA^3HDPDZwHP#i+&4aK@=ci8J4 zSRD15yjHX2j0yN&q_8pkTb+SV>LD!x+^2NAqoH4%iuD$!(xy*7+`bh#zfIC3sz7J-= z$rZ3vU*X{tE;Ss`;h?HSZeP*0pwiw|`kdX7Ix`q`+P0I{;rObPJYm-(?DCl^W1h}3 zWBKh?Jsr5C`Dpe`b`!*vuUXodvZ8sOT5Oaci~@sYI`lDOb4ZU<-3xq~9J!}130tV1 zr(sqK^&=$LejFVTutbpNy3f982MF)4d#gkny9L6jk}&o+@O0$2wi2D zdrJKPAsjwNZ8&VVAV1P!Sl>v9(iyym+cbFQ0>jXx*%FQEB%80lE6Ve4Y%BGq0V+*08$g!=&xRHi(_6A zUdUIP6RvU|-^88XdjKkF&z>hJ-s3t}Z)r%o1a%!24TLQFh;9Id8lNc9>xM<2kx|q1 zu?B{HMt}xTW(Yi+HSy{-3MUxn9Q&c{^KKXBFb-{nB)4qu{cU3lbP5z>&(yw7*uivj z+VPtw`gZq_HxUWZ9eddj$qBK*OHhYq2MR8JwIts1y_fJRL%S>dY$?#ox`h~yzODkW zPjk;{swCvzZbem{Qr_Ff)WYv47U?)8Gz;I>xxO}wN922p2+ahRfFnPBT8%zo$hbPr zJz_f5gLwLU^L|$qtDOCEscFyG=)Qd^0#WCKtL|51$%mher9Fjg#?&k0zf|v-xz&qbQW$*Ctg+oofJB+0M+n`dA8dkw`nY zDbBivgeKK03TemmyYBTm{=&DiywCUcu!KF_Y9^?z{Mpc}rGD*_SGGx}&ANDyrS#X; z0n+ znkebSERSr@=6@WQKWAo`)IY@sOBiK}yOo+;?Hf+OzDKS*4}0-#isA`o1TqRp26 zH0cSSFkjmX*=*0TMds&t0(CRA|MC(4HPoX1DZIr;5C~(s`=ZU8_2ALbQRwRGYHCVK zibh{DcWzCBPGO%t+3>rcTCRVaQw04$b$MmbCVCBaH%E)~Cf(hwXZD#AwXYw@)Q$zp)q_3CKJiqASpO?qN*KdOW>)yYu0!-F- zzb^V;7r|?3lSLyNM=#Z8iTi!JZ{iG--Nf5|W6jya{M!ctvWFhyhY9zyd2r`-gv6|B zKi1B;W*PMwvnKoHZ*u25k$;z<>hfx99=hn!*0)S2Ak{@_Wv0mis$o)D1 z!}!4SCN1+N-wwg_(C1mMB6cVN0xFUBMfp>1ST2#phrdn|sXK<%g?%9=m6H4`RXCSX+u3BRo)*=m%7xj_2Z6thL5Wv`FM?` z-$GI<=GHomVji0v8HI&8JGkLvj~;)(qtjP)5HC+MXcMC?!OJH#0%^*|GZFx z0^LW`Jfzy5KWX@ho+f(ke2JT|#@((`UxY|?|EoUIB zY*utf_rE<*uF0(!5!>jo_rhQmmb|p@eJ3T@_4AAcB!ayN^5fgL4J<%Vy+1acXH_fx zZpb7u0B34fecG&29F4R~hJ&twGT>aSCE@a&yQbw`b2LYp;%R7UMZUwu@6tz{?rXAGd$TG;r!u6HnRWQ z^!QjPd?p6j1^&&JB5cN7B-ecrIG)U5xm<+v1R zqMjIT^Y^?uqfYDBM5Satk)05bj!~OH4_R42f|f&H?q8>v9!WekSMH8U#SEf!9e>yw zL?G@6P?52_1J4_!1R)QmR(n(V^Qhet1l5qkhTknuW%WtSf~bfG@)wt*U!yk zMY~cQyJ1x{&C^s3NU=h;1uy7lCXxG!8cyif->vkkR6k7D3Y^r&-(=?69Nq@ZVL(=L z;g`SvY_z+fE{I@B6&-`Pt2gQ#Ipk{=auJRp9n!gxcOuU=s+L(&! zI;ub?{1_Ysky2~XP*~qaTc$7IHj_{STyv)Yf9T3YA7UoqFJnPUfsxisMX_VU%0T0< zN7M=WSyf(6(e_TO=8IkR!bh`Lc8R{fd|5ysun5ptTI4_Re@R%q93u}sB0Ub{BAE5hyVY%~5>{O%OcN=U$4M5d41Ru9l!r!8rdF46yvijsNB1-W1jfx2 zQ$X6va@yz;h`fNGpCuCQ<-i_AGWK!X%#{HYbU}>=pX_Z|T1-8fZ;>pcmimKf6{@iY zqYEGrE6%QSBg4baM-^jwbrGHv&44d0?C&E}0yCZ=$eMD;@D}WeDAxkZ%@)zZXbT`8 z4^7;r?PIV`W|chIAMb1gdhDlI|Ij|1nt>oL2ZF?$DJ<(LD3y~4SrSW-ycc-XFQ$&` zqNF~017{RvRJ&}A4!cbOi!YUl;D?TWedVO&z_}K<_b5oIG5_UK4clK@PQnIzu_&DfNEh`a#-qQKVL}!Pl*cQq$I3|C^_v|oA3%3DAF*xl&7YE0 zqWt!YXtvFeM&VSj!^4HvAQWjU+zPk1BUFHNKN}n2%HzMUyaeJij5MuDEoGepPCHZ? z5+Q&@mZI(q^ zSL_hDk~vmdgoM2%nRmkziaqJ!C;!A-1qKub-oi{Hw?@bv&7TWu4$={%8%`EeuC}Q;BaXwv1G%wIg z{ZVb%)FgPHw0pir&yMW)yX5i^v-kawG;J-L#yhes((`LTdbk_q$4%&>^#p|%GjLwl zJWK?2YhktuI1C!LIqN@3;Ug$*PTW(p-+44PR-j#0;!c6=wf8;h1`dMrW88tP=p0pmI_!oUnk1|lm2bB{xw$+E-%$jz9!sAYHKHJM?=~1H!hntyw z(~-5mjc9x?CPn>5)>l|NvPS{Q&!L41JHdlh+8a;2q)bo`f^|cJftitR9&FQw!np~d zS5J{{QM=!_fHP-Rk+n;D%4u$Q*TfP~&o53v@Uo0vb$%(eSs>&k{zA~TbgyAu756YV zDcU#P>5U!Hb1dRh=Ha!;ji0$i3YSc4KffF)!h7kHV-3L_$0YQM;J?hj1d1qc;-L*D zxNr}k!OM_1?M>!3zuKNCs z$B#19*HpVn|AR9Jcs3ygdakL)gX}|k*No8JN3(^bW2xHq!ceDzT->ucsWj3~MuP$+ z13?U_`sB+eQ@7zWH6~p#j4#8tFf4n8$4N)~GYcmo%)6aCX@lAv^a2lmxC}3Ndiwe+ z)8Uf8qWw1)fTLc-;rh99rOm`UeM4Qw!F|N!SYu_v$X@0$;`ZhdT0+jQ<#>OFpp#ju zBbah$`+-VT?s(esv`WuJ>36^EmNP-v!`ry7sQK@9KX%a1V6z)9u{(1xq{R!3V3dl& z3TA@#!+EO7BQrhi?#qxYpim3bRcYmNh1A!hUno-99DX(k5TkzISu#9eb|FozOQ!XG zf1J^ZS_iTtRJxOh~Ikl>A9-;6pBP`{+1^QKXGmk5CiOt;?_SF9J?lM6*$5uBcATRjcfOWPA z+W`5gHH!R0bL~cPD}GOby><|1_03iu>PO^32f`^5wwQc9^?PYD-92P=a^dCtMCRNJ z&{DC-mh{S%1g66J7fQvV6jzGNRiynZ^~^Jz*|Stkk$P)0m!cHPA8?Ibk*C3@jzfK) z0=#KfVVF%$cE%ts^|S()gva6YWAyduACjD6QdVDU#z zT8&{zkTIV(Kf;wb*Pk-_7Z=sw#$5pGyYb0@@EQ;`=xkT%3*T@z^DxGMTCl7gYPpEj zRy;7SY2~25(!y;yARU*g<0|E(Kk_eb>$TZEvLW%f3Or#yv~!`lHP>TkLeOk5`_X@< zMxk(fs^{riQA0#4>TTQ;eGC+2nKyEWb*jpKt{3KTM}I$E@^J%Sv79-pwCmMJ8?ucY zJ=2-)c;=%Q$KD@&L{}^j25$)Oq7w4P*%E+cMa#Ff&Xy$`zl^?t-}Q^r;2Cn#b41}7 zAX!@nydHqNm!{6EHLij@2ol-N%1|aPN+vb$Pru9K&BAyfai^d4WxvS4gF7*$-RR1y zbBXpy7w2FKXI3p$D_y(U6I9J0dGQHBy@PZ&gjW~XEpY}KEc(G)$z<)+L!@p8y#I)% zZ0h{IhaDy^qr!rZM9)%$upPQfx7WP=BY%=Zz4Q7Zx8ktpHH6WpuvVVPesP_%x?eV9 zd7@~%i0HCODccYRl{$gREv?I zlM>P}>Sf2xB@QbP?7|4QRb-Z;)tRpwF*fH7_?*Hh+gG!NeROk>AVM!9YL>c?9nx4V z!mAO|6?Mfg|fOn3AJ z_ONvd43Pb6>@AI5aB9^K<%82V;(<%x zvvAALbaGL%BOZ36bo)`8nvgfhH6A@%HtOXIWwTsO!&iFm`}Iq};^lq%+7$-6v!CW% z)=73i;P%ZZT5lI*muXLe&-;-(IH@B2k9}w5`@+wUSWvMm{O;mjjxBB-gp4RTA6`RL z)QU_n=*iakv-8_AS6n^b%}C1a9tD>my2!c-5c3tu!uhR)^MUa(cay89mBvwfD~GZI zm?wB`Jzgu+1|B&6Vojdu;<=Nb4p73gBUe;=l|_@xH${JO$*e+~e5su%@+u4+rkcDG zJGc#cNrK4T(~bv$Ksks0lRiI)8JHihn;)#(?X+R7A5jGLjtlDm6D%YNzhAQF`ndMk z%v_V>XZ^?I9Y6{w@0ayaueg7rMQFEp$3C13fr(MKI;KJ{yEQv9zEb7aXje8)3wL+L z5_ql%hzAzSWO={##)DpPpGT2e+Ohx;=GsRBpLMAq)vpUc2B5yWorXXpK+;Rc^pi6X z2oCi}o0&n6cKPM_KS}i)ZzuZZ6)qh1uxH5DGG?|CCkQB00DGT)Rp^Y!t|;N-^NaIZ zcF=iG^2V-1JxKI@@;1&F4<juCWub@|gbj6n zHhSD%lWZ1sb-E_qBs-lllu86{srd!383@9#8f}VbKACpvzobN;CVrTSZ(4j2f=A9L z(saYU2(q#vmEO=D>N&ik7@#LfTdM0RUo3QRBzCYqm>tB4m<;z%e7KE|Uc%GS(zL2| z$&cG_UB-^N3?NlU7>Y;JHBN5vWQjSB!bQfN`0e9V-B`niZ7&l+=Oco8BvW{C#!UJ& z_M38^PrMeli)L3274*}&&HIY!A_TGwN4~$Vlefo>yHk|$i;H7Rm^UHQc4Tn|B;o3J z`<+zKP(xt@5GAmXOA!I_h?)~DmA@$DLRaDKq#fD5?>ZxJ>U>(H{K#>ExBZj^TLXg@ z_-qGiNL9F7(n{0+skk!LkaTP&V$SZ>>1Z+AVUO1=Aa}ej*`pG5u}V zUxrL&2>`=r3LW8I_r>|~9DuknrfVZ-WbEjLVnD-Fj~i6)1aDYKNB^~3y$1BsfK7Y` z9lz9H8|XiGUwJaXCjMmn`>5Ps%k`fK(*JbJwQ=^e!QGu45$r6}+A&BodWS0II1S=VLc_15&!b(pn9x~lsZi}S~&iU;r!@#a>cfBypqHqP(`-C}R~gx#>8 z^~1m&^Z4Iz6<`d2M_*(0z*f7EbVOc9M~A?FMP-zKX>&nT*`6#;Twqx(J*@G!j+<`K zs+3{EDczp^U3GJpyD+n{g8Zda%mypsn&aCU(Z4;@U+mGq zfXsQ4B0u(i&Wf}hS=!9d&^s-JVt~gqj~4GZH7HS@Uno-Y*OB}s82f*5nW>MnJmW6L zH7-kwOpt<91}6UkqyeZYemK@kc`bY#AOr$zc$I)GP_h!oB`7a&+`$HZs6GKGR@1o9 z!10%_!6fpBiEUiJ>2a`xSy70ChLh8guyh242L`lqEryDar}!7m5d}5}sAeVO<%3~z zyXw!j53b(dVY67+D@c>f@6fXXV^a7ND+w4r`rt_}1VG3ChNqV~u`zzxV>HdvtCW=~ zk&fY!j>TG58fTH)*E_yKS~Wa$`sp{(^fixaQ~et}NpZCk62s^eRu#R?LdR%Kr|BlfnQ@7|Ur( z#|`J$PXb_Dc|r7dv=>+78vms|`m3yZFMVFa3#UN^DwnwXrNbht*;oHlY2^*n$)mJI!=ijw31 zD0HShwuQkdp1D)Bgd$NAaB7Ia?W+eoggeiKCr4UJ9dfR05uAN`V*qM4ou`^Dea~Na z^npL*Y71gF2@r4JvFs6F`PwB6fX`E}ba;!)|LS)S2X_a-!kl8v{LM6Tu7!qUO9|1c5?s|Z!fxg8qNvtwTExo1sZBR52I1owr{R&?KRmKf6i`@Wf5>B(gFOpbrs{Vw{Zrs znli9&g#S$Cvz_d^pyF4-e8~%c9ptYj-SclexoQ5|ZUWkN2mLhF;`=QY@`u~vy|_Jo z6gOO}ghvh(19l%Vkt?y8mK=wy0Z|x@Sk>W$#{CcMxJzd`P)E1tc6KGYO)Tq|fita5 zyTS9_eheHoqNAS5;fN0M4ity1thbR17&;6UFGR^R8Fpqk9e!83t zzdCUTknSD2#Fr`sx@0HgRM)6h5GON8ASIchJSoZE8A`+b7Rb+AM9yT_pg=&l6PEDCIT781>5F6OTY;u z)R%xeUbX|Iol?l_u~Ue7p9yuBGa=+>GOji znc%y^tl!0QNfp6I`a;`$z%|=#XR;z#X3=tVwu1Q~kIGm{dGEm&5Bd|bkh!v9Cmd5P zg}0~c@?Q5k@&4_4Sf%^*+l_h?zX4X?YtWsg!`J{EaP#;9ZoMTZJceq^@f=Z~mo5k5B;P)R=IIH%~yx%P`EEXE2eQuZMbB8U!Wv2S?vl@0Rt*wyTQ>Ug?h zQdgj@>B9gg=GsVP{#cjH|EMk*Ky8FPILd(0rCC? zH`4dTbbhHg8f|6aO0b;WsjBZ)p1J5ovv(C ze}LY#Cz(20IKgx8?I@BT0}DmA*WWI?(V3nbtzpP9y2xI-=JpL+zOtXJxcu66 zFuVf(^1rBi>!`SwrEPQ*0)!wzlHeX-kl^kFhu|`}ySr;}hu|)Q1`8G(65QQ_1{qv} zhrzxd+s;1ctoyBRt-J1h@BMF9!gTj^byZi@^HjALkF;MuTC(YFHdzkI$y!H-;KTJ; z=-=6i0G>Ar-gPDDWu{4wckw~SSFhEZm0`XPG>U@_T)|Kb7zb)zi@IOjMKnF@&<#&A z8l&$?B#HNr0Ho9Y^EhRB{tNR~flSrpv;ZHteL?D@tn$@9N3_Mh&-YZb5|~$H>%Cz_ zSR3iL4En2~3KTHB6}MV%LUv5w*FzS?S*{C^t5rWg0b6LV>!1()jiC7aMdAbXfn5fi9`JxrY|9G6*z0ps*t+n zlCO4*efNnzIWDA!p#zMZqzv-uRo+lKJGrcgZpWz)R}FeSuC|GMvUs-oYCbIcNYj-wM&#iK($*IH4Q-5f{7~E(CVw$t^n=gVhw7e$Y`73tT7KO}`LezRx69)V zU+;Ipnp*_D^_4^)kMq>N4tX=bsFuk24%Fi+lMw6;SWmg$u;Z)R@`u>8-D^Ja?AS1N z2A$?}rcPbSOqp?(+k4D<>+dy%2o#<;C#NO=*3%O5>w(<|c5YTf@(}XE`v-g)^TTsZ zK9{}e)gK)#Z84074rGB(2yYNH0*HkAHd#pVKmVD3fVRLY~3OiCu)%1s@rvc zNN*?A(nZnYxl|e6jq1K|#9!W?YGQiRF;wocxN&sIdX!et@!^&sUq`7ZhWe%Y++X2> z0`=1c;^6XxX7%6A&N9{4nmjqK$EpV=RFeO+(UCZuL3@&(y`WaNPZ~R6s zuO?7WvsDkA{t7sSd@zA1u6Vi%hPM9J zQMdz8@_Ks|yk4V`d9T>$hJ5pIYHDQ8WNbO85yS0zfpzhTLg&s{@{9 zDX|GAuP;7d_O<6^u$V~iKbDcU%WhsM2$;`uD}71B;}7HV@V;@M%;|I_L_D&0m6hm{ zbfO=EcjZ&#&rDk_eG~V6^`SFNTbF^^bPybW{C4w-=Oz5jEIHu`N6mf4FfNX-hSJg7 zpBf<(IT8zJ1~s}Z03>0%1@C+nzx90ZL`mQ!XS8Ta>%-kK(CI4SCvWEIk|;k*;A%3- zBEK!Tk#BHSG(;xb+}%F|VSDn6&F(9~X~wcCDZiK3+3NqwS?0C_UgR9uOOpV&(=wi> ziV=#94+Fg-%f1uUMrT2cg6z8VwF4doa4RG}TeE&<{^)8{P+`*RuCvY6PstTmF};8eTpPvb2y$#&-AT;!C{qe4&S} zNLg+b*uK<@QpIT(h{;F?shGr4njZ2VNJGUf(qRNo7Ckm^@H#j550s_5)*6zU8R;&p z^z1^;UN2DAZW75qPeh)xVKgn*TDfY!dQ07u^YOUFJhnEx^-Zi(!WegX@uBZ~AD)E( z`^TN2C(Giy6!`F~ue#O$u#NEEBJ{lQOY!T==eePOMLg{CxwB{oliv*cekUj>F z7*%S2X8b@~I&t}fCfr5dL6bqqh%9q2bEaW#@@O;1Iw$!tcS8quE7&<|=t~_oj!n;3 zD^_yk#+W+a{u;Knm|q73@0R!k+CFzByqfIysC1`qHNqj0Bj`G?YY(~?=EK2y*r&(A z=F<~cjS;@Vn&icM(9NRuDED=X(w`)f*7U-CAPT-dlonW-?Tu=^DxtVM(vJ1L(0M{TUf;|Dxt(IA5Y};@I;m_XkPE-<92NtwO(5o(vH6!?rEeyDJU5l=~RfxXv0D_0zd)I;tFs1NzqA>@HJwhj0eLV zRO{?8V67F|Xj>{O3v9d8$Q&E7x`JoA^1Ww=7d3F#{j!2o&Xfct<4^UFWG zZ(kaPz!J}-PTvz`HQFvWgQw6vM0sOJAIjZdWL|V$H0pTpwML^LZ~ENWXeaoN_sFKe zTP)^l|IHOIfqT+}q-U91;`N#tUe9~ffa%&6vGLeI-0(S~n;0?O5^Hh$@}Mj{+vQ`D zq;b)%qpir!U*%(_Hmy|cmBoyR*U!|Sno)VYdCXXlD;-<|kK2}M_MfA+D@Fo2_7LVl z--YWogWKXfnbe+9qg5}iTPhY_dF-k!;*-%;bd+CD2~#b&WZp_O6OkJIb2{uBLljsBF&dF2~J8#7DL=_}TC=Ju=k zzDr#OaX}CMo|xOVS-hQ#^I-nb7Ut#wYCQ!%l$)F%Mcq2|Wn0b4NU2pH#$?UOp&enJ znX=c(8-7|%AFA!}_gq^-P@wAKncnk9>`N-#4R`o6@wdSHWL&Q~ey51_cx&);lK#!Q znvYARZ2}eHpP{BvKiL>)w44K)TVvbgu%13Vi*p* zGg7SELy)&x+_B_(rvE6Jw}8ql)N3tkkU6NvhAO)7SI)rc9gdFkg#1R|<;zc}dx`NE ztu0wsEi%ilmr$9VDFcVs*9;;QVP|Z$`$T$TjL~r#-h;o`Z|^@g`Z=DBaLR8P*`A$v zWwVfcIGxk8lYmI>J0(!d>#E>)Jm0^mfOq*mj24MhGRL}^!!ZImTU|RIPEMg23#i-2 z$X0X5?}B+w-aoat_iGLGePfgHqTJ@2AMMc8i66yb?bEy5_wAm4I|WFOx#mjvOE1+t zVmtf;)dBdQaYJxIWrm5Kma({zxPoA#TkGCm>A?=}Jgo6nOdmDX$A;}pjqD7$Q@-z; z6smvEA<#on_YH+=Nl}DcYJPDu6oOWtGXF9YKhv7ib{e7ZMy+@YTd21JpQc{SSZ^+G zIt_5LezOY8kVsD0Ry)m<;r1QzY#>Cual*ZDLiT2B&b+woa9T+k^_skn1Do2d4xZ{y zuk{s5P))KnEQZT%(Q zG+U*gch^Jve&1YV#y|D%RNdVpom1~5qv`3J@tmhj5v~Tt{56iy!7=?Gy|}(`h?8%U zWpVY-H1xWMPnU7~(}d-(e1pCv&1E$8QT( zYfoZ^CwX(<4boNJa>TX?_HGS@r!Ypw8u^e76@T{)qmI~mBt0DdvL_6=b;W)7;G^~I z%-zCMqnJoD4cYI*?S0H0bVc@#I}X+K4EF%+H9>KaGzwj*Fyd3xV>MZ+)ka5FFA&kTGnI`Y9)R!e~9Ht-;}8!miT z2F%o-)88#$Vzx@puU{C6i0vu7l}}Tw-(ix1S1Zsc%a?GMeGGVO z)Tc>f#_?qES}{9hh_S=_<#lW!dP{SL?%*i>FS3HBNS=a~w4TFcNNY47IzK+VFDl|e zvYqz^LE6BP6TG9JOYTT-*Dod9J2L2v*TS|;7M=QlD;iF=8g8w{R8ip98n~iGp6%_w z*`yZ|j|@I5Wve%E$b_T;!{*rR$8WCY>&Irdh*pW4F5Sm2KuS0P%%n#1TN7s`i$_93 z*{BP4&$>$(ZX`V#P$q`_Z_41Ui}bZS0<4oASbvx;kJ$R{x+cl-rbHja{%tQ&v7UI) zD^UN5U}!zL{=kWNX;t#0n%<6ouLN>R>HM#JaS0ck*=M?#>8HP^t#dn&e`eqO1fGu` z+q^piDY-135vwy2+J8!E_m7F50~3{fm=XPBB0=Pzug+K(_j}uybWW8#wY*7I;ayU4Fj~A%1gnq~yBQMm^#xJcOxW3zgbf@2%wYZ`r z1pI{w%p<{(V6Gw&x;>gUkc9t3kMGeAB=#P7<$20a!M4Mfa^V0lbXZx5$U zJ@V}zH~|jqm%b%{bXA`q>=$*E?sDNdmD_+up0a{fZtcAYso|5A{YP5f_Zi!?yy_l= z3ZP2s%5$R8&d@$m*#--Yyi`IU|ICE{BPsrW_EAU-`6v5N@%u5JRmY!6X%QdUng5dc z8>U~+xN-@-ZET|fS|`?%R(XMNKQi2ZVIZ#c$p@~g|5_&y5K)a^A{^Oxi_=jveTLL>LGV7)*VK_F7Nf0k+AiHu!3vp7yOY{-ZQJW6Xh z+dyts9gzE%^#BCCgBOsVT`{1C0JP9`p@3YvfxOhbL;&n2FjWCk&aIUz!Np9EKl=T06)oaQsx@P*V5En>7>6L0SKl#2$W{kbW^JD zfvm=#3~^N2$7%RIPK|pzWx?Z`4-o==JR4AK%gYd9Pn`kTK+Y&U)~#p&=6Kn0Krk!^ zfG8LHdHf5pWwoSU_`MhA$@KU|HQou22ugp_p!xv-c}4)kvKPQp zEt7u-LV`2@qjX@02B^l!#$s&+;FQ{@m+W3TDuWsaAA)55{vD(N*^gRI!`uZ(ja^3n z5eez}uVIiMhZ$!n|9zNdR$!RX{~88Z%@Ok3E&x1bm&tvELJDjYS?u3b^$_S9TE`;0tbmum5a6hm8p9AoHPz`zj^FHSI-mgMm5l zU;Ld)dIVf^`yMS}HoKMD84xMAF*3okr)v}=62s}-uD$Sc2WX>uD@M=YDc=#TvhBPw z1VCTC9*Q{`uW1>6+Lf&J6#yM_LQsJOR-K9djONrYXSYk8B z=XTuGd!NmB4fn5pJIoS?$9IF`+xb>+<;}R54|b>B~l?F_(j{tQ~7U|K3FBc0eb&jhzpC8ezZ~kjm$=oLZB24$&#GC(=v=A`P<$o0)b)?{CNO2v@G0B&P6r{Azqql-&!tvOyzo` z3>NaG+@DsYT$ZGGo`VIM^#1kZfD0NZ^I8R-xqJKAA;?cDMT@zTK%Jre#|M-J(^{E5 zuyUD;_f#(UkTgYit;~J%51r~EnE%5S{9{MZQy*u~x!t5bll*mvq9sl$4`+xagb61{iK0vHy>%V;ASh4RyvvK($_*v^0@4;+( zMHIq+9+!Z0DuvpIz-Wi%pIOJ=X41gr37WPCS_V~pwX>=q07_s%etZM}n_cEd27u&| zVRl!SjYT|ClEtN^B{tWC`LVGK&ST_H4PIlkv!kzFN{0Zpiw#`Ud9uFA#-E07vAGEF zY5&@taf07#a8sEf4kzn9c7>-^VqK493&(;w!f*^>J1 zA1)~@8=yd~vT{oNYq&pFS0Vx&;J;k`zmMgkD*S)=D?$|n^A)64;hGLUv15e)_?drd zX9wjYMd|kXEL2~)st|vSjg76`*ryLDG28#v57@11h%!*gcAjGQhu!Lc1zi7wkd=sm zt68I~Tn2Cl%)foUR|I%2UGxMKr+~>f=4dh1vBi%f5%hiQ0(f?L(-pthNXY(PBN403 zc&NyDQQ{7tk^2uz4zzpxqaJ!>8fNwX6D4Dtb`Qqa)Y4k~^z}jf!L_iL&iLd%_c@mX zdAcs(?(W#{;RkRbeR(+5tv2l0dW^NfHXL2Vg_lL&h`D$;HYWoNh06c1C5WQMpE3Tn zx%!_Ux8^1x=hk>biaN_#fOTFuwar#k(Aw74*AqzdjnprdxB$V@oGf{YE=9TdlZ#2> z3#Hfyj>_=A@>;-MK*;nMZyf_&1PlOrLVH?z$7{5%ui5+n^txy|mfg1DZ*lI_?b&8A zu2t^1`}W&lZ`JQMRppDzzXe{L0BlxuB>SYdf-vB0l(#B80}&jzZk3icCL1Vb)ElMI z<0?Bgc$FOYNW-i-`9n#QY;=_W8LprO^sb0h-6M7UL?9F*AZBLlU`0n)BZ8 zmNdyiRr%jJq5pl4|9vX|+gtc=8T_{t|1$OeBY#_>_XI7eTywviOeEAApT)a>i%eD| zJ@lOZu!yX}#$?x`bwwqY)5Dgi#;d<9^uW9^ApLUuu&30C`?6s zcMd}B(DGc1&O$9Oy7`^bp5NC)b+pPt1UenT@m9~Tbu#Gu11)8r`9TsE^*aB!Z%h21 z@Uwon>VjM#h{SZ33%E#b#C0*J5L{um&bn3o;OK-GAtpMd$I$XrN_%e4^8=XE3}J9u zbuw_KW8G$&D9&XiX4zEJ1bvx7!fj!g=^xU#5YaamE1P69W(&Ds8y>O1T}YB`$w#P= zG>jH;Me{K1N0>ejAugKpl0(XN1B)H{q%})>e1?JBqnkx)+83*D-j@yK-9UHbGVuP) z?fj3(=>Pd6qe?xla6t%T2%I)?Z%OAdSsc8St0OYep$?G7mVAhu>^Lw&lAGK3_4PYi zQ7W-rU}^n+ZRO<}%zEydQkmvn2hF|$TfWctrfAm<%o|dftf#c^NQo+mwb8SsfHqlT zJNnaM&v7^9L+7OW3uB|E_VhWq&EWy^nN;kQ|hp%>U%#kPi5kFL3J>Bd1N(>*-kSi#{>?r+$ zm}3!jk5_|1;swPEQqt+B$Z8I(Xpt_3sFfw{@4>V-Kjp|qWG1{6Pkr$DgM7n|-Zim) zS`-WbDk;y5Wf^{QGZwy>-AG^~wqq-o12Fp^?2wFHR1Q+)X{Cn|m#bGh5BYeYjc=p2 zG{ks}y0IJ(Py3l>s_m^ob7zdfMjmXb5y`o5p*Gxkyw`c0PZXm@wue2~Y5B+5Wvf16AeInh!(>agcE~Hq4}2xOG8dlUp^$O3k^= zsHv5Rm8g4yyWjYOX}K$@nikwH4E61DX|hEx$3ICmEDsJdLZ5^g`?M7hJ{A>t34}e89L*>va?O^6jxb5s6~aCGf^g5m}MqZGU2h3G);6^h*im z1Qj(@)s}<}WwnAKPKq!MIfg=NYYrEFQU&=w2bjpHpkJ6H=Abev^JfyV3wO*@cZh)t z*8B^&D!d!G%_BJ3+}AgDD5A&0ei5g{uDu^5@@vQ6s~p8Uv|6pG?x$yl$K6b9fHqbe zAWZGgO|XW*_5m1RMnu<@Qk}pfAvcJKBJr3H_vBZ^PEzM4ku+>G2QbTG5}R*=uH3AL z3YFXQxxD%vb(N~xM3st9mAWw-=d~&Mu@KxjkMu3wbCEjY!t9GJzLmWSu?qP@WcQI0 zHpQuZnJ4v2ew=W$Ri;oXw>^jF=^spUHCaEi79Eb_<^v+cokCIQWu?x2Z{7|Mh3|NX zP|FR%JFo0OJ#gL`rZ@e8XY_`MjBy~vH=ya6GM`}v;t;(;XQEZBb7R(_bUN~4mYn2g z^pw_9b&tWMi*9*e#oON80-}fEFVNf(&%KX6G)`X-hA|%96L#BPx7HL>!cbRHHdT)+G`d}X=GJk;HrA(A^QW5?FE0( zM>{j{h$~$RcyH%qHtqdb!%fYyr*3n5Y<&A_N}sla0Il2MunYNIRAxrA(-KyBS0VaL zBh`y;D{Cuk1UaXp7_t+-M zH$wAR6~5Z|pBQy`yt9|bbYfX2PIhn;CGaKb7YP2mc7mP_4XQ8si$Jl{ncXM}9z?Gj z#DXD+9^8WoB0A$fe=3b13sIgRe7OPJ-{DAv6D`!ybC49^J|PNMYQX(Ooj(B5X8DRd zbRMa?cekrb++wU6)Y`?-HyNw2P{*5z?D~}< zXV(?HUXyUurXyi@ri@Ff~#=W_#L>dtuds`p)3B^lHDy+q-k- z#CENtd~97GhZQLSi&~0_(*6y!XKnUAGq!CY!yJjjG9!lV>rEGWPww>aqC9P0$4oQt zpDza;yf`%@97h=3M!vRmQ7&8RBsd+Q)me!vp0rywkaq+$AJCy+uI^p>YmYe*jdB_D zqz4#EluJa8CxH4qcDdq4?wT#@v_!EdP2X7E;?@YreFK>!g^-tX>Md`5-ohO+#W-Do97a71KD?$Hrd!C`vKRKQ5^4mF~j$iKuYQDUmVhXAQy^xF2vv- z1mMO9BtMy|URPQL=QrflgRVnXV=fP!e{AYl%z+6zNs+doAnNGP5qf(hmx}6g65JD+ z!G?9(tCdUx_nnk0Y>6D1;ih$rDA~JKRYHCd(oK|ysylP$0t*q+QzlL?Qrc6$WJG_I zhgn2bX(_<${F*EqwC0N{CJ`;l?8Al-NI4-asj{o6+fL)u3aEW`Mx7zMj1wC>0*ML6 z?8_)pw|=?jp@9mDx@wD~>@-)}=P%5$U`#L{6-`c##ZG~dQ=W6NL~L&`g9|?~W!3;_ zaV(#(tkeyrX$G7*y3?U*=!x3+&RtE_Ld6y3VarR{da8_|0Dc?ti12{Zo56Y~Py{am z^!B6EwcR-9g!}|+$r5vz+hRTS_JyT(g42=I0{mgRaQuQc?&U>O4gWq|{SFE`MM?Et z1J6kI2K`2C-A0Q-3^W|Hy%z<~TC@^EEX=I_Laa&LtxpxQtW<=P8>!fPEs>i!w62#} z7JQMCn|Zvh)Dczq8Z!SnLtzce4u3GUINj(6dB@xGAgIU=Q({dAZrdyV5H(0*eKEsD zSVe-v^z{y?4)5-X4~ofcsw?5o%0^PvmR#T|E=dyMV@>8Wop}s(kV{#mR~lNxjyix;>(H{WPAZ>?-5d2bTl%heQ-RX<&T*^ zaPP@~M$lo|wCRIN3wyWi-APA%Lf@C)oV*YIs~3QyOIOPX(STj&YoFbQcCQDjEW*Vm zyp;Ro_WGyG;T-jLwewx7Ez1!CyC+|eE_LnBRa-8abA3{U>SLR zOT^IiXU5*IHfbjb-YI!{d0kR>u+NYmzm*R=NV?tG!g${<26&B#%in1P(~A)_qg8D?!rEZ3=*i}74+KJfw`6=eZ$pq&ok^joo+lA z*58~{B{GGB8_&}|#~P$y+nC|#T2z%}%5>1nIqKle8~8o(q?7)29-2r&{V@~cW(p8T zHpKBN4<~QBtxmq?`&#FlzFqQ{x=k3y-{%;7cIQBMWZOx&tv$>Mz6q!N{+)*!Y$Sy? zM(>Mb-&iBAf6qy{76JABY+pik&5l;u?w0?R$@x0Ba#Ul+wjqDG3l!K#fI+(n`?O4} zvLql?l-{R3O4j7s{=$Sl-pVXytbRu-(Ls3DNtXvP>w}|!e9oB5ns=hpe9E|`3aWIZ zGh;X7T)MPgLc+aJo2fz_$D_w^S`uu+&i(Rk_+Be3J8pjwZztjsDoh_~<=$)Bw6&c& zc1_BI>(ZxMw*2wx{rPLa?Dw9xDE8v%iJeVQv7u!}ulN^1qMSAykLX4XY>01-^+sJn z^BpfxVxw&G9T_&7Gg+0Bb)ZiYPqYG819vnM9pu-GbZO3fN^t!?FW7J<#Z`%gLh{R{ zICpcRbU(0EKCgm1gmr$EBZgx^LQv+NiIPcvXBZA1C)ATu!vq4RzD`(mAGa zng(g`I1hKcRF73N+*Z_S0_&wfxEl`-NgHCc2=eN6Zf9A3B_mhC*HWhi=Qq;JM-qVV zyAq`oV~-MhF!qDmpzB=n<^8?_Oja8xu0qbiRzv-iY3$@5qSXm~(^Bt~B^DZaa)zrJ zcIjd$l`;IT>Ydb;?m2wj>x+gm*OrRh58ylrMcBiTBCfNmfd|1L@|EGaZz9LO)zurk zd;VsWLc%KXUF=;X=5i;#2SO-$a)#DJOt!dRlx~>gZRI=8I=XHD^Q=m3*xtmBN}~7r zY%YR1g{2r&yMD2v--(ir*bX6jh4JRovZvl;yG&ks-;(I36LA%7UMIe{Vsj9{r7Pdk zOn34by(xyg$gfo#6;z(kn!w%>SBWC~4K!!GtT&sU@TIt8{{6Bbjdmsz&QjTq6 z&eACkDvlOL73UYE31o3KIqudR2@5kh z^UjkJ#@{4vRD#@K!Z7*7`vjgC3GN5uJI)=!aS_mViBD7a5I4in`uu1QrHQ2GjpK}U z1NrK-3-PHtUD&zrcbvo78>LH~=QDg;j)#0CujWRVT49(l+QIso^VY15BQsE+OU!jj zzBatBzizOtSQBwK<#Og!r1J*2)aq1W5`SkdYPuo)F!|i7M>PSs7d+cYH#(yvR-&G7 z(DmIzc4!mNL`RVwIDru!Fifmdjhyu=H4l@$CGs*;o*Ox^?f2y+(iw$C-AqwCRW>8N z60SYJ|HhJI+O(!!nIAw2(}$h|#6?mi0(W2hekT-%=4aIHJIqcX&J9lE_u4`dyrh zJyKyaU)oiIXkempB^DV-6BOo@o_L;^&yVZmqjrw-3BMLHScpnN=u!kSv45Q$!V&9h zWHu_NI5JQVWU?uzWEv~w_&~oM>9zoxwudJt_OJprz@{>2F|4g>Oj^L1pTg zM&~*C7N856rAqDkQC=FEUI_P!(r7ryqeso1fAHE;swKIBF-QNJe()EEOTRkH+?i5J zdkY-pd-mS1g&s+jDFoXuH}azLnpu5mW$kDP1cmVAOvP5j8z7PT)W z9E=pWULv~PvV{3!@kbj*B{~f}=$=1}S1F(#EGnKJ$bf0k#nOzz2qRg33wI=q7UiDU ze1m7Z_C1P)DW^|nJP@f4eemmL#26{Z-ocDa?}+FwRsB5Z9gRmYz^x0Ivw1T zB<|aRx2LozusHjiwMPe-Yn(DGg}6=2Tx2VIti%f!(OTJeX-B@@FJ3TfhNFKp;@S92 z=oCD3;`3tgsqF4G+5T9r*Mt*jqp1=I#w;762$W-r#~Bq5GN;LmRx-JkPh46PUDU5mOb0`S@T;M(a}z0v?EM^`DJcTNp;5ujj1mYjh2ZjWb8r2ImayI( zCy?+N4>#$OgyUwK=)EoL#2Sb?3U!=83@11LP`SO%K`zljLwSkGjK`{I$1Hz-lSHAmt@U}0d#A46YBWrTs2k38y!A-Srk09sam-Kt!z7qTx&xd z0WQ)MCqNwJRl()nL?*X4(rj=&x`XpHLnZdyda~ku5YGH#NZEvvN@a}uRoajT`}TCy zqVi~B$CCTAW80>J`06R|ml@QD>1W=Pt8-|ot}p1p3uI-gHR~T#7mz0%jLP3UjP{Bn zmM|+t^iio-prk~4^T_hZuMIAHgDzckLzKkd71=jQe2*%?D&&h=A+pTf-nq%0xZz1D zeLYyHRchr}q0379-IjEvn4sUM@gw2j1?$bRK#O>+YY_>QPkpaOCu&amF7o2;P`=T! zUQ02gz>jlqBolYU?j2r^@o?&yLY^VvD=(TG)7%Eg@a8=)ZFZ-TOGjm++5xBjizmkA z*j=2LvZFm~D;|77;8>l}?|7`m6%mfq%K?tj5YUaQkAZ&iNsf3S&K-?)PQcMSyr9>C z^Fn;6SrLZ*$NRq-xXCsxt!V@}D!gJq8d6E*wO>=Q*LPt|`9h2Mb5Xs@#E93$mwj;9 zV1W;()@wpAcd4EhT}5rtwzGSj8|R|?EBer602qBZ9)x7Yo#OcNuA;7r%36+?I}IwT zzTj)^+Som&(^6+~DtW)VSSqt{NtXDl)aq>ulk6%T%(AhU>2R3cCW!z8L0Ro)20DL3 znI1hU-sIkf(S{%^)R#YPf(B#v#J$+WmC4!lW{~k5p!R|M=&{Vr637YOTl63LJFiR3`__cI{?v z&Q15-L-wX*Ko2ZAnOt`*^Im`>M4Eg}9}a7*&fO)LwX&@2d-WZf!hM01)v5=O3l|xG zQ&)8&Rlg8zy|~2W>y$lkz0)UN=W#X9n@8q4|E(9p6ZtKi?K68R z#|~}7$>TO;44%AH(`QA&#L8&6%Cb)15Eg5+?He5IKDKXn9DFZVYnimHr#q~?Kf7t; z0Vss9_daH(v32@PNY{k%AFxM+vJ~)?v?v5EH-**r++W#Q8N1Bux+t#62&;rHzEb2K zkx-!y9F-!Pm+7CA{0#bPLkVdm49kou$)=M(Ca0Yf>t_=+*lLc)!W$94P&{B;(+dtu z+xvp(--g*Xh6owGm)YbmK+5+V$Kl3~S0qqgP<=7euCgV(y!J&L@)Js3!g`tuCvjnd zoZmc;(R)}yR$|5>cVQwy*72^HVz19w<)N3tPna#4txgJ@p`1g=>+0GiXSDrP^v|qr}%w|JOZAvIR=Rdc*X}w-Rd&-oRXKjWZ{BtP zyEmQ!<=1@%{cj3hBKMkHAm7D_U|yhj${Dk$+6sH`B`}DEkfod;nMW$EzYlrC>UU zR0XY`^M<0>VGPKcT~w7vC#hY(PsLy$rxS%XqI0_T0Ln01LM~~bRu~;4f7A_rCBGZr zS&!jCvKv?@^0JSiA0Z6??2RZ2rX_9!*kI^Ta+AxLG@lJLj38M=rI@yv>y4Nt@m1Yq z4qH*8205Mps%J7w!u`g3jM6w!fEg!LHO##SKdGYgIS)<`H}18->FfMch1R>cu6PQJ z6P<_>2tjBGz42hYSij?&=BOm`LNRE0jgosYDa)rZl(@M#ldwTq*EF^?dd?FiSdTLr z78A~kr1)Q@%1JF*@f-1hm`Miu-I7GDd%n)No>g_lWkhtX?nO^!lJ2fB&=8eAUA;3_ z30Wo0AG+*$WuG2N(Yp?qB~-^n$H!%y(QWPiH&LH>_wdN2l@BEq zeJBkWH#%`B7{MufshM;}h#FB=(w{4=ntIT_z%OZA*RU8r~ly#xx zBcQ`@eEx}~5msPP6cl0K7n|)k`QdjdVLYhpu4v~sK%#-Zc;%lbqKoGmf z#s;UPA$2KHDYhb>Oinw*A9qatiL)a9YuJ!#*vnIicri(95jCVjsgyxk@oW4KQNr!- zILIP8kjQ6g?UE{Zp3f8(V+3R326s@p|MJ%dy~gOc3|m(fm_bxg^AyCzM5&&>`C6i@6N)ONVmpN4AZx~}E8!kK zEUx=rOhA~2f|fNYA!?yyuV+7O&n#a!66_GYV2zA5&B?8l^$LFGShPrBQ1)6Qps|gT zHh5uiPNJ!_t!JOnwO98}ZhQ`OVd=JKv?9Xx#Uv82I2jTlhJB;i^Ey$4A3&>=VW%sY z6RCO3g(Ug+I_2GxMy0UpuiYr%yLUXjo7N<(PY_QkcWE2k4Tc!Nd-ly--DPH#9pS(k zI^f*6rP!q0v=()89if3yfXUNS-iu79ShoBAHu)-?IE9^r3(X>u??VY=(vyr*M=SC9 zEQ|})W^hMA-Id6F0O(#e42E$bk=|l?|8f}2hI5Cn5!_8c@fm$YFU1~h>kWSG9t)CM z>PghrX!Tj9`&pj5FZOq+^g|qw7QERBDAEjd@h%9wcq7k~Xcr zsSBTfph|;BjAlm4f_06n5#b%i7gq9aFY7e7^-|Lt0#UuT~$UT+HfqoHfSeUfaLfRD@w|cVWKgw@ugag(Wz}? z8zNqeaaXD^3s$;wl2+?8$5Qm-NRfQcEf_&n$n>b32^)zAB3=|>RGskjWntr|JK}); zr-xu@-j_1h5?Bdy^9x79{1YW7!lnV0CQN6L?g+MVBI1RW1VuV0gUD-&^s#fH=~Wlu z>EjYcB|kyri5pRI19O3dQf_+;J1yI8?j0-k`>?#Nt#zwBO^ys90DmMNi)AS4DiX$HHrSL0h}juXkR1V99a{ zakl&%W2RDFP=H|ZF=L*q{o9d;DB1@7=5L}A5%`T%s2|NCOoh|^Iru|u{fqr-1$pmv zxXts=o=%cXbW5VCm=xEGpmsae3FT!Fe7cltWL_& zHypGnDeqQ_+NO{vKcxsRSFH7MkW0hsVob;zmxx+aPJ-LRyN`OjF_Nq6|7kvdbD zg|a3E97Q~r_~eCxk|+&Q88(gdVy0v|n8Ax^ig&i5@PXa z_+V^_Uhd9*HOXhj`6d+{IcIH*hqW2ISRh+@@dBB(=v^5jwhFpz^ESm5x#e$damoF( z#xylp6PD3bMFEOEkzfUUkf2z$F#5ZEY9c;0E0t^MIGlLOFvPVnOw|Q=3>Wur-8!$f zTtt)V(N2-`ZzU}hC?=j>s*0|ee|W2?xhWp~i(M4klP0`ISvNa-PJ(TvnB!VfE7>h`thnsH~CMG?0(X-orV&jrP%y=YeEo>82gEWRo@TWm$dlN^Wi zhGLHiOp1+>)GIZW5#xDxkv7FT+WS{~#m|BHWwyG;9djzwXwf-&J7)gqZ zq6WRJv}I{wlg;P@P)FsvB#l76V33^|1v~~C>R`s8+Qo&UF*kWH3oUE45`iC{;&;Tg zQsNTA9NHmlsmap8S3=WZOMyI(d|2#Ud>`h>;<@hJJ7XdiytFXUF?be5+dT<3!Sv7) zUyISwU+n00@7Quw(vuiw5SmglV&ly)M_Ad46&60Rre(s@=@zOw5r$8epASo{h^Y{l zrfTn#U~MS|&&3rX$CoL5kS>TQcmiWMUHqwvJ)_aD7vF~=`vVgFa`HBfjBDuGAx$^d#m=+QY2}~ikr;gBlC##V2^=_XIP;k zYt+~rqe?7*eHZSWMpuz(UgqI>`K@73N!eH zZSX8lcOy%P-%F?*8!m#>qzGohDdJYiSB0nDXog$j|5tl&9TZphwTT8of(32dwSk}s z5Zv9}Em(jM+&x%ucXtU60U8U9ySqEVT^eUP$?tu?nLGd8x^rv(__|J=I^Cyw%i4Rb zZEHP`4u;&&X38Dy@A>T3!6yyUZJe|Yk?Nb5xq-CP%yLa6V%|~aWql^k!l!g;j-W%)Rs%5tf>5N+Vd144x6~zK^k$^9%i>%c3o zxf7wJr`TBfCBbD|2QVwUrDI3;rJK56l^1T=T$XL}*Oswk zq4%j3rbvU-Qw@Z!#UjOPHS1~Yv1;JPBaKA}K%j5=2A~F)j{AyWpdJCCT2lT;nc9ok zBcO^ElZvrfVaM>beW;uP_aI}^fh{+eDC>wS-=J#4xW(u%j|7eE8&ir5r)~P@g^%O_bUUn&Gm7D`NlFMb3+U?q~25?-rHlK<(S4NQb==>;%+cEL-9T zgo(mD{Vd9YV1lp+#N@AtCiE~U(JC=P(V2aU`Do`aMPEkUy@wbJR()rOVU$G^ujMT7 zpJ&)I2*$}~x2t*Q`-NtWJN<}y>oP@f7_L^Nym2^2(w6PP>6lbRo&0;Ggxjqr$DrP9r;t3*xHF-ty=SfA0; zoMU-tQtEn<``BKFdGZUgTq$TD)I*`oMig)uQG7x#w3GzXUN{5C_UVHd8`D0>`h=X6 zQaii;ov$t#v{1+eTWqOCBjZ#+T#T~`xTP&&mafT=0%Z*i$)6NR~1YSQC87l9Mz|IaQA!PV<0E# zxTkzFHeXQlL{%d7pnLy<3oGTTB$s_Nte;#t!chB_o+M(f`xVn(o zG*fUJm}8MwPkrRUAJ$idtVJfO=hFU`MLFI^7>DvMB6nmF-ePQAnr=n8%#6H}LOaL6 zS8~1nyR4wZeqCt(s+;t@e9>({+mw0jiyFQR7@o2*1#(^MTLTQoa}h5nL^!oQ4n?zk z@FF()m(Oy7MWK%GH&jHM5f=KLpWncxrISM?!z#8NkSeGPt6Wv!z#v*k7DaxirOW*! zBd>pj=4+3f=Kd}fy(L1@SB1O|*}$pGjSuti&|FP)px(KKovtoKLGa7_Go^sFIqwHe zD;*73CQuXR5X$psi3LMXn3V8c*UES+W<1#6O*3U<{w01cS&#|SmUyIF*&$VKZ556! z$PLa7RVxh6oe<>o0op;PKz@v_sk+i^XK%b+hOKr%bE(k_R1$)vV%qHP1X5RVHz7va zZDG{G@h)kXn6`}tNoMsyh5QPp71vaGW`qEif0*U_*kLD-^GyzFtDWjUODiB%$XR2K zRe&QyWG3_??}%ICVTlE5gE;_E*c_ zMKVgAl5Yz3KOG%7+O<0`49o`pKvoB$F4+7|kqnH0wWHyj{y%jl7GG(&8QfZCS|(0H zY&TWv`z=ysl`4A0GJ2$AU}{lZ)Buwl85b-&gEJ6U(opyriN7D8NF(6VcZ zwLKTomxgwRy!)eOeH^e*es;S3sqd2^TL-k;C}JKPP?X)%0_&?U#;{t_*;P=UAz~=m z${l<`lzX%LMpp=~Z8N3JScTbP%d>Sh1+RA2pYacH`>(2 zEo4Ed7+)Vv6#b7(1hHX7f>0CeC$KC?aH4*y(auvWA}DGg4-5|K#8wzn^ZNL7hhkSW z&#NoSE7Swls+8ARw!l=i-;JXvvbp`jJ@^z?+}bWBh`7APs-c0}AahYT4w zs0`^PPZF!X3>Vb$l>e5_m?B)+M-D8vUdumSc8nGQ3kOF%oPhnab{tS{B_`eTyuXxC ztS6&C9hmLD2<487Ffr2A%sieb2*dqm-=vyxa|dLB2KFH>Oh|7$>MAQKRh!?Ou4^~c z^c%x=yQOoO4gcwmZE!VyF8~bmKrJOqc~PrBP$SB3m`Wl@o7tfOnWEW#0NFUspN@53 z8Vx$SH-)H0z5i#4|6lmr2Ll_(6)jFP{-65UCkl#*iN!%t{G^{>Ix(C2J%@N*&=sYh z)6Isa561(n|1k>aNrHB|Iy$NUn1TGjws<;~P-x=>0p^sD(7W9qeybNgNWJP+7tj)s z_xa(X914~F@RB;(z1b8Bs{`l>$5+X$Lv-$tB|}D(O=iit;#l>Z6q2%BsNL&5!EFG{{hU+~c@{`KHd$+|a4@nnaiFALIrLKQx*ZpKBH!Me z^~mBR#Kgru!f8go6>cPvmp+M(Pe{mM_kM7VfdAko+eU0kf?M(9(?6B@2>Trq%E=_w zetoo9$3g|r1$&(v09bqp5UXelvNo?J> z_M^UHAKn=sCI9=&zYpih&{w=ny8rg2f8WSLQEN2YD+FdD`~TAn|MiM23Q@Ctoc%6tDA_DXE+xcS6-0ZL2%Q|rNkq^sX3=1Ieb;}Ev=(-P58$Jlk7u*QlFltf^4 z2Z}XVYC?L{1No!Q$xvxn%N889{Ow&Pscmm4lKl=N9A(k9MCT*j1@g+u4qrM~&Wg_8 z&OqT(1P^A)b#iI5dOJqjg#rw_wWXatF4$dYEQw~dj0ullLfHZy;5F#yVCW18iG88v zsW+dm*^Xg|kIX$1_4bo-7)P33dx0?8`I{R$F|g#kE%M+kj)nm@6eY(XtmeWB6;S1u z(HXdt|99Xs!G*!vsxixS`!#;F2qKjqFmkB|022}Cja$-(*|puIc2}5X$ui)ID;HDQ zEw2%;PB5{)_Axoj63Mxd5LgA>U-1gK9XIY+hu1@XBC>seP2ZTShC-|Sstxfs7im_D zd)9LW%M9IvLc_wcLaQ%~Q31haRa>?s>5l(tkUj!fTKcS*WKfKWOP*Pk$Js9^cZjj{tIA}t;nZy1{Uv3x_J zcyx4<&HgA}v;#CWEbM6cAf0~sLDrVhrTV8wM%Hv&W64WjENAE!J@}RR!)YYv z4@r@ee|0rF4lZ`N`%_V8^DLKEhg-^S{+~Z3ZJ)>%_zM0_;q|Y1ya4ROFAUg4f7OX> z2a)P;KFr$>1A?*5b1OebDveyMCnU3Ret(gtgnU6BfEFzwc+&W9(Qu)JFx}V3PWo65 zt){s>dS6cTcXVQS1onW@TTJyXm6Is^VI4pGGQc z#pYQzm5C9_LWIA$%eIfk+LnJblM>>@|102I39|9uiCt8dUC>G({;7&Y!C{$6ydcV- zLef5dLt8>}?o9Gk`8ykn?uFYpH1dFOM-B3j)+B!Xoyo0Y!-&XF^x{vyzg#K9q7a5) zAAZ|DnUni7>pV{vJp7YMOWOsdhuWValD<07og(#x`<+Pzn8Lg&89ILAxl4TcQ~5DX z1kocSu#Ud|)1PkjA+d!gt5J&VPX>)z;qp!|F@t~nZn)PBQFq}BNX}jp&boiA6cH;p z#dLi#kC+~z$|j9I;>d&%d}tK6%RV)4H?`T_7!Ol44M&hFIP|+ngiV6`bl&uHS8xmC ze}%|q4ZG0&spcvb`qG83uVBd%*P}O*m^tp38FczbGWci2k4Ay>)B7@UjMHjDKPF>}c{JL^dj4=os_E6^gL$#r109 zAwNz&bv)X_Zs~97|Cf98B;4;2s*4Uii)ds|*>U5qJpaoy|DDwTe{okPU#ko*5+3&c zoj6$(!fB;z(`lv19j;-=y!Jw+0a%%TKsmGG%hOG-0Ew!;ek*{~4Kad;QKvrLX(z*_ zg4*+0VAoN|ne*QYpvY%8w5zPa{YR{{FqivzN}kUhV00}p;di& zPw1>yyA?Y9-BaK_6V)m~6l-in2TM>Kg_!O9&VtYL=F~?nozqUD#Bd=C(ImDFio5E{ zVzsWlF-Jo*x>%b)c+z~*YN1WrG392r3x!_NXl{tEY6{%E;dyqs^9>^Pc6Ga%#|89h z!$K(*q;`HAO)*$BL>l{7vxrdmvq!QtOgSBrHaxaa=NXB;+{cma+mIjMGP5$-^NceK~ROa-uV2C11os1kJq`%g#pSbjveo=F3ctI%dr;|Z>U%?QW*jNJA*A>I z1pGX#3J~SF!2&~A=U3xy<#7*p9)^&kFvMOj$OP&7-GxoKBlN6&!#7C@#f{~_?5FOhjE^L`G`i~ zR!Ty5`nxQkH6Y-6Cc7(YitJkJpK8;9JG>pe(bc=W;IU+_u9!AU3)?E^hM&Bki~~3p@7_)y2KYED ziSsPV6FKqQXMW};Xa;9JT^21Xc8ug%m2IfDUoaprSsu!Mj`Jxw1}^k_F7!Pt+{l0y z9Y*;;G^WKW%JiI1#9Ug!Jd3ir1?k*Mgp*9oaxc*(Rryh(1&+Y{YlZ7XM{3wV{~b(-lE$ z^oHHsFy~)*AGvy)HDWiSW6O2AMLh!<-uCO4qdF8u*(35OLI6oO%g&>A!-@i(o>hC0E!Ugmqc*e(+d zQq0?8o4p^}V#<2sBy80lChw+Pe2Er#`7RZ}q2w_V_l3Rb%NPwu(a93N*2M1nte5!K zvfIY5c;MgW$NkY36tS=Uzr?zJB8owrBj>GDxS_zTVB~?-sFY~_k>1P?xwXV$SFMsX z$FsSfvOQ1p{9bqQQ`hl<4k50Sux1fE0jyUl-c7Nsa{13*Si5F-*GD|8+z}70&~t>k*=}3|Ps%u$ zTb@^Yfmw9a)YQ*`8}pvf#X|FiCz~DQ*FuWz;`WhherWQz}Speym2Xgo9UWCuf> zCw_kM=I7CRmlv70ORpRNYr}<$F z@U+CP=%@(6m#^>B5Dc4#<9h3S29$3~Q z3{stN#{y&-c0NtQ&y)U_#L}GbYZZLTep#U49afyp52r-$2XL*gX_yK`43e$=n=f_= z)ng;KyKW;VPRlL%A(bq?x4DaQ6bUGm1PR*WA(=ZBuQu4m&9Mx@Q zpF)nH^rr(mLD{B1CagNByyk{(u9>RLLC2BNTLr9F4Bnr2++7Z@DdJHjNtcN8(Dz6+ zYRtKhp!CQS@6NPSc59=9Ql>Vor*?ASCJ5;$`KKeeE>|5o1R%v&{7#_Yvs~oI_RM?{ zw-dqE`sf*E9>?8N3&+%(N`meI_+$5)T`mcj^oR4r;t&%l1F2y(PU_IYei}ab4elK&Pb?tuhYA+~$;LEPe zQ*J=!!VJ+}eoDB%wh1%BIW!kll)bfzkeym$QKhhBfs zfD5YeGSexEIHxw*b9n2XSH+ce=mDR&#D4}YRye$Td75fVD4VM*$85CUu#?##v(eaM z`-=vFm-;f9;I$GMdFk$xhE*FV6^THc0hOu9%`z5ljhKJ&Yx9o84bCLNG(#}QY#g30 z&AXUeQXpv-Mb2thF0Uo%2DgPYgrQv_q#~f25Tdc^!s*S_CLfnSmL0L|;>Ll*G1vem z@;I_TfF#o=i%Q`?qF!^5-wDVOT*_b^y&>&$g7Q*e(-K6zZ=4%(=sT4M@X6s7yh#uB z3}oo=CbMT>KkN_keezrOxo825{%EXN<4$&5>NnYrlBKZzl#8yUnsRfszU*|vrEf^? zaGJ-Ader_HBAc5Dg^j;>3i6;^w= z%he5S0axF=i3-|j*VTvxMf*kg;eE!POJM2e@rOv*R{~IU8m_l)dFj&P2eUTmQ0ok9 zosYETzWT^Ege60XMy`V5J6{}LR(~JvD&TfugghJ7+~MMM^P^zBkjtmLn6&kwK6dVo zIIqIklPBg2JD`g1JV+Vx>GSFvyGCt$r0#RjJnX%qt=LCf>Gh$<@pUdbE|nt=%HVge zOZK4S)+-LfJzsnlKtB~F*8ucIw}_AE(2oDk;I46x@*InWB=k@U4)7H`=XNy=?gAgdfz22+y0xz?zHGU~FMj?0E+(~R$cN1POvNC)T{X=QUy zTtW4Q6KHigkHk`V@aQM>RdaD35jl5XEzFK*F^^f3^ql8a8hFsT;CUFs*~L;PN#kQH zdles^@xx2!2C+7Z;P?0fZKHlQQm%imRC$>Tbxp@G?l-h>JN%NV zFZ?ylb33kJUyo>{NIu0bD<7e&u^8^RU-7SGj6^PII*YHO0uUW-&UZAdoF)02%E!xZip77@O zD^BPWsXfu5K<*bAz)|b#bv2iQA*Xz;D^(}KxP@5+h<*G+%9IBqTwM-Hri51M@&n2C zntKI#;O&|}8q+=Wc6ZJTs00Jc=$0%o7hjGW%~;$BU{ib8Re6N{#4)Okb`E(^iYoX5 z9)7%oE~l(W*v!ZHYr%;DF7Q^Ld3j71x@TdNou_<;uaGK!*b2ZP;vcG>nY zeJ`OCvbyzKf(>4O7LQHwxGgaHj!eV6aBJ;ig}30qSDdL#6yP)Mrt+3Iqd<$seWoa< zPVqsx)F@ZY<}wa&A5=wHj{rhz`(?MWoNzC2=8a*uBux00Jd^m6P}`MOl6=W8HJNV7 z^MhHIm$S>c%LgL)pxfChui$&rDz$p})3R5IS(834g;`BHkMa6#V3g}-2SN?mcrxKg%K9M5cvo5c5^nvBnmWo-b|-ZieoAGpOquDj9o>6{Vy9y#Wi^lT!=>Z z!|I|J!*IjNx@!bl=%=-|M4moqBbnhH$ zI~3vC$;!ZRn}(Hd1_?2D>961AXs?jH<5B*zCFiS4ttj4szxnpVd%Ig|z%(Wm+nfal zBK0K`v`$0;%`O;?m2u<@JxQmI1%OVgSCqEjK*5;?Q4lu}daXh7b+iy(WH=GkkAbCe`sL~+@Ht}l%7Auxk(0k?3Y;O*s) zIP|TF1~@3!G%Y`>!K9}9QR5OePE%2y&w(KW$;n@gEc?5VQ{D`DJ8S;{j z^L`K`40PbJN47H7R|L802Tj;h&O%T3`8Z>Cl#iy5%r8#j(_eBpggDMOYg-P<#4xryH7(nVPAj}zwBerjt0HKx5q~o=Xw1GuPUOd zE9dJ0{d!X|aepf>he$rlAgxtuko?9n<)P#7TE;_p(Mr^2lRz$U%fa``(RO=wh!GcY z;^>+g%cy^&O|KK9+kPZwxpR)KPnbQg(&LNK_Xk?ZP3$cfK1AJ@B~yXmDbl_ z`{uxgnX#xqXYB2F-L`doN&wbZSxMiT3O{56NX*DD)*mIjD=x&Kf%^uaz|l7Yd+)bo zj|y_evo#D#^#{ZoKNUOMD3`0a{iCtD6`hiQV;ojZZ-m*~#!H06i5H zTp#xLGFWE(^D2os-N@%`xYsIN--whe5+oAnrMn$N?eSaku|Nr41Vd~co~=I5PY%ZW z@`1o3qD{|$J6GC0u5|7@inH%Oao~`I9Usm|;k;i17L^ZOdZVyr+X8T=CGWw>**=B+ zj;F#J-tf&Y=h3ZjTW6Il0E1ncn1c&vNypN+s3) zYFt;Y3Qo=kN&G<>s#dB;8hdd}WElhx2IT&d(iq$6NJ)r9xWj_mi~?q4p@6s0fMY|1 z7YVWX+{x?xoP~1vXvZ8rr=H>50+}TBDg^RZ9<0|Y#V(_rQUW%?$qV14cqZJe82#R+ zN4cTW%_1p=RoCXFbb*2)cPW#W@hXyyGxz#G-OE2R+=lDW%{`jZbateBci3@)xT{6f zNu$q+-mianVBg#%JgoIW8x-lA712$ioK8?6zY;4#Lj_hOX?3|`%)~qexL2pV@{N*w zwrl~1-k?#m1p7q@U^n2r%$liotU$4b?vJ}-aKilAOM;To*~)H%nsZWQ9Z{Q{kD5Mj z4>#FqEi7(3waCGdkz|aYAbHZB0k{i@y*?Lh0EmC}jo~*c3k@!kuVGlfHkfTf{2y&RRgBXO$B8I;B)5^ zM2vgG8P2L)r$JAwV+zIE!*<-p)#F`=&8XK>OP|%peF7#sY7?BU7#odbDGMUM-OC}9 zeC1m@urdzUXWp=ycx8^S*W?1+?QvFx>F?8nS_qj*!vTxMmJU+2k(3ZrnOz#;yEa^b ztX(<11Yypv_1PJ=#1`s0L;*uZe03YzZLptT9d}9bEE_};a#Y0HGjhDIL%h%-&2g`c zqxoS-@^d%;Iy>VATX%8d0~l%PpoBk=a^*m(a>=~x4gQj|kZS!x|4LNL0hJWB2zUZJ zYz8Tba@mOD6;DpiiFrsCKN6vY-YILUZD~VL+J==xixI9>mlfR;J)^lOPw~lpt>fuc zhI#MbRY0o|zk)Z!{F9et#bv3B@G@Sa@^wP%cxwX`? zAtd>c=bN}g5!#d*UDC4iqSfzha?-c5`yu<5+(QKWwei;t;#UD7``iPC)TmV4*57@+ zT=;vHD%X^da^v&W9lyTp9FN3b3ui0r3#-`VI|K%wy04HsOV8HMt^QDj!qq}6IGFYuV z$9f+w7Dp|1f3^gLp)rYjuU5MGI3zu2F=B#0npdiE%Fg=HvJWF1SD?d3VS>ohR>8|3 z%hoFb%u@0+kQUzhi1H<1XU6fc{aCcm`@9Kiz~h8xO|RR>+!M3Q>89v9X;Q+1v8^ zA1lp0!W66(o}M)+*50w;ON_9;?oq4SERf@Iq@~ulkoUiz_tbK7z8oR%eL_`tO?~dZ z$YSC-XN#z+rGI9Ln?=i=MDOy|Q{uf}^J=jy?1;eC;9qR7_Tx?-=SDxJEFqD+R61T$ z+bwZ#qeFePY;uw`PfFe_yC2KET&E+}0H=!8pjnVE-bjhMx#Avq1k6)A8x|?Iy+hlA zyy+nD1*ak08}i%Ni-YN=t2Jm7v65fc(_{k)2^IMHIyJr&RIzV@r6lJg2MOx6(Z`R?*D@EQ z$9o&6RyE$uS(>RjHHCI)cH0p!1WRm+(%(H7R`awewI6H+o#zCuMduEL7_yTq*Lt!F1Hf zo?+0neJU&i`_R4pQ1b8j;lA)!Ar~rA#f2!Tk^Aa6BIhl~mP6eoHdBhT*ZyUZ83RO- zdKje(n0E40y46E~Ea0Zo_aQq$-oWz9T8@mNP^TyP%SokU*fHXW3^`tfy((wS)<&JK;Fm3b}Pvx(p{o zjugj5xjRI~NO(nSf4Ymk>LcN5yHI01z!J<-qN0zQLVo!6I-kGBeHD^Av2%9%53mbmzkcRK-FLUjT&91w1#~B&T~EDx)Gh*Dqhw@@a5TqCZGj{5j71 z2ulw8ri|O#`DI*s=Un-5!K&~Dvl93F#*@xAPol^Q!B#<%Pl{;7blpF_2i3b!mxVlY zYI))aXGVN;_&b!x2o-Hy^ z8=D(`pW5Gd&~@>id34QKz9=qIXsa^L)piw)dJ_R?16ltC`-J97Ef&JIZ>+RHJTFK3 zYy4Ra8<2^7fXVb^Mey~7X;~bhK!F}nhpbxWnP4 z=~m+5dx;O)u1uo%+=w(Dl-x~kci0Iw6g9aq7ruG8h0@N}LeY}gG52(g1)UrPbzGxd zfiaOMalL?PV6MOF{SL>vv8HEJm$D;o09TSkV@!8k`P#s%kl&_DADMbtt9FTUBZTm7Vdb(;VnbixC2hXE z1bF6WB(G)J*s(1)uHz+&zk_O)F=hVqp~?8^8%yDckJd}1f$`^0vxl;kYy52OJ6hlROw8kBM8iX;r*;gm2(~6_tm+!lHNhl|t++|)Z z)a>^x_8?*V2uG4yd|X;Jo z)T`V8mRr6Nu>I_UBdh;W_#Sw`la`!)EZz7N?4nH7YXk{x%`MNjdw%C+LJrdS0(GS$ z4!t?x<>;)$O5#JLzcl1Kj7_DSV8R(^`m}!H`S1{H+oJ@zB;0wL%LpmvV`{75x?=p! zk}$GBKxa-X2-&cCagMw}Dz2SLqp+|y6$DdfA%|-8~7+t!yn6EkFdRK8fpN}Vhq?pAK2C$5Pr5lHN|nS z{fPJeq2(ju(iL;=C3v!OOGGECCXX&qPe6y0gvTpGS6ZLEDD~|$&B_5}xYQMld`2wg z8HP}r+-fhB8}%(PulzXIN2Hk918w_`aN+`~D3>j9mu-wmciI|t)BC;C4jon}@dKO% zD{|sO%=Gw~ad8mx={@#DDjIQ#4}YwS_94SY>jVX@vmHvP=|X2Xp%O~sTb((1s$~&7 zcV1JKp&w7pUwc&R@FOM%3<#H07tSgxJWqL&-p}IQW$9Mt@uB9U?v)_q4d`*r{4t|ilB|^`O*8aoot!O%z*6ir@U=cOBcRWwBUyA8x%I^EF*U@Z#aG^X zwRctHPR}eAliy!z*y>F!F5tL5z>QXeF6!>*Vq72NLs-nKe2!R>%+-%Z4tys}T~Crr zQCs>0o%gaxPr<_OjB!sw89XkL^2UhPKvu1}p1N4WjeH+tM`Az;c7PK?+*f3~>k68(f)<`=mxaB^4|hJ5I#+2^=7VdQE+-BGJS({K>t(AW`#v_zo2y7-=RJ{qOV9(w zecmUzB=5~4#2xbvbHBy9J{{iGp=a!hOG3NuqRm9vZ#`?`amlRJ3!#e;#S!7=S9ITW zggNa}3~l1HU|Hz~eJCi^;G@GVC_|N;OquLa&Uu{-v1g;#z0aZCE6<^%kE=IX>rZAa zV_={ASne~k3gv;iADR|499vu?q;y!a0ngtIygc{WOgOL9tv(Zg)V)&Vnj%N48k$mUB6p zKiU{+X;HE+PGRoJf$fRk8t=2`2+vz^D!mRxVsD7_RyL@dIj-=wxhFek5Lge1hv%uY zJf#WLp^0eiy(f>~;ovQen)sfd)Nnjnhxc7IE7+2JGj=~sp6wCU$MRGltQCTjsYa^u zj=TQh9SIE?jYH)-!Ocki+M+YQj>WThl(Y>Z}uw3t_PgXKVFG>bZJF``{AG!6@xT%Eh89erJPwgwD9Oj21WA zATH~)?Z3f)63#l}>R%IU&5hI;?`?wB(go%l-JfHt!~;QDO2T~93F+sw^mygdf4j7^ zF~G!6T^Om@j`Bfdd*z#Z>>_}rEn~9*lUBdpBN%kPYtP`UpAUoa$^+UhxT%Rg;FL*w z=08hKTy$)7huxq0CIdbDJu-oB4TGME4}4EgI^;2DS$ywOaW(StokgThdQ7Yl<2w5) zxK2uJ$@ReDgqm})&q?Xg*xu|y-!ThYRy;3;EgK(vmv;ny983n)vh8TDJQP3lYhjBd zGv~;*jEaKL07<3~GE|S&EV(VSAC^*+)uaSTG!?J88I^`QM9nocwT#R@GsTUlqL#Jal_cm!DwLf zVU5g|tJP`d<&fVdUA18>2VKuylGn2Yf%T3TbEbstjbBZQ@6Do%%4yk+s9n?dIx)MMwkYtGWF625b?K_It@-ADgLSRkHI1XXT8Qam z;!C%|*-zL(RvcG~MH+oW`pkWSOSNITC<4(%Q zJN=G~2D-$>IY_>ox3`2l1u~g^UQGdW+UuxAR}RFzFh?M7L^*QfECjtvlm>~Su<+dN zu)0^>9!ob2`|f#(TYQyx^v?bU%7s_`9Q!I9Gn4Wt$(}ZFZ@7W6lb;QDZD9@g-PjH# zQ<~;K-c81EiuFt;Uz;*u_B~&BqbW_yoIg7hw1~Z~ZW6DEqOnU?Fi@EW*W3~MZ{R~o zWxNcb5hB_N^5&eVOOEB^~Oiw2S&IhrtTIXl9I@@Z76H>F4S*uf7~isRzVsSK`P^7T}; zzJ%bWJjuX4Ev%anB6AYv*A5cXq`JqAlr|EFD!Wk+Jno!5z&l^U1jT`#w{HRQ-C6dG zREM9C(pq!U^G@z?9j4zpZT(1Qz6BXRyakdTSZcpO*{C>94Phw_tNi9d4h%mj*Xmkn z_nUh?QB(o2N9K`bb1&*S8yS!z`$2h_{lmw~1&Km%kIpqSk9m@ArznL-uX4;8~l2^0B2omiqqWjLj65f!s+a z;PG60i@Bq!sY(R_)U9k#qzmjMN zn0~lSD-~t7via*3_zXAZB$MpQDSt8QGJ8kGuqvCioiRl)Q?L2)i|Tl_V%q&jHC8UV;pi&st(S zn8-K%(*v@jq$*5El1?%&trL^aN)TiUi%%}-rPn`lWlUQh7XxcTi7e{y9#e}LcDe(5 zB};5YnT2+?^ycx@Xn8SN5D0bqAAwT~`e%6sgnHn1^p=-;-q2*ExZiFkIhEB+j7vvm z7qKG!PMY%cv=i-sQvD7+kIHh`3$k1kr$WBXZ3);5*v5vkIggfZIDY!$Q42<>qO2jg zjn-w9J7sP^r|%s36EytWIaW{}0Ijw>ZWiEAbKF8*GoD$YfX$>)_T@ge8>@xCTs5rFuS@0mnEIZEGLsw|ux>lis;lMa20zSp8>p_}3RV z!v8D3X!1-H#s}0dBL#?55-+x!q63$W9tVrJJm(Vr2^jk)jOMSu?tkC^TLS+-NkEVr b@xkYHVUN{BW{=ei=toLSUbI5^qu>7kL1S6y literal 0 HcmV?d00001 diff --git a/cglm/docs/source/color.rst b/cglm/docs/source/color.rst new file mode 100644 index 0000000..62a4dc1 --- /dev/null +++ b/cglm/docs/source/color.rst @@ -0,0 +1,34 @@ +.. default-domain:: C + +color +================================================================================ + +Header: cglm/color.h + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_luminance` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: float glm_luminance(vec3 rgb) + + | averages the color channels into one value + + This function uses formula in COLLADA 1.5 spec which is + + .. code-block:: text + + luminance = (color.r * 0.212671) + + (color.g * 0.715160) + + (color.b * 0.072169) + + It is based on the ISO/CIE color standards (see ITU-R Recommendation BT.709-4), + that averages the color channels into one value + + Parameters: + | *[in]* **rgb** RGB color diff --git a/cglm/docs/source/conf.py b/cglm/docs/source/conf.py new file mode 100644 index 0000000..fdf39a0 --- /dev/null +++ b/cglm/docs/source/conf.py @@ -0,0 +1,207 @@ +# -*- coding: utf-8 -*- +# +# cglm documentation build configuration file, created by +# sphinx-quickstart on Tue Jun 6 20:31:05 2017. +# +# This file is execfile()d with the current directory set to its +# containing dir. +# +# Note that not all possible configuration values are present in this +# autogenerated file. +# +# All configuration values have a default; values that are commented out +# serve to show the default. + +# If extensions (or modules to document with autodoc) are in another directory, +# add these directories to sys.path here. If the directory is relative to the +# documentation root, use os.path.abspath to make it absolute, like shown here. +# +# import os +# import sys +# sys.path.insert(0, os.path.abspath('.')) + + +# -- General configuration ------------------------------------------------ + +# If your documentation needs a minimal Sphinx version, state it here. +# +# needs_sphinx = '3.0' + +# Add any Sphinx extension module names here, as strings. They can be +# extensions coming with Sphinx (named 'sphinx.ext.*') or your custom +# ones. +extensions = [ + 'sphinx.ext.doctest', + 'sphinx.ext.todo', + 'sphinx.ext.coverage', + 'sphinx.ext.mathjax', + 'sphinx.ext.ifconfig', + 'sphinx.ext.viewcode', + 'sphinx.ext.githubpages' +] + +# Add any paths that contain templates here, relative to this directory. +templates_path = ['_templates'] + +# The suffix(es) of source filenames. +# You can specify multiple suffix as a list of string: +# +# source_suffix = ['.rst', '.md'] +source_suffix = '.rst' + +# The master toctree document. +master_doc = 'index' + +# General information about the project. +project = u'cglm' +copyright = u'2017, Recep Aslantas' +author = u'Recep Aslantas' + +# The version info for the project you're documenting, acts as replacement for +# |version| and |release|, also used in various other places throughout the +# built documents. +# +# The short X.Y version. +version = u'0.9.6' +# The full version, including alpha/beta/rc tags. +release = u'0.9.6' + +# The language for content autogenerated by Sphinx. Refer to documentation +# for a list of supported languages. +# +# This is also used if you do content translation via gettext catalogs. +# Usually you set "language" from the command line for these cases. +language = 'en' + +# List of patterns, relative to source directory, that match files and +# directories to ignore when looking for source files. +# This patterns also effect to html_static_path and html_extra_path +exclude_patterns = [] + +# The name of the Pygments (syntax highlighting) style to use. +pygments_style = 'sphinx' + +# If true, `todo` and `todoList` produce output, else they produce nothing. +todo_include_todos = False + + +# -- Options for HTML output ---------------------------------------------- + +# The theme to use for HTML and HTML Help pages. See the documentation for +# a list of builtin themes. +# +html_theme = 'sphinx_rtd_theme' +pygments_style = 'monokai' + +# Theme options are theme-specific and customize the look and feel of a theme +# further. For a list of options available for each theme, see the +# documentation. +# +# html_theme_options = {} + +html_theme_options = { + # 'github_banner': 'true', + # 'github_button': 'true', + # 'github_user': 'recp', + # 'github_repo': 'cglm', + # 'travis_button': 'true', + # 'show_related': 'true', + # 'fixed_sidebar': 'true' +} + +# Add any paths that contain custom static files (such as style sheets) here, +# relative to this directory. They are copied after the builtin static files, +# so a file named "default.css" will overwrite the builtin "default.css". +html_static_path = ['sphinx-static'] + +# Add customm CSS and JS files +html_css_files = ['theme_overrides.css'] +html_js_files = [] + +# -- Options for HTMLHelp output ------------------------------------------ + +# Output file base name for HTML help builder. +htmlhelp_basename = 'cglmdoc' + + +# -- Options for LaTeX output --------------------------------------------- + +latex_elements = { + # The paper size ('letterpaper' or 'a4paper'). + # + # 'papersize': 'letterpaper', + + # The font size ('10pt', '11pt' or '12pt'). + # + # 'pointsize': '10pt', + + # Additional stuff for the LaTeX preamble. + # + # 'preamble': '', + + # Latex figure (float) alignment + # + # 'figure_align': 'htbp', +} + +# Grouping the document tree into LaTeX files. List of tuples +# (source start file, target name, title, +# author, documentclass [howto, manual, or own class]). +latex_documents = [ + (master_doc, 'cglm.tex', u'cglm Documentation', + u'Recep Aslantas', 'manual'), +] + + +# -- Options for manual page output --------------------------------------- + +# One entry per manual page. List of tuples +# (source start file, name, description, authors, manual section). +man_pages = [ + (master_doc, 'cglm', u'cglm Documentation', + [author], 1) +] + + +# -- Options for Texinfo output ------------------------------------------- + +# Grouping the document tree into Texinfo files. List of tuples +# (source start file, target name, title, author, +# dir menu entry, description, category) +texinfo_documents = [ + (master_doc, 'cglm', u'cglm Documentation', + author, 'cglm', 'One line description of project.', + 'Miscellaneous'), +] + +# -- Options for Epub output ------------------------------------------------- + +# Bibliographic Dublin Core info. +epub_title = project +epub_author = author +epub_publisher = author +epub_copyright = copyright + +# The unique identifier of the text. This can be a ISBN number +# or the project homepage. +# +# epub_identifier = '' + +# A unique identification for the text. +# +# epub_uid = '' + +# A list of files that should not be packed into the epub file. +epub_exclude_files = ['search.html'] + + +# -- Extension configuration ------------------------------------------------- + +# -- Options for todo extension ---------------------------------------------- + +# If true, `todo` and `todoList` produce output, else they produce nothing. +todo_include_todos = True + +# -- Options for the C domain ------------------------------------------------ + +c_id_attributes = ['__restrict'] diff --git a/cglm/docs/source/curve.rst b/cglm/docs/source/curve.rst new file mode 100644 index 0000000..26c9b75 --- /dev/null +++ b/cglm/docs/source/curve.rst @@ -0,0 +1,41 @@ +.. default-domain:: C + +Curve +================================================================================ + +Header: cglm/curve.h + +Common helpers for common curves. For specific curve see its header/doc +e.g bezier + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_smc` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: float glm_smc(float s, mat4 m, vec4 c) + + | helper function to calculate **S** * **M** * **C** multiplication for curves + + | this function does not encourage you to use SMC, instead it is a helper if you use SMC. + + | if you want to specify S as vector then use more generic glm_mat4_rmc() func. + + | Example usage: + + .. code-block:: c + + Bs = glm_smc(s, GLM_BEZIER_MAT, (vec4){p0, c0, c1, p1}) + + Parameters: + | *[in]* **s** parameter between 0 and 1 (this will be [s3, s2, s, 1]) + | *[in]* **m** basis matrix + | *[out]* **c** position/control vector + + Returns: + scalar value e.g. Bs diff --git a/cglm/docs/source/euler.rst b/cglm/docs/source/euler.rst new file mode 100644 index 0000000..74ba4c2 --- /dev/null +++ b/cglm/docs/source/euler.rst @@ -0,0 +1,182 @@ +.. default-domain:: C + +euler angles +============ + +Header: cglm/euler.h + +You may wonder what **glm_euler_sq** type ( **_sq** stands for sequence ) and +:c:func:`glm_euler_by_order` do. +I used them to convert euler angles in one coordinate system to another. For +instance if you have **Z_UP** euler angles and if you want to convert it +to **Y_UP** axis then :c:func:`glm_euler_by_order` is your friend. For more +information check :c:func:`glm_euler_order` documentation + +You must pass arrays as array, if you use C compiler then you can use something +like this: + +.. code-block:: c + + float pitch, yaw, roll; + mat4 rot; + + /* pitch = ...; yaw = ...; roll = ... */ + glm_euler((vec3){pitch, yaw, roll}, rot); + +Rotation Conveniention +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ +Current *cglm*'s euler functions uses these convention: + +* Tait–Bryan angles (x-y-z convention) +* Intrinsic rotations (pitch, yaw and roll). + This is reserve order of extrinsic (elevation, heading and bank) rotation +* Right hand rule (actually all rotations in *cglm* use **RH**) +* All angles used in *cglm* are **RADIANS** not degrees + + +**NOTE**: The default :c:func:`glm_euler` function is the short name of +:c:func:`glm_euler_xyz` this is why you can't see :c:func:`glm_euler_xyz`. +When you see an euler function which doesn't have any X, Y, Z suffix then +assume that uses **_xyz** (or instead it accept order as parameter). + +If rotation doesn't work properly, your options: + +1. If you use (or paste) degrees convert it to radians before calling an euler function + +.. code-block:: c + + float pitch, yaw, roll; + mat4 rot; + + /* pitch = degrees; yaw = degrees; roll = degrees */ + glm_euler((vec3){glm_rad(pitch), glm_rad(yaw), glm_rad(roll)}, rot); + +2. Convention mismatch. You may have extrinsic angles, + if you do (if you must) then consider to use reverse order e.g if you have + **xyz** extrinsic then use **zyx** + +3. *cglm* may implemented it wrong, consider to create an issue to report it + or pull request to fix it + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Types: + +1. glm_euler_sq + +Functions: + +1. :c:func:`glm_euler_order` +#. :c:func:`glm_euler_angles` +#. :c:func:`glm_euler` +#. :c:func:`glm_euler_xyz` +#. :c:func:`glm_euler_zyx` +#. :c:func:`glm_euler_zxy` +#. :c:func:`glm_euler_xzy` +#. :c:func:`glm_euler_yzx` +#. :c:func:`glm_euler_yxz` +#. :c:func:`glm_euler_by_order` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: glm_euler_sq glm_euler_order(int ord[3]) + + | packs euler angles order to glm_euler_sq enum. + + To use :c:func:`glm_euler_by_order` function you need *glm_euler_sq*. You + can get it with this function. + + You can build param like this: + + | X = 0, Y = 1, Z = 2 + + if you have ZYX order then you pass this: [2, 1, 0] = ZYX. + if you have YXZ order then you pass this: [1, 0, 2] = YXZ + + As you can see first item specifies which axis will be first then the + second one specifies which one will be next an so on. + + Parameters: + | *[in]* **ord** euler angles order [Angle1, Angle2, Angle2] + + Returns: + packed euler order + +.. c:function:: void glm_euler_angles(mat4 m, vec3 dest) + + | extract euler angles (in radians) using xyz order + + Parameters: + | *[in]* **m** affine transform + | *[out]* **dest** angles vector [x, y, z] + +.. c:function:: void glm_euler(vec3 angles, mat4 dest) + + | build rotation matrix from euler angles + + this is alias of glm_euler_xyz function + + Parameters: + | *[in]* **angles** angles as vector [Xangle, Yangle, Zangle] + | *[in]* **dest** rotation matrix + +.. c:function:: void glm_euler_xyz(vec3 angles, mat4 dest) + + | build rotation matrix from euler angles + + Parameters: + | *[in]* **angles** angles as vector [Xangle, Yangle, Zangle] + | *[in]* **dest** rotation matrix + +.. c:function:: void glm_euler_zyx(vec3 angles, mat4 dest) + + | build rotation matrix from euler angles + + Parameters: + | *[in]* **angles** angles as vector [Xangle, Yangle, Zangle] + | *[in]* **dest** rotation matrix + +.. c:function:: void glm_euler_zxy(vec3 angles, mat4 dest) + + | build rotation matrix from euler angles + + Parameters: + | *[in]* **angles** angles as vector [Xangle, Yangle, Zangle] + | *[in]* **dest** rotation matrix + +.. c:function:: void glm_euler_xzy(vec3 angles, mat4 dest) + + | build rotation matrix from euler angles + + Parameters: + | *[in]* **angles** angles as vector [Xangle, Yangle, Zangle] + | *[in]* **dest** rotation matrix + +.. c:function:: void glm_euler_yzx(vec3 angles, mat4 dest) + + build rotation matrix from euler angles + + Parameters: + | *[in]* **angles** angles as vector [Xangle, Yangle, Zangle] + | *[in]* **dest** rotation matrix + +.. c:function:: void glm_euler_yxz(vec3 angles, mat4 dest) + + | build rotation matrix from euler angles + + Parameters: + | *[in]* **angles** angles as vector [Xangle, Yangle, Zangle] + | *[in]* **dest** rotation matrix + +.. c:function:: void glm_euler_by_order(vec3 angles, glm_euler_sq ord, mat4 dest) + + | build rotation matrix from euler angles with given euler order. + + Use :c:func:`glm_euler_order` function to build *ord* parameter + + Parameters: + | *[in]* **angles** angles as vector [Xangle, Yangle, Zangle] + | *[in]* **ord** euler order + | *[in]* **dest** rotation matrix diff --git a/cglm/docs/source/features.rst b/cglm/docs/source/features.rst new file mode 100644 index 0000000..f3dee71 --- /dev/null +++ b/cglm/docs/source/features.rst @@ -0,0 +1,29 @@ +Features +================================================================================ + +* **scalar** and **simd** (sse, avx, neon, wasm...) optimizations +* option to use different clipspaces e.g. Left Handed, Zero-to-One... (currently right handed negative-one is default) +* array api and struct api, you can use arrays or structs. +* general purpose matrix operations (mat4, mat3) +* chain matrix multiplication (square only) +* general purpose vector operations (cross, dot, rotate, proj, angle...) +* affine transformations +* matrix decomposition (extract rotation, scaling factor) +* optimized affine transform matrices (mul, rigid-body inverse) +* camera (lookat) +* projections (ortho, perspective) +* quaternions +* euler angles / yaw-pitch-roll to matrix +* extract euler angles +* inline or pre-compiled function call +* frustum (extract view frustum planes, corners...) +* bounding box (AABB in Frustum (culling), crop, merge...) +* 2d bounding box (crop, merge...) +* bounding sphere +* project, unproject +* easing functions +* curves +* curve interpolation helpers (SMC, deCasteljau...) +* helpers to convert cglm types to Apple's simd library to pass cglm types to Metal GL without packing them on both sides +* ray intersection helpers +* and others... diff --git a/cglm/docs/source/frustum.rst b/cglm/docs/source/frustum.rst new file mode 100644 index 0000000..956f48c --- /dev/null +++ b/cglm/docs/source/frustum.rst @@ -0,0 +1,168 @@ +.. default-domain:: C + +frustum +============= + +Header: cglm/frustum.h + +cglm provides convenient functions to extract frustum planes, corners... +All extracted corners are **vec4** so you must create array of **vec4** +not **vec3**. If you want to store them to save space you msut convert them +yourself. + +**vec4** is used to speed up functions need to corners. This is why frustum +functions use *vec4* instead of *vec3* + +Currently related-functions use [-1, 1] clip space configuration to extract +corners but you can override it by prodiving **GLM_CUSTOM_CLIPSPACE** macro. +If you provide it then you have to all bottom macros as *vec4* + +Current configuration: + +.. code-block:: c + + /* near */ + GLM_CSCOORD_LBN {-1.0f, -1.0f, -1.0f, 1.0f} + GLM_CSCOORD_LTN {-1.0f, 1.0f, -1.0f, 1.0f} + GLM_CSCOORD_RTN { 1.0f, 1.0f, -1.0f, 1.0f} + GLM_CSCOORD_RBN { 1.0f, -1.0f, -1.0f, 1.0f} + + /* far */ + GLM_CSCOORD_LBF {-1.0f, -1.0f, 1.0f, 1.0f} + GLM_CSCOORD_LTF {-1.0f, 1.0f, 1.0f, 1.0f} + GLM_CSCOORD_RTF { 1.0f, 1.0f, 1.0f, 1.0f} + GLM_CSCOORD_RBF { 1.0f, -1.0f, 1.0f, 1.0f} + + +Explain of short names: + * **LBN**: left bottom near + * **LTN**: left top near + * **RTN**: right top near + * **RBN**: right bottom near + * **LBF**: left bottom far + * **LTF**: left top far + * **RTF**: right top far + * **RBF**: right bottom far + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +.. code-block:: c + + GLM_LBN 0 /* left bottom near */ + GLM_LTN 1 /* left top near */ + GLM_RTN 2 /* right top near */ + GLM_RBN 3 /* right bottom near */ + + GLM_LBF 4 /* left bottom far */ + GLM_LTF 5 /* left top far */ + GLM_RTF 6 /* right top far */ + GLM_RBF 7 /* right bottom far */ + + GLM_LEFT 0 + GLM_RIGHT 1 + GLM_BOTTOM 2 + GLM_TOP 3 + GLM_NEAR 4 + GLM_FAR 5 + +Functions: + +1. :c:func:`glm_frustum_planes` +#. :c:func:`glm_frustum_corners` +#. :c:func:`glm_frustum_center` +#. :c:func:`glm_frustum_box` +#. :c:func:`glm_frustum_corners_at` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_frustum_planes(mat4 m, vec4 dest[6]) + + | extracts view frustum planes + + planes' space: + - if m = proj: View Space + - if m = viewProj: World Space + - if m = MVP: Object Space + + You probably want to extract planes in world space so use viewProj as m + Computing viewProj: + + .. code-block:: c + + glm_mat4_mul(proj, view, viewProj); + + Exracted planes order: [left, right, bottom, top, near, far] + + Parameters: + | *[in]* **m** matrix + | *[out]* **dest** exracted view frustum planes + +.. c:function:: void glm_frustum_corners(mat4 invMat, vec4 dest[8]) + + | extracts view frustum corners using clip-space coordinates + + corners' space: + - if m = invViewProj: World Space + - if m = invMVP: Object Space + + You probably want to extract corners in world space so use **invViewProj** + Computing invViewProj: + + .. code-block:: c + + glm_mat4_mul(proj, view, viewProj); + ... + glm_mat4_inv(viewProj, invViewProj); + + if you have a near coord at **i** index, + you can get it's far coord by i + 4; + follow example below to understand that + + For instance to find center coordinates between a near and its far coord: + + .. code-block:: c + + for (j = 0; j < 4; j++) { + glm_vec3_center(corners[i], corners[i + 4], centerCorners[i]); + } + + corners[i + 4] is far of corners[i] point. + + Parameters: + | *[in]* **invMat** matrix + | *[out]* **dest** exracted view frustum corners + +.. c:function:: void glm_frustum_center(vec4 corners[8], vec4 dest) + + | finds center of view frustum + + Parameters: + | *[in]* **corners** view frustum corners + | *[out]* **dest** view frustum center + +.. c:function:: void glm_frustum_box(vec4 corners[8], mat4 m, vec3 box[2]) + + | finds bounding box of frustum relative to given matrix e.g. view mat + + Parameters: + | *[in]* **corners** view frustum corners + | *[in]* **m** matrix to convert existing conners + | *[out]* **box** bounding box as array [min, max] + +.. c:function:: void glm_frustum_corners_at(vec4 corners[8], float splitDist, float farDist, vec4 planeCorners[4]) + + | finds planes corners which is between near and far planes (parallel) + + this will be helpful if you want to split a frustum e.g. CSM/PSSM. This will + find planes' corners but you will need to one more plane. + Actually you have it, it is near, far or created previously with this func ;) + + Parameters: + | *[in]* **corners** frustum corners + | *[in]* **splitDist** split distance + | *[in]* **farDist** far distance (zFar) + | *[out]* **planeCorners** plane corners [LB, LT, RT, RB] diff --git a/cglm/docs/source/getting_started.rst b/cglm/docs/source/getting_started.rst new file mode 100644 index 0000000..143b192 --- /dev/null +++ b/cglm/docs/source/getting_started.rst @@ -0,0 +1,105 @@ +Getting Started +================================ + +Types: +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +**cglm** uses **glm** prefix for all functions e.g. glm_lookat. You can see supported types in common header file: + +.. code-block:: c + :linenos: + + typedef float vec2[2]; + typedef float vec3[3]; + typedef int ivec3[3]; + typedef CGLM_ALIGN_IF(16) float vec4[4]; + typedef vec4 versor; + typedef vec3 mat3[3]; + + #ifdef __AVX__ + typedef CGLM_ALIGN_IF(32) vec4 mat4[4]; + #else + typedef CGLM_ALIGN_IF(16) vec4 mat4[4]; + #endif + +As you can see types don't store extra information in favor of space. +You can send these values e.g. matrix to OpenGL directly without casting or calling a function like *value_ptr* + +Alignment Is Required: +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +**vec4** and **mat4** requires 16 (32 for **mat4** if AVX is enabled) byte alignment because **vec4** and **mat4** operations are vectorized by SIMD instructions (SSE/AVX/NEON). + +**UPDATE:** + By starting v0.4.5 cglm provides an option to disable alignment requirement, it is enabled as default + + | Check :doc:`opt` page for more details + + Also alignment is disabled for older msvc versions as default. Now alignment is only required in Visual Studio 2017 version 15.6+ if CGLM_ALL_UNALIGNED macro is not defined. + +Allocations: +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ +*cglm* doesn't alloc any memory on heap. So it doesn't provide any allocator. +You must allocate memory yourself. You should alloc memory for out parameters too if you pass pointer of memory location. When allocating memory, don't forget that **vec4** and **mat4** require alignment. + +.. note:: Unaligned **vec4** and unaligned **mat4** operations will be supported in the future. Check todo list. + Because you may want to multiply a CGLM matrix with external matrix. + There is no guarantee that non-CGLM matrix is aligned. Unaligned types will have *u* prefix e.g. **umat4** + +Array vs Struct: +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ +*cglm* uses arrays for vector and matrix types. So you can't access individual +elements like vec.x, vec.y, vec.z... You must use subscript to access vector elements +e.g. vec[0], vec[1], vec[2]. + +Also I think it is more meaningful to access matrix elements with subscript +e.g **matrix[2][3]** instead of **matrix._23**. Since matrix is array of vectors, +vectors are also defined as array. This makes types homogeneous. + +**Return arrays?** + +Since C doesn't support return arrays, cglm also doesn't support this feature. + +Function design: +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. image:: cglm-intro.png + :width: 492px  + :height: 297px + :align: center + +cglm provides a few way to call a function to do same operation. + +* Inline - *glm_, glm_u* +* Pre-compiled - *glmc_, glmc_u* + +For instance **glm_mat4_mul** is inline (all *glm_* functions are inline), to make it non-inline (pre-compiled), +call it as **glmc_mat4_mul** from library, to use unaligned version use **glm_umat4_mul** (todo). + +Most functions have **dest** parameter for output. For instance mat4_mul func looks like this: + +.. code-block:: c + + CGLM_INLINE + void + glm_mat4_mul(mat4 m1, mat4 m2, mat4 dest) + +The dest parameter is out parameter. Result will be stored in **dest**. +Also in this case matrix multiplication order is dest = m1 * m2. + +* Changing parameter order will change the multiplication order. +* You can pass all parameter same (this is similar to m1 `*=` m1), you can pass **dest** as m1 or m2 (this is similar to m1 `*=` m2) + +**v** postfix in function names +------------------------------- + +You may see **v** postfix in some function names, v stands for vector. +For instance consider a function that accepts three parameters x, y, z. +This function may be overloaded by **v** postfix to accept vector (vec3) instead of separate parameters. +In some places the v means that it will be apply to a vector. + +**_to** postfix in function names +--------------------------------- + +*_to* version of function will store the result in specified parameter instead of in-out parameter. +Some functions don't have _to prefix but they still behave like this e.g. glm_mat4_mul. diff --git a/cglm/docs/source/index.rst b/cglm/docs/source/index.rst new file mode 100644 index 0000000..1cc3bbe --- /dev/null +++ b/cglm/docs/source/index.rst @@ -0,0 +1,53 @@ +.. cglm documentation master file, created by + sphinx-quickstart on Tue Jun 6 20:31:05 2017. + You can adapt this file completely to your liking, but it should at least + contain the root `toctree` directive. + +cglm Documentation +================================ + +**cglm** is an optimized 3D math library written in C99 (compatible with C89). +It is similar to the original **glm** library, except **cglm** is mainly for +**C**. + +**cglm** stores matrices as column-major order but in the future row-major is +considered to be supported as optional. + +.. toctree:: + :maxdepth: 2 + :caption: Getting Started: + + features + build + getting_started + +.. toctree:: + :maxdepth: 2 + :caption: How To: + + opengl + +.. toctree:: + :maxdepth: 3 + :caption: API: + + api + +.. toctree:: + :maxdepth: 2 + :caption: Options: + + opt + +.. toctree:: + :maxdepth: 2 + :caption: Troubleshooting: + + troubleshooting + +Indices and Tables: +=================== + +* :ref:`genindex` +* :ref:`modindex` +* :ref:`search` diff --git a/cglm/docs/source/io.rst b/cglm/docs/source/io.rst new file mode 100644 index 0000000..3b82380 --- /dev/null +++ b/cglm/docs/source/io.rst @@ -0,0 +1,147 @@ +.. default-domain:: C + +io (input / output e.g. print) +================================================================================ + +Header: cglm/io.h + +There are some built-in print functions which may save your time, +especially for debugging. + +All functions accept **FILE** parameter which makes very flexible. +You can even print it to file on disk. + +In general you will want to print them to console to see results. +You can use **stdout** and **stderr** to write results to console. +Some programs may occupy **stdout** but you can still use **stderr**. +Using **stderr** is suggested. + +Example to print mat4 matrix: + +.. code-block:: c + + mat4 transform; + /* ... */ + glm_mat4_print(transform, stderr); + +.. note:: print functions use **%0.4f** precision if you need more + (you probably will in some cases), you can change it temporary. + cglm may provide precision parameter in the future. + +Changes since **v0.7.3**: +* Now mis-alignment of columns are fixed: larger numbers are printed via %g and others are printed via %f. Column widths are calculated before print. +* Now values are colorful ;) +* Some print improvements +* New options with default values: + +.. code-block:: c + + #define CGLM_PRINT_PRECISION 5 + #define CGLM_PRINT_MAX_TO_SHORT 1e5 + #define CGLM_PRINT_COLOR "\033[36m" + #define CGLM_PRINT_COLOR_RESET "\033[0m" + +* Inline prints are only enabled in DEBUG mode and if **CGLM_DEFINE_PRINTS** is defined. + +Check options page. + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_mat4_print` +#. :c:func:`glm_mat3_print` +#. :c:func:`glm_vec4_print` +#. :c:func:`glm_ivec4_print` +#. :c:func:`glm_vec3_print` +#. :c:func:`glm_ivec3_print` +#. :c:func:`glm_vec2_print` +#. :c:func:`glm_ivec2_print` +#. :c:func:`glm_versor_print` +#. :c:func:`glm_aabb_print` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_mat4_print(mat4 matrix, FILE * __restrict ostream) + + | print matrix to given stream + + Parameters: + | *[in]* **matrix** matrix + | *[in]* **ostream** FILE to write + +.. c:function:: void glm_mat3_print(mat3 matrix, FILE * __restrict ostream) + + | print matrix to given stream + + Parameters: + | *[in]* **matrix** matrix + | *[in]* **ostream** FILE to write + +.. c:function:: void glm_vec4_print(vec4 vec, FILE * __restrict ostream) + + | print vector to given stream + + Parameters: + | *[in]* **vec** vector + | *[in]* **ostream** FILE to write + +.. c:function:: void glm_ivec4_print(ivec4 vec, FILE * __restrict ostream) + + | print vector to given stream + + Parameters: + | *[in]* **vec** vector + | *[in]* **ostream** FILE to write + +.. c:function:: void glm_vec3_print(vec3 vec, FILE * __restrict ostream) + + | print vector to given stream + + Parameters: + | *[in]* **vec** vector + | *[in]* **ostream** FILE to write + +.. c:function:: void glm_ivec3_print(ivec3 vec, FILE * __restrict ostream) + + | print vector to given stream + + Parameters: + | *[in]* **vec** vector + | *[in]* **ostream** FILE to write + +.. c:function:: void glm_vec2_print(vec2 vec, FILE * __restrict ostream) + + | print vector to given stream + + Parameters: + | *[in]* **vec** vector + | *[in]* **ostream** FILE to write + +.. c:function:: void glm_ivec2_print(ivec2 vec, FILE * __restrict ostream) + + | print vector to given stream + + Parameters: + | *[in]* **vec** vector + | *[in]* **ostream** FILE to write + + +.. c:function:: void glm_versor_print(versor vec, FILE * __restrict ostream) + + | print quaternion to given stream + + Parameters: + | *[in]* **vec** quaternion + | *[in]* **ostream** FILE to write + +.. c:function:: void glm_aabb_print(versor vec, const char * __restrict tag, FILE * __restrict ostream) + + | print aabb to given stream + + Parameters: + | *[in]* **vec** aabb (axis-aligned bounding box) + | *[in]* **tag** tag to find it more easily in logs + | *[in]* **ostream** FILE to write diff --git a/cglm/docs/source/ivec2.rst b/cglm/docs/source/ivec2.rst new file mode 100644 index 0000000..0f374f7 --- /dev/null +++ b/cglm/docs/source/ivec2.rst @@ -0,0 +1,260 @@ +.. default-domain:: C + +ivec2 +===== + +Header: cglm/ivec2.h + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. GLM_IVEC2_ONE_INIT +#. GLM_IVEC2_ZERO_INIT +#. GLM_IVEC2_ONE +#. GLM_IVEC2_ZERO + +Functions: + +1. :c:func:`glm_ivec2` +#. :c:func:`glm_ivec2_copy` +#. :c:func:`glm_ivec2_zero` +#. :c:func:`glm_ivec2_one` +#. :c:func:`glm_ivec2_dot` +#. :c:func:`glm_ivec2_cross` +#. :c:func:`glm_ivec2_add` +#. :c:func:`glm_ivec2_adds` +#. :c:func:`glm_ivec2_sub` +#. :c:func:`glm_ivec2_subs` +#. :c:func:`glm_ivec2_mul` +#. :c:func:`glm_ivec2_scale` +#. :c:func:`glm_ivec2_div` +#. :c:func:`glm_ivec2_divs` +#. :c:func:`glm_ivec2_mod` +#. :c:func:`glm_ivec2_distance2` +#. :c:func:`glm_ivec2_distance` +#. :c:func:`glm_ivec2_maxv` +#. :c:func:`glm_ivec2_minv` +#. :c:func:`glm_ivec2_clamp` +#. :c:func:`glm_ivec2_abs` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_ivec2(int * v, ivec2 dest) + + init ivec2 using vec3 or vec4 + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec2_copy(ivec2 a, ivec2 dest) + + copy all members of [a] to [dest] + + Parameters: + | *[in]* **a** source vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec2_zero(ivec2 v) + + set all members of [v] to zero + + Parameters: + | *[out]* **v** vector + +.. c:function:: void glm_ivec2_one(ivec2 v) + + set all members of [v] to one + + Parameters: + | *[out]* **v** vector + +.. c:function:: int glm_ivec2_dot(ivec2 a, ivec2 b) + + dot product of ivec2 + + Parameters: + | *[in]* **a** vector1 + | *[in]* **b** vector2 + + Returns: + dot product + +.. c:function:: int glm_ivec2_cross(ivec2 a, ivec2 b) + + cross product of two vector (RH) + + | ref: http://allenchou.net/2013/07/cross-product-of-2d-vectors/ + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + + Returns: + Z component of cross product + +.. c:function:: void glm_ivec2_add(ivec2 a, ivec2 b, ivec2 dest) + + add vector [a] to vector [b] and store result in [dest] + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec2_adds(ivec2 v, int s, ivec2 dest) + + add scalar s to vector [v] and store result in [dest] + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination + +.. c:function:: void glm_ivec2_sub(ivec2 a, ivec2 b, ivec2 dest) + + subtract vector [b] from vector [a] and store result in [dest] + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec2_subs(ivec2 v, int s, ivec2 dest) + + subtract scalar s from vector [v] and store result in [dest] + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination + +.. c:function:: void glm_ivec2_mul(ivec2 a, ivec2 b, ivec2 dest) + + multiply vector [a] with vector [b] and store result in [dest] + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec2_scale(ivec2 v, int s, ivec2 dest) + + multiply vector [a] with scalar s and store result in [dest] + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination + +.. c:function:: void glm_ivec2_div(ivec2 a, ivec2 b, ivec2 dest) + + div vector with another component-wise division: d = a / b + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** result = (a[0] / b[0], a[1] / b[1], a[2] / b[2]) + +.. c:function:: void glm_ivec2_divs(ivec2 v, int s, ivec2 dest) + + div vector with scalar: d = v / s + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** result = (a[0] / s, a[1] / s, a[2] / s) + +.. c:function:: void glm_ivec2_mod(ivec2 a, ivec2 b, ivec2 dest) + + mod vector with another component-wise modulo: d = a % b + + Parameters: + | *[in]* **a** vector + | *[in]* **b** scalar + | *[out]* **dest** result = (a[0] % b[0], a[1] % b[1]) + +.. c:function:: int glm_ivec2_distance2(ivec2 a, ivec2 b) + + squared distance between two vectors + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + + Returns: + squared distance (distance * distance) + +.. c:function:: float glm_ivec2_distance(ivec2 a, ivec2 b) + + distance between two vectors + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + + Returns: + distance + +.. c:function:: void glm_ivec2_fill(ivec2 v, int val) + + fill a vector with specified value + + Parameters: + | *[out]* **v** vector + | *[in]* **val** value + +.. c:function:: bool glm_ivec2_eq(ivec2 v, int val) + + check if vector is equal to value + + Parameters: + | *[in]* **v** vector + | *[in]* **val** value + +.. c:function:: bool glm_ivec2_eqv(ivec2 v1, ivec2 v2) + + check if vector is equal to another vector + + Parameters: + | *[in]* **vec** vector 1 + | *[in]* **vec** vector 2 + + +.. c:function:: void glm_ivec2_maxv(ivec2 a, ivec2 b, ivec2 dest) + + set each member of dest to greater of vector a and b + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec2_minv(ivec2 a, ivec2 b, ivec2 dest) + + set each member of dest to lesser of vector a and b + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec2_clamp(ivec2 v, int minVal, int maxVal) + + clamp each member of [v] between minVal and maxVal (inclusive) + + Parameters: + | *[in, out]* **v** vector + | *[in]* **minVal** minimum value + | *[in]* **maxVal** maximum value + +.. c:function:: void glm_ivec2_abs(ivec2 v, ivec2 dest) + + absolute value of each vector item + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination vector diff --git a/cglm/docs/source/ivec3.rst b/cglm/docs/source/ivec3.rst new file mode 100644 index 0000000..26be5fa --- /dev/null +++ b/cglm/docs/source/ivec3.rst @@ -0,0 +1,272 @@ +.. default-domain:: C + +ivec3 +===== + +Header: cglm/ivec3.h + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. GLM_IVEC3_ONE_INIT +#. GLM_IVEC3_ZERO_INIT +#. GLM_IVEC3_ONE +#. GLM_IVEC3_ZERO + +Functions: + +1. :c:func:`glm_ivec3` +#. :c:func:`glm_ivec3_copy` +#. :c:func:`glm_ivec3_zero` +#. :c:func:`glm_ivec3_one` +#. :c:func:`glm_ivec3_dot` +#. :c:func:`glm_ivec3_norm2` +#. :c:func:`glm_ivec3_norm` +#. :c:func:`glm_ivec3_add` +#. :c:func:`glm_ivec3_adds` +#. :c:func:`glm_ivec3_sub` +#. :c:func:`glm_ivec3_subs` +#. :c:func:`glm_ivec3_mul` +#. :c:func:`glm_ivec3_scale` +#. :c:func:`glm_ivec3_div` +#. :c:func:`glm_ivec3_divs` +#. :c:func:`glm_ivec3_mod` +#. :c:func:`glm_ivec3_distance2` +#. :c:func:`glm_ivec3_distance` +#. :c:func:`glm_ivec3_fill` +#. :c:func:`glm_ivec3_eq` +#. :c:func:`glm_ivec3_eqv` +#. :c:func:`glm_ivec3_maxv` +#. :c:func:`glm_ivec3_minv` +#. :c:func:`glm_ivec3_clamp` +#. :c:func:`glm_ivec2_abs` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_ivec3(ivec4 v4, ivec3 dest) + + init ivec3 using ivec4 + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec3_copy(ivec3 a, ivec3 dest) + + copy all members of [a] to [dest] + + Parameters: + | *[in]* **a** source vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec3_zero(ivec3 v) + + set all members of [v] to zero + + Parameters: + | *[out]* **v** vector + +.. c:function:: void glm_ivec3_one(ivec3 v) + + set all members of [v] to one + + Parameters: + | *[out]* **v** vector + +.. c:function:: int glm_ivec3_dot(ivec3 a, ivec3 b) + + dot product of ivec3 + + Parameters: + | *[in]* **a** vector1 + | *[in]* **b** vector2 + + Returns: + dot product + +.. c:function:: int glm_ivec3_norm2(ivec3 v) + + norm * norm (magnitude) of vector + + we can use this func instead of calling norm * norm, because it would call + sqrtf function twice but with this func we can avoid func call, maybe this is + not good name for this func + + Parameters: + | *[in]* **v** vector + + Returns: + square of norm / magnitude, cast to an integer + +.. c:function:: int glm_ivec3_norm(ivec3 vec) + + | euclidean norm (magnitude), also called L2 norm + | this will give magnitude of vector in euclidean space + + Parameters: + | *[in]* **vec** vector + +.. c:function:: void glm_ivec3_add(ivec3 a, ivec3 b, ivec3 dest) + + add vector [a] to vector [b] and store result in [dest] + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec3_adds(ivec3 v, int s, ivec3 dest) + + add scalar s to vector [v] and store result in [dest] + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination + +.. c:function:: void glm_ivec3_sub(ivec3 a, ivec3 b, ivec3 dest) + + subtract vector [b] from vector [a] and store result in [dest] + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec3_subs(ivec3 v, int s, ivec3 dest) + + subtract scalar s from vector [v] and store result in [dest] + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination + +.. c:function:: void glm_ivec3_mul(ivec3 a, ivec3 b, ivec3 dest) + + multiply vector [a] with vector [b] and store result in [dest] + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec3_scale(ivec3 v, int s, ivec3 dest) + + multiply vector [a] with scalar s and store result in [dest] + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination + +.. c:function:: void glm_ivec3_div(ivec3 a, ivec3 b, ivec3 dest) + + div vector with another component-wise division: d = a / b + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** result = (a[0] / b[0], a[1] / b[1], a[2] / b[2]) + +.. c:function:: void glm_ivec3_divs(ivec3 v, int s, ivec3 dest) + + div vector with scalar: d = v / s + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** result = (a[0] / s, a[1] / s, a[2] / s) + +.. c:function:: void glm_ivec3_mod(ivec3 a, ivec3 b, ivec3 dest) + + mod vector with another component-wise modulo: d = a % b + + Parameters: + | *[in]* **a** vector + | *[in]* **b** scalar + | *[out]* **dest** result = (a[0] % b[0], a[1] % b[1], a[2] % b[2]) + +.. c:function:: int glm_ivec3_distance2(ivec3 a, ivec3 b) + + squared distance between two vectors + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + + Returns: + squared distance (distance * distance) + +.. c:function:: float glm_ivec3_distance(ivec3 a, ivec3 b) + + distance between two vectors + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + + Returns: + distance + +.. c:function:: void glm_ivec3_fill(ivec3 v, int val) + + fill a vector with specified value + + Parameters: + | *[out]* **v** vector + | *[in]* **val** value + +.. c:function:: bool glm_ivec3_eq(ivec3 v, int val) + + check if vector is equal to value + + Parameters: + | *[in]* **v** vector + | *[in]* **val** value + +.. c:function:: bool glm_ivec3_eqv(ivec3 v1, ivec3 v2) + + check if vector is equal to another vector + + Parameters: + | *[in]* **vec** vector 1 + | *[in]* **vec** vector 2 + +.. c:function:: void glm_ivec3_maxv(ivec3 a, ivec3 b, ivec3 dest) + + set each member of dest to greater of vector a and b + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec3_minv(ivec3 a, ivec3 b, ivec3 dest) + + set each member of dest to lesser of vector a and b + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec3_clamp(ivec3 v, int minVal, int maxVal) + + clamp each member of [v] between minVal and maxVal (inclusive) + + Parameters: + | *[in, out]* **v** vector + | *[in]* **minVal** minimum value + | *[in]* **maxVal** maximum value + +.. c:function:: void glm_ivec3_abs(ivec3 v, ivec3 dest) + + absolute value of each vector item + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination vector diff --git a/cglm/docs/source/ivec4.rst b/cglm/docs/source/ivec4.rst new file mode 100644 index 0000000..26d380e --- /dev/null +++ b/cglm/docs/source/ivec4.rst @@ -0,0 +1,179 @@ +.. default-domain:: C + +ivec4 +===== + +Header: cglm/ivec4.h + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. GLM_IVEC4_ONE_INIT +#. GLM_IVEC4_ZERO_INIT +#. GLM_IVEC4_ONE +#. GLM_IVEC4_ZERO + +Functions: + +1. :c:func:`glm_ivec4` +#. :c:func:`glm_ivec4_copy` +#. :c:func:`glm_ivec4_zero` +#. :c:func:`glm_ivec4_one` +#. :c:func:`glm_ivec4_add` +#. :c:func:`glm_ivec4_adds` +#. :c:func:`glm_ivec4_sub` +#. :c:func:`glm_ivec4_subs` +#. :c:func:`glm_ivec4_mul` +#. :c:func:`glm_ivec4_scale` +#. :c:func:`glm_ivec4_distance2` +#. :c:func:`glm_ivec4_distance` +#. :c:func:`glm_ivec4_maxv` +#. :c:func:`glm_ivec4_minv` +#. :c:func:`glm_ivec4_clamp` +#. :c:func:`glm_ivec4_abs` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_ivec4(ivec3 v3, int last, ivec4 dest) + + init ivec4 using ivec3 + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec4_copy(ivec4 a, ivec4 dest) + + copy all members of [a] to [dest] + + Parameters: + | *[in]* **a** source vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec4_zero(ivec4 v) + + set all members of [v] to zero + + Parameters: + | *[out]* **v** vector + +.. c:function:: void glm_ivec4_one(ivec4 v) + + set all members of [v] to one + + Parameters: + | *[out]* **v** vector + +.. c:function:: void glm_ivec4_add(ivec4 a, ivec4 b, ivec4 dest) + + add vector [a] to vector [b] and store result in [dest] + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec4_adds(ivec4 v, int s, ivec4 dest) + + add scalar s to vector [v] and store result in [dest] + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination + +.. c:function:: void glm_ivec4_sub(ivec4 a, ivec4 b, ivec4 dest) + + subtract vector [b] from vector [a] and store result in [dest] + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec4_subs(ivec4 v, int s, ivec4 dest) + + subtract scalar s from vector [v] and store result in [dest] + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination + +.. c:function:: void glm_ivec4_mul(ivec4 a, ivec4 b, ivec4 dest) + + multiply vector [a] with vector [b] and store result in [dest] + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec4_scale(ivec4 v, int s, ivec4 dest) + + multiply vector [a] with scalar s and store result in [dest] + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination + +.. c:function:: int glm_ivec4_distance2(ivec4 a, ivec4 b) + + squared distance between two vectors + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + + Returns: + squared distance (distance * distance) + +.. c:function:: float glm_ivec4_distance(ivec4 a, ivec4 b) + + distance between two vectors + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + + Returns: + distance + +.. c:function:: void glm_ivec4_maxv(ivec4 a, ivec4 b, ivec4 dest) + + set each member of dest to greater of vector a and b + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec4_minv(ivec4 a, ivec4 b, ivec4 dest) + + set each member of dest to lesser of vector a and b + + Parameters: + | *[in]* **a** first vector + | *[in]* **b** second vector + | *[out]* **dest** destination + +.. c:function:: void glm_ivec4_clamp(ivec4 v, int minVal, int maxVal) + + clamp each member of [v] between minVal and maxVal (inclusive) + + Parameters: + | *[in, out]* **v** vector + | *[in]* **minVal** minimum value + | *[in]* **maxVal** maximum value + +.. c:function:: void glm_ivec4_abs(ivec4 v, ivec4 dest) + + absolute value of each vector item + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination vector diff --git a/cglm/docs/source/mat2.rst b/cglm/docs/source/mat2.rst new file mode 100644 index 0000000..60b2360 --- /dev/null +++ b/cglm/docs/source/mat2.rst @@ -0,0 +1,191 @@ +.. default-domain:: C + +mat2 +==== + +Header: cglm/mat2.h + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. GLM_mat2_IDENTITY_INIT +#. GLM_mat2_ZERO_INIT +#. GLM_mat2_IDENTITY +#. GLM_mat2_ZERO + +Functions: + +1. :c:func:`glm_mat2_copy` +#. :c:func:`glm_mat2_identity` +#. :c:func:`glm_mat2_identity_array` +#. :c:func:`glm_mat2_zero` +#. :c:func:`glm_mat2_mul` +#. :c:func:`glm_mat2_transpose_to` +#. :c:func:`glm_mat2_transpose` +#. :c:func:`glm_mat2_mulv` +#. :c:func:`glm_mat2_scale` +#. :c:func:`glm_mat2_det` +#. :c:func:`glm_mat2_inv` +#. :c:func:`glm_mat2_trace` +#. :c:func:`glm_mat2_swap_col` +#. :c:func:`glm_mat2_swap_row` +#. :c:func:`glm_mat2_rmc` +#. :c:func:`glm_mat2_make` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_mat2_copy(mat2 mat, mat2 dest) + + copy mat2 to another one (dest). + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat2_identity(mat2 mat) + + copy identity mat2 to mat, or makes mat to identity + + Parameters: + | *[out]* **mat** matrix + +.. c:function:: void glm_mat2_identity_array(mat2 * __restrict mat, size_t count) + + make given matrix array's each element identity matrix + + Parameters: + | *[in,out]* **mat** matrix array (must be aligned (16/32) if alignment is not disabled) + | *[in]* **count** count of matrices + +.. c:function:: void glm_mat2_zero(mat2 mat) + + make given matrix zero + + Parameters: + | *[in,out]* **mat** matrix + +.. c:function:: void glm_mat2_mul(mat2 m1, mat2 m2, mat2 dest) + + multiply m1 and m2 to dest + + m1, m2 and dest matrices can be same matrix, it is possible to write this: + + .. code-block:: c + + mat2 m = GLM_mat2_IDENTITY_INIT; + glm_mat2_mul(m, m, m); + + Parameters: + | *[in]* **m1** left matrix + | *[in]* **m2** right matrix + | *[out]* **dest** destination matrix + +.. c:function:: void glm_mat2_transpose_to(mat2 m, mat2 dest) + + transpose mat4 and store in dest + source matrix will not be transposed unless dest is m + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat2_transpose(mat2 m) + + transpose mat2 and store result in same matrix + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat2_mulv(mat2 m, vec2 v, vec2 dest) + + multiply mat2 with vec2 (column vector) and store in dest vector + + Parameters: + | *[in]* **mat** mat2 (left) + | *[in]* **v** vec2 (right, column vector) + | *[out]* **dest** destination (result, column vector) + +.. c:function:: void glm_mat2_scale(mat2 m, float s) + + multiply matrix with scalar + + Parameters: + | *[in, out]* **m** matrix + | *[in]* **s** scalar + +.. c:function:: float glm_mat2_det(mat2 mat) + + returns mat2 determinant + + Parameters: + | *[in]* **mat** matrix + + Returns: + mat2 determinant + +.. c:function:: void glm_mat2_inv(mat2 mat, mat2 dest) + + inverse mat2 and store in dest + + Parameters: + | *[in]* **mat** matrix + | *[out]* **dest** destination (inverse matrix) + +.. c:function:: void glm_mat2_trace(mat2 m) + + | sum of the elements on the main diagonal from upper left to the lower right + + Parameters: + | *[in]* **m** matrix + + Returns: + trace of matrix + +.. c:function:: void glm_mat2_swap_col(mat2 mat, int col1, int col2) + + swap two matrix columns + + Parameters: + | *[in, out]* **mat** matrix + | *[in]* **col1** col1 + | *[in]* **col2** col2 + +.. c:function:: void glm_mat2_swap_row(mat2 mat, int row1, int row2) + + swap two matrix rows + + Parameters: + | *[in, out]* **mat** matrix + | *[in]* **row1** row1 + | *[in]* **row2** row2 + +.. c:function:: float glm_mat2_rmc(vec2 r, mat2 m, vec2 c) + + | **rmc** stands for **Row** * **Matrix** * **Column** + + | helper for R (row vector) * M (matrix) * C (column vector) + + | the result is scalar because R * M = Matrix1x2 (row vector), + | then Matrix1x2 * Vec2 (column vector) = Matrix1x1 (Scalar) + + Parameters: + | *[in]* **r** row vector or matrix1x2 + | *[in]* **m** matrix2x2 + | *[in]* **c** column vector or matrix2x1 + + Returns: + scalar value e.g. Matrix1x1 + +.. c:function:: void glm_mat2_make(const float * __restrict src, mat2 dest) + + Create mat2 matrix from pointer + + .. note:: **@src** must contain at least 4 elements. + + Parameters: + | *[in]* **src** pointer to an array of floats + | *[out]* **dest** destination matrix2x2 diff --git a/cglm/docs/source/mat2x3.rst b/cglm/docs/source/mat2x3.rst new file mode 100644 index 0000000..06780d6 --- /dev/null +++ b/cglm/docs/source/mat2x3.rst @@ -0,0 +1,140 @@ +.. default-domain:: C + +mat2x3 +====== + +Header: cglm/mat2x3.h + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. GLM_MAT2X3_ZERO_INIT +#. GLM_MAT2X3_ZERO + +Functions: + +1. :c:func:`glm_mat2x3_copy` +#. :c:func:`glm_mat2x3_zero` +#. :c:func:`glm_mat2x3_make` +#. :c:func:`glm_mat2x3_mul` +#. :c:func:`glm_mat2x3_mulv` +#. :c:func:`glm_mat2x3_transpose` +#. :c:func:`glm_mat2x3_scale` + +Represented +~~~~~~~~~~~ + +.. csv-table:: mat2x3 + :header: "", "column 1", "column 2" + + "row 1", "m00", "m10" + "row 2", "m01", "m11" + "row 3", "m02", "m12" + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_mat2x3_copy(mat2x3 mat, mat2x3 dest) + + copy mat2x3 to another one (dest). + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat2x3_zero(mat2x3 mat) + + make given matrix zero + + Parameters: + | *[in,out]* **mat** matrix + +.. c:function:: void glm_mat2x3_make(const float * __restrict src, mat2x3 dest) + + Create mat2x3 matrix from pointer + + .. note:: **@src** must contain at least 6 elements. + + Parameters: + | *[in]* **src** pointer to an array of floats + | *[out]* **dest** destination matrix2x3 + +.. c:function:: void glm_mat2x3_mul(mat2x3 m1, mat3x2 m2, mat3 dest) + + multiply m1 and m2 to dest + + .. code-block:: c + + glm_mat2x3_mul(mat2x3, mat3x2, mat3); + + Parameters: + | *[in]* **m1** left matrix (mat2x3) + | *[in]* **m2** right matrix (mat3x2) + | *[out]* **dest** destination matrix (mat3) + + .. csv-table:: mat2x3 + :header: "", "column 1", "column 2" + + "row 1", "a00", "a10" + "row 2", "a01", "a11" + "row 3", "a02", "a12" + + .. csv-table:: mat3x2 + :header: "", "column 1", "column 2", "column 3" + + "row 1", "b00", "b10", "b20" + "row 2", "b01", "b11", "b21" + + .. csv-table:: mat3x3 + :header: "", "column 1", "column 2", "column 3" + + "row 1", "a00 * b00 + a10 * b01", "a00 * b10 + a10 * b11", "a00 * b20 + a10 * b21" + "row 2", "a01 * b00 + a11 * b01", "a01 * b10 + a11 * b11", "a01 * b20 + a11 * b21" + "row 3", "a02 * b00 + a12 * b01", "a02 * b10 + a12 * b11", "a02 * b20 + a12 * b21" + +.. c:function:: void glm_mat2x3_mulv(mat2x3 m, vec2 v, vec3 dest) + + multiply mat2x3 with vec2 (column vector) and store in dest column vector + + Parameters: + | *[in]* **m** mat2x3 (left) + | *[in]* **v** vec3 (right, column vector) + | *[out]* **dest** destination (result, column vector) + + .. csv-table:: mat2x3 + :header: "", "column 1", "column 2" + + "row 1", "m00", "m10" + "row 2", "m01", "m11" + "row 3", "m02", "m12" + + .. csv-table:: column vec2 (1x2) + :header: "", "column 1" + + "row 1", "v0" + "row 2", "v1" + + .. csv-table:: column vec3 (1x3) + :header: "", "column 1" + + "row 1", "m00 * v0 + m10 * v1" + "row 2", "m01 * v0 + m11 * v1" + "row 3", "m02 * v0 + m12 * v1" + +.. c:function:: void glm_mat2x3_transpose(mat2x3 m, mat3x2 dest) + + transpose matrix and store in dest + + Parameters: + | *[in]* **m** matrix + | *[out]* **dest** destination + +.. c:function:: void glm_mat2x3_scale(mat2x3 m, float s) + + multiply matrix with scalar + + Parameters: + | *[in, out]* **m** matrix + | *[in]* **s** scalar diff --git a/cglm/docs/source/mat2x4.rst b/cglm/docs/source/mat2x4.rst new file mode 100644 index 0000000..0acc643 --- /dev/null +++ b/cglm/docs/source/mat2x4.rst @@ -0,0 +1,145 @@ +.. default-domain:: C + +mat2x4 +====== + +Header: cglm/mat2x4.h + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. GLM_MAT2X4_ZERO_INIT +#. GLM_MAT2X4_ZERO + +Functions: + +1. :c:func:`glm_mat2x4_copy` +#. :c:func:`glm_mat2x4_zero` +#. :c:func:`glm_mat2x4_make` +#. :c:func:`glm_mat2x4_mul` +#. :c:func:`glm_mat2x4_mulv` +#. :c:func:`glm_mat2x4_transpose` +#. :c:func:`glm_mat2x4_scale` + +Represented +~~~~~~~~~~~ + +.. csv-table:: mat2x4 + :header: "", "column 1", "column 2" + + "row 1", "m00", "m10" + "row 2", "m01", "m11" + "row 3", "m02", "m12" + "row 4", "m03", "m13" + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_mat2x4_copy(mat2x4 mat, mat2x4 dest) + + copy mat2x4 to another one (dest). + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat2x4_zero(mat2x4 mat) + + make given matrix zero + + Parameters: + | *[in,out]* **mat** matrix + +.. c:function:: void glm_mat2x4_make(const float * __restrict src, mat2x4 dest) + + Create mat2x4 matrix from pointer + + .. note:: **@src** must contain at least 8 elements. + + Parameters: + | *[in]* **src** pointer to an array of floats + | *[out]* **dest** destination matrix2x4 + +.. c:function:: void glm_mat2x4_mul(mat2x4 m1, mat4x2 m2, mat4 dest) + + multiply m1 and m2 to dest + + .. code-block:: c + + glm_mat2x4_mul(mat2x4, mat4x2, mat4); + + Parameters: + | *[in]* **m1** left matrix (mat2x4) + | *[in]* **m2** right matrix (mat4x2) + | *[out]* **dest** destination matrix (mat4) + + .. csv-table:: mat2x4 + :header: "", "column 1", "column 2" + + "row 1", "a00", "a10" + "row 2", "a01", "a11" + "row 3", "a02", "a12" + "row 4", "a03", "a13" + + .. csv-table:: mat4x2 + :header: "", "column 1", "column 2", "column 3", "column 4" + + "row 1", "b00", "b10", "b20", "b30" + "row 2", "b01", "b11", "b21", "b31" + + .. csv-table:: mat4x4 + :header: "", "column 1", "column 2", "column 3", "column 4" + + "row 1", "a00 * b00 + a10 * b01", "a00 * b10 + a10 * b11", "a00 * b20 + a10 * b21", "a00 * b30 + a10 * b31" + "row 2", "a01 * b00 + a11 * b01", "a01 * b10 + a11 * b11", "a01 * b20 + a11 * b21", "a01 * b30 + a11 * b31" + "row 3", "a02 * b00 + a12 * b01", "a02 * b10 + a12 * b11", "a02 * b20 + a12 * b21", "a02 * b30 + a12 * b31" + "row 4", "a03 * b00 + a13 * b01", "a03 * b10 + a13 * b11", "a03 * b20 + a13 * b21", "a03 * b30 + a13 * b31" + +.. c:function:: void glm_mat2x4_mulv(mat2x4 m, vec2 v, vec4 dest) + + multiply mat2x4 with vec2 (column vector) and store in dest column vector + + Parameters: + | *[in]* **m** mat2x4 (left) + | *[in]* **v** vec2 (right, column vector) + | *[out]* **dest** destination (result, column vector) + + .. csv-table:: mat2x4 + :header: "", "column 1", "column 2" + + "row 1", "m00", "m10" + "row 2", "m01", "m11" + "row 3", "m02", "m12" + "row 4", "m03", "m13" + + .. csv-table:: column vec2 (1x2) + :header: "", "column 1" + + "row 1", "v0" + "row 2", "v1" + + .. csv-table:: column vec4 (1x4) + :header: "", "column 1" + + "row 1", "m00 * v0 + m10 * v1" + "row 2", "m01 * v0 + m11 * v1" + "row 3", "m02 * v0 + m12 * v1" + "row 4", "m03 * v0 + m13 * v1" + +.. c:function:: void glm_mat2x4_transpose(mat2x4 m, mat4x2 dest) + + transpose matrix and store in dest + + Parameters: + | *[in]* **m** matrix + | *[out]* **dest** destination + +.. c:function:: void glm_mat2x4_scale(mat2x4 m, float s) + + multiply matrix with scalar + + Parameters: + | *[in, out]* **m** matrix + | *[in]* **s** scalar diff --git a/cglm/docs/source/mat3.rst b/cglm/docs/source/mat3.rst new file mode 100644 index 0000000..34d4ab7 --- /dev/null +++ b/cglm/docs/source/mat3.rst @@ -0,0 +1,201 @@ +.. default-domain:: C + +mat3 +==== + +Header: cglm/mat3.h + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. GLM_MAT3_IDENTITY_INIT +#. GLM_MAT3_ZERO_INIT +#. GLM_MAT3_IDENTITY +#. GLM_MAT3_ZERO +#. glm_mat3_dup(mat, dest) + +Functions: + +1. :c:func:`glm_mat3_copy` +#. :c:func:`glm_mat3_identity` +#. :c:func:`glm_mat3_identity_array` +#. :c:func:`glm_mat3_zero` +#. :c:func:`glm_mat3_mul` +#. :c:func:`glm_mat3_transpose_to` +#. :c:func:`glm_mat3_transpose` +#. :c:func:`glm_mat3_mulv` +#. :c:func:`glm_mat3_quat` +#. :c:func:`glm_mat3_scale` +#. :c:func:`glm_mat3_det` +#. :c:func:`glm_mat3_inv` +#. :c:func:`glm_mat3_trace` +#. :c:func:`glm_mat3_swap_col` +#. :c:func:`glm_mat3_swap_row` +#. :c:func:`glm_mat3_rmc` +#. :c:func:`glm_mat3_make` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_mat3_copy(mat3 mat, mat3 dest) + + copy mat3 to another one (dest). + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat3_identity(mat3 mat) + + copy identity mat3 to mat, or makes mat to identity + + Parameters: + | *[out]* **mat** matrix + +.. c:function:: void glm_mat3_identity_array(mat3 * __restrict mat, size_t count) + + make given matrix array's each element identity matrix + + Parameters: + | *[in,out]* **mat** matrix array (must be aligned (16/32) if alignment is not disabled) + | *[in]* **count** count of matrices + +.. c:function:: void glm_mat3_zero(mat3 mat) + + make given matrix zero + + Parameters: + | *[in,out]* **mat** matrix to + +.. c:function:: void glm_mat3_mul(mat3 m1, mat3 m2, mat3 dest) + + multiply m1 and m2 to dest + + m1, m2 and dest matrices can be same matrix, it is possible to write this: + + .. code-block:: c + + mat3 m = GLM_MAT3_IDENTITY_INIT; + glm_mat3_mul(m, m, m); + + Parameters: + | *[in]* **m1** left matrix + | *[in]* **m2** right matrix + | *[out]* **dest** destination matrix + +.. c:function:: void glm_mat3_transpose_to(mat3 m, mat3 dest) + + transpose mat4 and store in dest + source matrix will not be transposed unless dest is m + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat3_transpose(mat3 m) + + transpose mat3 and store result in same matrix + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat3_mulv(mat3 m, vec3 v, vec3 dest) + + multiply mat3 with vec3 (column vector) and store in dest vector + + Parameters: + | *[in]* **m** mat3 (left) + | *[in]* **v** vec3 (right, column vector) + | *[out]* **dest** destination (result, column vector) + +.. c:function:: void glm_mat3_quat(mat3 m, versor dest) + + convert mat3 to quaternion + + Parameters: + | *[in]* **m** rotation matrix + | *[out]* **dest** destination quaternion + +.. c:function:: void glm_mat3_scale(mat3 m, float s) + + multiply matrix with scalar + + Parameters: + | *[in, out]* **m** matrix + | *[in]* **s** scalar + +.. c:function:: float glm_mat3_det(mat3 mat) + + returns mat3 determinant + + Parameters: + | *[in]* **mat** matrix + + Returns: + mat3 determinant + +.. c:function:: void glm_mat3_inv(mat3 mat, mat3 dest) + + inverse mat3 and store in dest + + Parameters: + | *[in]* **mat** matrix + | *[out]* **dest** destination (inverse matrix) + +.. c:function:: void glm_mat3_trace(mat3 m) + + | sum of the elements on the main diagonal from upper left to the lower right + + Parameters: + | *[in]* **m** matrix + + Returns: + trace of matrix + +.. c:function:: void glm_mat3_swap_col(mat3 mat, int col1, int col2) + + swap two matrix columns + + Parameters: + | *[in, out]* **mat** matrix + | *[in]* **col1** col1 + | *[in]* **col2** col2 + +.. c:function:: void glm_mat3_swap_row(mat3 mat, int row1, int row2) + + swap two matrix rows + + Parameters: + | *[in, out]* **mat** matrix + | *[in]* **row1** row1 + | *[in]* **row2** row2 + +.. c:function:: float glm_mat3_rmc(vec3 r, mat3 m, vec3 c) + + | **rmc** stands for **Row** * **Matrix** * **Column** + + | helper for R (row vector) * M (matrix) * C (column vector) + + | the result is scalar because R * M = Matrix1x3 (row vector), + | then Matrix1x3 * Vec3 (column vector) = Matrix1x1 (Scalar) + + Parameters: + | *[in]* **r** row vector or matrix1x3 + | *[in]* **m** matrix3x3 + | *[in]* **c** column vector or matrix3x1 + + Returns: + scalar value e.g. Matrix1x1 + +.. c:function:: void glm_mat3_make(const float * __restrict src, mat3 dest) + + Create mat3 matrix from pointer + + .. note:: **@src** must contain at least 9 elements. + + Parameters: + | *[in]* **src** pointer to an array of floats + | *[out]* **dest** destination matrix3x3 diff --git a/cglm/docs/source/mat3x2.rst b/cglm/docs/source/mat3x2.rst new file mode 100644 index 0000000..7604728 --- /dev/null +++ b/cglm/docs/source/mat3x2.rst @@ -0,0 +1,137 @@ +.. default-domain:: C + +mat3x2 +====== + +Header: cglm/mat3x2.h + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. GLM_MAT3X2_ZERO_INIT +#. GLM_MAT3X2_ZERO + +Functions: + +1. :c:func:`glm_mat3x2_copy` +#. :c:func:`glm_mat3x2_zero` +#. :c:func:`glm_mat3x2_make` +#. :c:func:`glm_mat3x2_mul` +#. :c:func:`glm_mat3x2_mulv` +#. :c:func:`glm_mat3x2_transpose` +#. :c:func:`glm_mat3x2_scale` + +Represented +~~~~~~~~~~~ + +.. csv-table:: mat3x2 + :header: "", "column 1", "column 2", "column 3" + + "row 1", "m00", "m10", "m20" + "row 2", "m01", "m11", "m21" + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_mat3x2_copy(mat3x2 mat, mat3x2 dest) + + copy mat3x2 to another one (dest). + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat3x2_zero(mat3x2 mat) + + make given matrix zero + + Parameters: + | *[in,out]* **mat** matrix + +.. c:function:: void glm_mat3x2_make(const float * __restrict src, mat3x2 dest) + + Create mat3x2 matrix from pointer + + .. note:: **@src** must contain at least 6 elements. + + Parameters: + | *[in]* **src** pointer to an array of floats + | *[out]* **dest** destination matrix3x2 + +.. c:function:: void glm_mat3x2_mul(mat3x2 m1, mat2x3 m2, mat2 dest) + + multiply m1 and m2 to dest + + .. code-block:: c + + glm_mat3x2_mul(mat3x2, mat2x3, mat2); + + Parameters: + | *[in]* **m1** left matrix (mat3x2) + | *[in]* **m2** right matrix (mat2x3) + | *[out]* **dest** destination matrix (mat2) + + .. csv-table:: mat3x2 + :header: "", "column 1", "column 2", "column 3" + + "row 1", "a00", "a10", "a20" + "row 2", "a01", "a11", "a21" + + .. csv-table:: mat2x3 + :header: "", "column 1", "column 2" + + "row 1", "b00", "b10" + "row 2", "b01", "b11" + "row 3", "b02", "b12" + + .. csv-table:: mat2x2 + :header: "", "column 1", "column 2" + + "row 1", "a00 * b00 + a10 * b01 + a20 * b02", "a00 * b10 + a10 * b11 + a20 * b12" + "row 2", "a01 * b00 + a11 * b01 + a21 * b02", "a01 * b10 + a11 * b11 + a21 * b12" + +.. c:function:: void glm_mat3x2_mulv(mat3x2 m, vec3 v, vec2 dest) + + multiply mat3x2 with vec3 (column vector) and store in dest vector + + Parameters: + | *[in]* **m** mat3x2 (left) + | *[in]* **v** vec3 (right, column vector) + | *[out]* **dest** destination (result, column vector) + + .. csv-table:: mat3x2 + :header: "", "column 1", "column 2", "column 3" + + "row 1", "m00", "m10", "m20" + "row 2", "m01", "m11", "m21" + + .. csv-table:: column vec3 (1x3) + :header: "", "column 1" + + "row 1", "v0" + "row 2", "v1" + "row 3", "v2" + + .. csv-table:: column vec2 (1x2) + :header: "", "column 1" + + "row 1", "m00 * v0 + m10 * v1 + m20 * v2" + "row 2", "m01 * v0 + m11 * v1 + m21 * v2" + +.. c:function:: void glm_mat3x2_transpose(mat3x2 m, mat2x3 dest) + + transpose matrix and store in dest + + Parameters: + | *[in]* **m** matrix + | *[out]* **dest** destination + +.. c:function:: void glm_mat3x2_scale(mat3x2 m, float s) + + multiply matrix with scalar + + Parameters: + | *[in, out]* **m** matrix + | *[in]* **s** scalar diff --git a/cglm/docs/source/mat3x4.rst b/cglm/docs/source/mat3x4.rst new file mode 100644 index 0000000..cddce47 --- /dev/null +++ b/cglm/docs/source/mat3x4.rst @@ -0,0 +1,147 @@ +.. default-domain:: C + +mat3x4 +====== + +Header: cglm/mat3x4.h + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. GLM_MAT3X4_ZERO_INIT +#. GLM_MAT3X4_ZERO + +Functions: + +1. :c:func:`glm_mat3x4_copy` +#. :c:func:`glm_mat3x4_zero` +#. :c:func:`glm_mat3x4_make` +#. :c:func:`glm_mat3x4_mul` +#. :c:func:`glm_mat3x4_mulv` +#. :c:func:`glm_mat3x4_transpose` +#. :c:func:`glm_mat3x4_scale` + +Represented +~~~~~~~~~~~ + +.. csv-table:: mat3x4 + :header: "", "column 1", "column 2", "column 3" + + "row 1", "m00", "m10", "m20" + "row 2", "m01", "m11", "m21" + "row 3", "m02", "m12", "m22" + "row 4", "m03", "m13", "m23" + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_mat3x4_copy(mat3x4 mat, mat3x4 dest) + + copy mat3x4 to another one (dest). + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat3x4_zero(mat3x4 mat) + + make given matrix zero + + Parameters: + | *[in,out]* **mat** matrix + +.. c:function:: void glm_mat3x4_make(const float * __restrict src, mat3x4 dest) + + Create mat3x4 matrix from pointer + + .. note::: **@src** must contain at least 12 elements. + + Parameters: + | *[in]* **src** pointer to an array of floats + | *[out]* **dest** destination matrix3x4 + +.. c:function:: void glm_mat3x4_mul(mat3x4 m1, mat4x3 m2, mat4 dest) + + multiply m1 and m2 to dest + + .. code-block:: c + + glm_mat3x4_mul(mat3x4, mat4x3, mat4); + + Parameters: + | *[in]* **m1** left matrix (mat3x4) + | *[in]* **m2** right matrix (mat4x3) + | *[out]* **dest** destination matrix (mat4) + + .. csv-table:: mat3x4 + :header: "", "column 1", "column 2", "column 3" + + "row 1", "a00", "a10", "a20" + "row 2", "a01", "a11", "a21" + "row 3", "a02", "a12", "a22" + "row 4", "a03", "a13", "a23" + + .. csv-table:: mat4x3 + :header: "", "column 1", "column 2", "column 3", "column 4" + + "row 1", "b00", "b10", "b20", "b30" + "row 2", "b01", "b11", "b21", "b31" + "row 3", "b02", "b12", "b22", "b32" + + .. csv-table:: mat4x4 + :header: "", "column 1", "column 2", "column 3", "column 4" + + "row 1", "a00 * b00 + a10 * b01 + a20 * b02", "a00 * b10 + a10 * b11 + a20 * b12", "a00 * b20 + a10 * b21 + a20 * b22", "a00 * b30 + a10 * b31 + a20 * b32" + "row 2", "a01 * b00 + a11 * b01 + a21 * b02", "a01 * b10 + a11 * b11 + a21 * b12", "a01 * b20 + a11 * b21 + a21 * b22", "a01 * b30 + a11 * b31 + a21 * b32" + "row 3", "a02 * b00 + a12 * b01 + a22 * b02", "a02 * b10 + a12 * b11 + a22 * b12", "a02 * b20 + a12 * b21 + a22 * b22", "a02 * b30 + a12 * b31 + a22 * b32" + "row 4", "a03 * b00 + a13 * b01 + a23 * b02", "a03 * b10 + a13 * b11 + a23 * b12", "a03 * b20 + a13 * b21 + a23 * b22", "a03 * b30 + a13 * b31 + a23 * b32" + +.. c:function:: void glm_mat3x4_mulv(mat3x4 m, vec3 v, vec4 dest) + + multiply mat3x4 with vec3 (column vector) and store in dest vector + + Parameters: + | *[in]* **m** mat3x4 (left) + | *[in]* **v** vec3 (right, column vector) + | *[out]* **dest** destination (result, column vector) + + .. csv-table:: mat3x4 + :header: "", "column 1", "column 2", "column 3" + + "row 1", "m00", "m10", "m20" + "row 2", "m01", "m11", "m21" + "row 3", "m02", "m12", "m22" + "row 4", "m03", "m13", "m23" + + .. csv-table:: column vec3 (1x3) + :header: "", "column 1" + + "row 1", "v0" + "row 2", "v1" + "row 3", "v2" + + .. csv-table:: column vec4 (1x4) + :header: "", "column 1" + + "row 1", "m00 * v0 + m10 * v1 + m20 * v2" + "row 2", "m01 * v0 + m11 * v1 + m21 * v2" + "row 3", "m02 * v0 + m12 * v1 + m22 * v2" + "row 4", "m03 * v0 + m13 * v1 + m23 * v2" + +.. c:function:: void glm_mat3x4_transpose(mat3x4 m, mat4x3 dest) + + transpose matrix and store in dest + + Parameters: + | *[in]* **m** matrix + | *[out]* **dest** destination + +.. c:function:: void glm_mat3x4_scale(mat3x4 m, float s) + + multiply matrix with scalar + + Parameters: + | *[in, out]* **m** matrix + | *[in]* **s** scalar diff --git a/cglm/docs/source/mat4.rst b/cglm/docs/source/mat4.rst new file mode 100644 index 0000000..ed85d46 --- /dev/null +++ b/cglm/docs/source/mat4.rst @@ -0,0 +1,315 @@ +.. default-domain:: C + +mat4 +==== + +Header: cglm/mat4.h + +Important: :c:func:`glm_mat4_scale` multiplies mat4 with scalar, if you need to +apply scale transform use :c:func:`glm_scale` functions. + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. GLM_MAT4_IDENTITY_INIT +#. GLM_MAT4_ZERO_INIT +#. GLM_MAT4_IDENTITY +#. GLM_MAT4_ZERO +#. glm_mat4_udup(mat, dest) +#. glm_mat4_dup(mat, dest) + +Functions: + +1. :c:func:`glm_mat4_ucopy` +#. :c:func:`glm_mat4_copy` +#. :c:func:`glm_mat4_identity` +#. :c:func:`glm_mat4_identity_array` +#. :c:func:`glm_mat4_zero` +#. :c:func:`glm_mat4_pick3` +#. :c:func:`glm_mat4_pick3t` +#. :c:func:`glm_mat4_ins3` +#. :c:func:`glm_mat4_mul` +#. :c:func:`glm_mat4_mulN` +#. :c:func:`glm_mat4_mulv` +#. :c:func:`glm_mat4_mulv3` +#. :c:func:`glm_mat4_trace` +#. :c:func:`glm_mat4_trace3` +#. :c:func:`glm_mat4_quat` +#. :c:func:`glm_mat4_transpose_to` +#. :c:func:`glm_mat4_transpose` +#. :c:func:`glm_mat4_scale_p` +#. :c:func:`glm_mat4_scale` +#. :c:func:`glm_mat4_det` +#. :c:func:`glm_mat4_inv` +#. :c:func:`glm_mat4_inv_fast` +#. :c:func:`glm_mat4_swap_col` +#. :c:func:`glm_mat4_swap_row` +#. :c:func:`glm_mat4_rmc` +#. :c:func:`glm_mat4_make` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_mat4_ucopy(mat4 mat, mat4 dest) + + copy mat4 to another one (dest). u means align is not required for dest + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat4_copy(mat4 mat, mat4 dest) + + copy mat4 to another one (dest). + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat4_identity(mat4 mat) + + copy identity mat4 to mat, or makes mat to identity + + Parameters: + | *[out]* **mat** matrix + +.. c:function:: void glm_mat4_identity_array(mat4 * __restrict mat, size_t count) + + make given matrix array's each element identity matrix + + Parameters: + | *[in,out]* **mat** matrix array (must be aligned (16/32) if alignment is not disabled) + | *[in]* **count** count of matrices + +.. c:function:: void glm_mat4_zero(mat4 mat) + + make given matrix zero + + Parameters: + | *[in,out]* **mat** matrix to + +.. c:function:: void glm_mat4_pick3(mat4 mat, mat3 dest) + + copy upper-left of mat4 to mat3 + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat4_pick3t(mat4 mat, mat4 dest) + + copy upper-left of mat4 to mat3 (transposed) + the postfix t stands for transpose + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat4_ins3(mat3 mat, mat4 dest) + + copy mat3 to mat4's upper-left. this function does not fill mat4's other + elements. To do that use glm_mat4. + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat4_mul(mat4 m1, mat4 m2, mat4 dest) + + multiply m1 and m2 to dest + + m1, m2 and dest matrices can be same matrix, it is possible to write this: + + .. code-block:: c + + mat4 m = GLM_MAT4_IDENTITY_INIT; + glm_mat4_mul(m, m, m); + + Parameters: + | *[in]* **m1** left matrix + | *[in]* **m2** right matrix + | *[out]* **dest** destination matrix + +.. c:function:: void glm_mat4_mulN(mat4 * __restrict matrices[], int len, mat4 dest) + + mupliply N mat4 matrices and store result in dest + | this function lets you multiply multiple (more than two or more...) + | matrices + + | multiplication will be done in loop, this may reduce instructions + | size but if **len** is too small then compiler may unroll whole loop + + .. code-block:: c + + mat m1, m2, m3, m4, res; + glm_mat4_mulN((mat4 *[]){&m1, &m2, &m3, &m4}, 4, res); + + Parameters: + | *[in]* **matrices** array of mat4 + | *[in]* **len** matrices count + | *[out]* **dest** destination matrix + +.. c:function:: void glm_mat4_mulv(mat4 m, vec4 v, vec4 dest) + + multiply mat4 with vec4 (column vector) and store in dest vector + + Parameters: + | *[in]* **m** mat4 (left) + | *[in]* **v** vec4 (right, column vector) + | *[out]* **dest** vec4 (result, column vector) + +.. c:function:: void glm_mat4_mulv3(mat4 m, vec3 v, float last, vec3 dest) + + | multiply **vec3** with **mat4** and get **vec3** as result + | + | actually the result is **vec4**, after multiplication, + the last component is trimmed, if you need the result's last component + then don't use this function and consider to use **glm_mat4_mulv()** + + Parameters: + | *[in]* **m** mat4(affine transform) + | *[in]* **v** vec3 + | *[in]* **last** 4th item to make it vec4 + | *[out]* **dest** result vector (vec3) + +.. c:function:: void glm_mat4_trace(mat4 m) + + | sum of the elements on the main diagonal from upper left to the lower right + + Parameters: + | *[in]* **m** matrix + + Returns: + trace of matrix + +.. c:function:: void glm_mat4_trace3(mat4 m) + + | trace of matrix (rotation part) + | sum of the elements on the main diagonal from upper left to the lower right + + Parameters: + | *[in]* **m** matrix + + Returns: + trace of matrix + +.. c:function:: void glm_mat4_quat(mat4 m, versor dest) + + convert mat4's rotation part to quaternion + + Parameters: + | *[in]* **m** affine matrix + | *[out]* **dest** destination quaternion + +.. c:function:: void glm_mat4_transpose_to(mat4 m, mat4 dest) + + transpose mat4 and store in dest + source matrix will not be transposed unless dest is m + + Parameters: + | *[in]* **m** matrix + | *[out]* **dest** destination matrix + +.. c:function:: void glm_mat4_transpose(mat4 m) + + transpose mat4 and store result in same matrix + + Parameters: + | *[in]* **m** source + | *[out]* **dest** destination matrix + +.. c:function:: void glm_mat4_scale_p(mat4 m, float s) + + scale (multiply with scalar) matrix without simd optimization + + Parameters: + | *[in, out]* **m** matrix + | *[in]* **s** scalar + +.. c:function:: void glm_mat4_scale(mat4 m, float s) + + scale (multiply with scalar) matrix + THIS IS NOT SCALE TRANSFORM, use glm_scale for that. + + Parameters: + | *[in, out]* **m** matrix + | *[in]* **s** scalar + +.. c:function:: float glm_mat4_det(mat4 mat) + + mat4 determinant + + Parameters: + | *[in]* **mat** matrix + + Return: + | determinant + +.. c:function:: void glm_mat4_inv(mat4 mat, mat4 dest) + + inverse mat4 and store in dest + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination matrix (inverse matrix) + +.. c:function:: void glm_mat4_inv_fast(mat4 mat, mat4 dest) + + inverse mat4 and store in dest + + | this func uses reciprocal approximation without extra corrections + | e.g Newton-Raphson. this should work faster than normal, + | to get more precise use glm_mat4_inv version. + + .. note:: You will lose precision, glm_mat4_inv is more accurate + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat4_swap_col(mat4 mat, int col1, int col2) + + swap two matrix columns + + Parameters: + | *[in, out]* **mat** matrix + | *[in]* **col1** col1 + | *[in]* **col2** col2 + +.. c:function:: void glm_mat4_swap_row(mat4 mat, int row1, int row2) + + swap two matrix rows + + Parameters: + | *[in, out]* **mat** matrix + | *[in]* **row1** row1 + | *[in]* **row2** row2 + +.. c:function:: float glm_mat4_rmc(vec4 r, mat4 m, vec4 c) + + | **rmc** stands for **Row** * **Matrix** * **Column** + + | helper for R (row vector) * M (matrix) * C (column vector) + + | the result is scalar because R * M = Matrix1x4 (row vector), + | then Matrix1x4 * Vec4 (column vector) = Matrix1x1 (Scalar) + + Parameters: + | *[in]* **r** row vector or matrix1x4 + | *[in]* **m** matrix4x4 + | *[in]* **c** column vector or matrix4x1 + + Returns: + scalar value e.g. Matrix1x1 + +.. c:function:: void glm_mat4_make(const float * __restrict src, mat4 dest) + + Create mat4 matrix from pointer + + .. note:: **@src** must contain at least 16 elements. + + Parameters: + | *[in]* **src** pointer to an array of floats + | *[out]* **dest** destination matrix4x4 diff --git a/cglm/docs/source/mat4x2.rst b/cglm/docs/source/mat4x2.rst new file mode 100644 index 0000000..35ab2b8 --- /dev/null +++ b/cglm/docs/source/mat4x2.rst @@ -0,0 +1,141 @@ +.. default-domain:: C + +mat4x2 +====== + +Header: cglm/mat4x2.h + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. GLM_MAT4X2_ZERO_INIT +#. GLM_MAT4X2_ZERO + +Functions: + +1. :c:func:`glm_mat4x2_copy` +#. :c:func:`glm_mat4x2_zero` +#. :c:func:`glm_mat4x2_make` +#. :c:func:`glm_mat4x2_mul` +#. :c:func:`glm_mat4x2_mulv` +#. :c:func:`glm_mat4x2_transpose` +#. :c:func:`glm_mat4x2_scale` + +Represented +~~~~~~~~~~~ + +.. csv-table:: mat4x2 + :header: "", "column 1", "column 2", "column 3", "column4" + + "row 1", "m00", "m10", "m20", "m30" + "row 2", "m01", "m11", "m21", "m31" + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_mat4x2_copy(mat4x2 mat, mat4x2 dest) + + copy mat4x2 to another one (dest). + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat4x2_zero(mat4x2 mat) + + make given matrix zero + + Parameters: + | *[in,out]* **mat** matrix + +.. c:function:: void glm_mat4x2_make(const float * __restrict src, mat4x2 dest) + + Create mat4x2 matrix from pointer + + .. note:: **@src** must contain at least 8 elements. + + Parameters: + | *[in]* **src** pointer to an array of floats + | *[out]* **dest** destination matrix4x2 + +.. c:function:: void glm_mat4x2_mul(mat4x2 m1, mat2x4 m2, mat2 dest) + + multiply m1 and m2 to dest + + .. code-block:: c + + glm_mat4x2_mul(mat4x2, mat2x4, mat2); + + Parameters: + | *[in]* **m1** left matrix (mat4x2) + | *[in]* **m2** right matrix (mat2x4) + | *[out]* **dest** destination matrix (mat2) + + .. csv-table:: mat4x2 + :header: "", "column 1", "column 2", "column 3", "column 4" + + "row 1", "a00", "a10", "a20", "a30" + "row 2", "a01", "a11", "a21", "a31" + + .. csv-table:: mat2x4 + :header: "", "column 1", "column 2" + + "row 1", "b00", "b10" + "row 2", "b01", "b11" + "row 3", "b02", "b12" + "row 4", "b03", "b13" + + .. csv-table:: mat2x2 + :header: "", "column 1", "column 2" + + "row 1", "a00 * b00 + a10 * b01 + a20 * b02 + a30 * b03", "a00 * b10 + a10 * b11 + a20 * b12 + a30 * b13" + "row 2", "a01 * b00 + a11 * b01 + a21 * b02 + a31 * b03", "a01 * b10 + a11 * b11 + a21 * b12 + a31 * b13" + +.. c:function:: void glm_mat4x2_mulv(mat4x2 m, vec4 v, vec2 dest) + + multiply mat4x2 with vec4 (column vector) and store in dest vector + + Parameters: + | *[in]* **m** mat4x2 (left) + | *[in]* **v** vec4 (right, column vector) + | *[out]* **dest** destination (result, column vector) + + .. csv-table:: mat4x2 + :header: "", "column 1", "column 2", "column 3" + + "row 1", "m00", "m10", "m20" + "row 2", "m01", "m11", "m21" + "row 3", "m02", "m12", "m22" + "row 4", "m03", "m13", "m23" + + .. csv-table:: column vec4 (1x4) + :header: "", "column 1" + + "row 1", "v0" + "row 2", "v1" + "row 3", "v2" + "row 4", "v3" + + .. csv-table:: column vec2 (1x2) + :header: "", "column 1" + + "row 1", "m00 * v0 + m10 * v1 + m20 * v2 + m30 * v3" + "row 2", "m01 * v0 + m11 * v1 + m21 * v2 + m31 * v3" + +.. c:function:: void glm_mat4x2_transpose(mat4x2 m, mat2x4 dest) + + transpose matrix and store in dest + + Parameters: + | *[in]* **m** matrix + | *[out]* **dest** destination + +.. c:function:: void glm_mat4x2_scale(mat4x2 m, float s) + + multiply matrix with scalar + + Parameters: + | *[in, out]* **m** matrix + | *[in]* **s** scalar diff --git a/cglm/docs/source/mat4x3.rst b/cglm/docs/source/mat4x3.rst new file mode 100644 index 0000000..5136e88 --- /dev/null +++ b/cglm/docs/source/mat4x3.rst @@ -0,0 +1,144 @@ +.. default-domain:: C + +mat4x3 +====== + +Header: cglm/mat4x3.h + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. GLM_MAT4X3_ZERO_INIT +#. GLM_MAT4X3_ZERO + +Functions: + +1. :c:func:`glm_mat4x3_copy` +#. :c:func:`glm_mat4x3_zero` +#. :c:func:`glm_mat4x3_make` +#. :c:func:`glm_mat4x3_mul` +#. :c:func:`glm_mat4x3_mulv` +#. :c:func:`glm_mat4x3_transpose` +#. :c:func:`glm_mat4x3_scale` + +Represented +~~~~~~~~~~~ + +.. csv-table:: mat4x3 + :header: "", "column 1", "column 2", "column 3", "column4" + + "row 1", "m00", "m10", "m20", "m30" + "row 2", "m01", "m11", "m21", "m31" + "row 3", "m02", "m12", "m22", "m32" + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_mat4x3_copy(mat4x3 mat, mat4x3 dest) + + copy mat4x3 to another one (dest). + + Parameters: + | *[in]* **mat** source + | *[out]* **dest** destination + +.. c:function:: void glm_mat4x3_zero(mat4x3 mat) + + make given matrix zero + + Parameters: + | *[in,out]* **mat** matrix + +.. c:function:: void glm_mat4x3_make(const float * __restrict src, mat4x3 dest) + + Create mat4x3 matrix from pointer + + .. note:: **@src** must contain at least 12 elements. + + Parameters: + | *[in]* **src** pointer to an array of floats + | *[out]* **dest** destination matrix4x3 + +.. c:function:: void glm_mat4x3_mul(mat4x3 m1, mat3x4 m2, mat3 dest) + + multiply m1 and m2 to dest + + .. code-block:: c + + glm_mat4x3_mul(mat4x3, mat3x4, mat3); + + Parameters: + | *[in]* **m1** left matrix (mat4x3) + | *[in]* **m2** right matrix (mat3x4) + | *[out]* **dest** destination matrix (mat3) + + .. csv-table:: mat4x3 + :header: "", "column 1", "column 2", "column 3", "column 4" + + "row 1", "a00", "a10", "a20", "a30" + "row 2", "a01", "a11", "a21", "a31" + "row 3", "a02", "a12", "a22", "a32" + + .. csv-table:: mat3x4 + :header: "", "column 1", "column 2", "column 3" + + "row 1", "b00", "b10", "b20" + "row 2", "b01", "b11", "b21" + "row 3", "b02", "b12", "b22" + "row 4", "b03", "b13", "b23" + + .. csv-table:: mat3x3 + :header: "", "column 1", "column 2", "column 3" + + "row 1", "a00 * b00 + a10 * b01 + a20 * b02 + a30 * b03", "a00 * b10 + a10 * b11 + a20 * b12 + a30 * b13", "a00 * b20 + a10 * b21 + a20 * b22 + a30 * b23" + "row 2", "a01 * b00 + a11 * b01 + a21 * b02 + a31 * b03", "a01 * b10 + a11 * b11 + a21 * b12 + a31 * b13", "a01 * b20 + a11 * b21 + a21 * b22 + a31 * b23" + "row 3", "a02 * b00 + a12 * b01 + a22 * b02 + a32 * b03", "a02 * b10 + a12 * b11 + a22 * b12 + a32 * b13", "a02 * b20 + a12 * b21 + a22 * b22 + a32 * b23" + +.. c:function:: void glm_mat4x3_mulv(mat4x3 m, vec4 v, vec3 dest) + + multiply mat4x3 with vec4 (column vector) and store in dest column vector + + Parameters: + | *[in]* **m** mat4x3 (left) + | *[in]* **v** vec4 (right, column vector) + | *[out]* **dest** destination (result, column vector) + + .. csv-table:: mat4x3 + :header: "", "column 1", "column 2", "column 3", "column 4" + + "row 1", "m00", "m10", "m20", "m30" + "row 2", "m01", "m11", "m21", "m31" + "row 3", "m02", "m12", "m22", "m32" + + .. csv-table:: column vec4 (1x4) + :header: "", "column 1" + + "row 1", "v0" + "row 2", "v1" + "row 3", "v2" + "row 4", "v3" + + .. csv-table:: column vec3 (1x3) + :header: "", "column 1" + + "row 1", "m00 * v0 + m10 * v1 + m20 * v2 + m30 * v3" + "row 2", "m01 * v0 + m11 * v1 + m21 * v2 + m31 * v3" + "row 3", "m02 * v0 + m12 * v1 + m22 * v2 + m32 * v3" + +.. c:function:: void glm_mat4x3_transpose(mat4x3 m, mat3x4 dest) + + transpose matrix and store in dest + + Parameters: + | *[in]* **m** matrix + | *[out]* **dest** destination + +.. c:function:: void glm_mat4x3_scale(mat4x3 m, float s) + + multiply matrix with scalar + + Parameters: + | *[in, out]* **m** matrix + | *[in]* **s** scalar diff --git a/cglm/docs/source/noise.rst b/cglm/docs/source/noise.rst new file mode 100644 index 0000000..0976eee --- /dev/null +++ b/cglm/docs/source/noise.rst @@ -0,0 +1,60 @@ +.. default-domain:: C + +perlin +================================================================================ + +Header: cglm/noise.h + +Classic Perlin noise implementation. + +Based on the work of Stefan Gustavson and Ashima Arts on "webgl-noise": +https://github.com/stegu/webgl-noise +Following Stefan Gustavson's paper "Simplex noise demystified": +http://www.itn.liu.se/~stegu/simplexnoise/simplexnoise.pdf + +Implementation based on glm::perlin function: +https://github.com/g-truc/glm/blob/master/glm/gtc/noise.inl + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_perlin_vec4` +#. :c:func:`glm_perlin_vec3` +#. :c:func:`glm_perlin_vec2` + + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: float glm_perlin_vec4(vec4 point) + + | Classic Perlin noise + + Parameters: + | *[in]* **point** 4D point + + Returns: + | noise value + + +.. c:function:: float glm_perlin_vec3(vec3 point) + + | Classic Perlin noise + + Parameters: + | *[in]* **point** 3D point + + Returns: + | noise value + +.. c:function:: float glm_perlin_vec2(vec2 point) + + | Classic Perlin noise + + Parameters: + | *[in]* **point** 2D point + + Returns: + | noise value diff --git a/cglm/docs/source/opengl.rst b/cglm/docs/source/opengl.rst new file mode 100644 index 0000000..7cc8559 --- /dev/null +++ b/cglm/docs/source/opengl.rst @@ -0,0 +1,61 @@ +How to send vector or matrix to OpenGL like API +================================================== + +*cglm*'s vector and matrix types are arrays. So you can send them directly to a +function which accepts pointer. But you may got warnings for matrix because it is +two dimensional array. + +Passing / Uniforming Matrix to OpenGL: +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ +**glUniformMatrix4fv** accepts float pointer, you can pass matrix to that parameter +and it should work but with warnings. "You can pass" doesn't mean that you must pass like that. + +**Correct options:** + +Correct doesn't mean correct way to use OpenGL it is just shows correct way to pass cglm type to it. + +1. Pass first column +--------------------- + +The goal is that pass address of matrix, first element of matrix is also address of matrix, +because it is array of vectors and vector is array of floats. + +.. code-block:: c + + mat4 matrix; + /* ... */ + glUniformMatrix4fv(location, 1, GL_FALSE, matrix[0]); + +array of matrices: + +.. code-block:: c + + mat4 matrix; + /* ... */ + glUniformMatrix4fv(location, count, GL_FALSE, matrix[0][0]); + +2. Cast matrix to pointer +-------------------------- + +.. code-block:: c + + mat4 matrix; + /* ... */ + glUniformMatrix4fv(location, count, GL_FALSE, (float *)matrix); + +in this way, passing array of matrices is same + +Passing / Uniforming Vectors to OpenGL: +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +You don't need to do extra thing when passing cglm vectors to OpengL or other APIs. +Because a function like **glUniform4fv** accepts vector as pointer. cglm's vectors +are array of floats. So you can pass it directly to those functions: + +.. code-block:: c + + vec4 vec; + /* ... */ + glUniform4fv(location, 1, vec); + +this show how to pass **vec4** others are same. diff --git a/cglm/docs/source/opt.rst b/cglm/docs/source/opt.rst new file mode 100644 index 0000000..d41549e --- /dev/null +++ b/cglm/docs/source/opt.rst @@ -0,0 +1,132 @@ +.. default-domain:: C + +🛠️ Options +=============================================================================== + +A few options are provided via macros. + +❗️ IMPORTANT ❗️ + +It's a good idea to set up your config macros in build settings like CMake, Xcode, or Visual Studio. This is especially important if you're using features like Modules in Xcode, where adding macros directly before the **cglm** headers might not work. + +Alignment Option +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +By default, **cglm** requires types to be aligned with specific byte requirements: + +- vec3: 8 bytes +- vec4: 16 bytes +- mat4: 16 bytes (32 on AVX) +- versor: 16 bytes + +Starting with **v0.4.5**, **cglm** offers an option to relax these alignment requirements. To use this option, define the **CGLM_ALL_UNALIGNED** macro before including any headers. This definition can be made within Xcode, Visual Studio, other IDEs, or directly in your build system. If using pre-compiled versions of **cglm**, you'll need to compile them with the **CGLM_ALL_UNALIGNED** macro. + +**❗️NOTE:❗️** If you're using **cglm** across multiple interdependent projects: + +- Always or never use the **CGLM_ALL_UNALIGNED** macro in all linked projects to avoid configuration conflicts. A **cglm** header from one project could require alignment, while a header from another might not, leading to **cglm** functions accessing invalid memory locations. + +- **Key Point:** Maintain the same **cglm** configuration across all your projects. For example, if you activate **CGLM_ALL_UNALIGNED** in one project, ensure it's set in the others too. + +**❗️NOTE:❗️** + +While **CGLM_ALL_UNALIGNED** allows for flexibility in alignment, it doesn't override C's fundamental alignment rules. For example, an array like *vec4* decays to a pointer (float*) in functions, which must adhere to the alignment requirements of a float pointer (4 bytes). This adherence is crucial because **cglm** directly dereferences these pointers instead of copying data, and failing to meet alignment requirements can lead to unpredictable errors, such as crashes. + +You can use `CGLM_ALIGN` and `CGLM_ALIGN_MAT` macros for aligning local variables or struct members. However, when dealing with dynamic memory allocation or custom memory locations, you'll need to ensure alignment requirements are met appropriately for those cases + +Clipspace Option[s] +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +By starting **v0.8.3** cglm provides options to switch between clipspace configurations. + +Clipspace related files are located at `include/cglm/[struct]/clipspace.h` but +these are included in related files like `cam.h`. If you don't want to change your existing +clipspace configuration and want to use different clipspace function like `glm_lookat_zo` or `glm_lookat_lh_zo`... +then you can include individual headers or just define `CGLM_CLIPSPACE_INCLUDE_ALL` which will include all headers for you. + +1. **CGLM_CLIPSPACE_INCLUDE_ALL** +2. **CGLM_FORCE_DEPTH_ZERO_TO_ONE** +3. **CGLM_FORCE_LEFT_HANDED** + + +1. **CGLM_CLIPSPACE_INCLUDE_ALL**: + +By defining this macro, **cglm** will include all clipspace functions for you by just using +`#include cglm/cglm.h` or `#include cglm/struct.h` or `#include cglm/call.h` + +Otherwise you need to include header you want manually e.g. `#include cglm/clipspace/view_rh_zo.h` + +2. **CGLM_FORCE_DEPTH_ZERO_TO_ONE** + +This is similar to **GLM**'s **GLM_FORCE_DEPTH_ZERO_TO_ONE** option. +This will set clip space between 0 to 1 which makes **cglm** Vulkan, Metal friendly. + +You can use functions like `glm_lookat_lh_zo()` individually. By setting **CGLM_FORCE_DEPTH_ZERO_TO_ONE** +functions in cam.h for instance will use `_zo` versions. + +3. **CGLM_FORCE_LEFT_HANDED** + +Force **cglm** to use the left handed coordinate system by default, currently **cglm** uses right handed coordinate system as default, +you can change this behavior with this option. + +**VERY VERY IMPORTANT:** + +Be careful if you include **cglm** in multiple projects. + +SSE and SSE2 Shuffle Option +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ +**_mm_shuffle_ps** generates **shufps** instruction even if registers are same. +You can force it to generate **pshufd** instruction by defining +**CGLM_NO_INT_DOMAIN** macro. As default it is not defined. + +SSE3 and SSE4 Dot Product Options +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +You have to extra options for dot product: **CGLM_SSE4_DOT** and **CGLM_SSE3_DOT**. + +- If **SSE4** is enabled then you can define **CGLM_SSE4_DOT** to force cglm to use **_mm_dp_ps** instruction. +- If **SSE3** is enabled then you can define **CGLM_SSE3_DOT** to force cglm to use **_mm_hadd_ps** instructions. + +otherwise cglm will use custom cglm's hadd functions which are optimized too. + +Struct API Options +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +To configure the Struct API namespace, you can define the following macros before including the cglm/struct.h header: + +- **CGLM_OMIT_NS_FROM_STRUCT_API**: omits CGLM_STRUCT_API_NS (`glms_`) namespace completely if there is sub namespace e.g `mat4_`, `vec4_` ... DEFAULT is not defined +- **CGLM_STRUCT_API_NS**: define name space for struct api, DEFAULT is **glms** +- **CGLM_STRUCT_API_NAME_SUFFIX**: define name suffix, DEFAULT is **empty** e.g defining it as #define CGLM_STRUCT_API_NAME_SUFFIX s will add s suffix to mat4_mul -> mat4s_mul + + +Print Options +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +1. **CGLM_DEFINE_PRINTS** +2. **CGLM_NO_PRINTS_NOOP** (use CGLM_DEFINE_PRINTS) + +Inline prints are only enabled in **DEBUG** mode or if **CGLM_DEFINE_PRINTS** is defined. +**glmc_** versions will always print too. + +Because **cglm** tried to enable print functions in debug mode and disable them in +release/production mode to eliminate printing costs when we do not need them. + +**cglm** checks **DEBUG** or **_DEBUG** macros to test debug mode, if these are not working for you then you can use +**CGLM_DEFINE_PRINTS** to force enable, or create a PR to introduce new macro to test against debugging mode. + +If DEBUG mode is not enabled then print functions will be emptied to eliminate print function calls. +You can disable this feature too by defining **CGLM_DEFINE_PRINTS** macro top of cglm header +or in project/build settings... + +3. **CGLM_PRINT_PRECISION** 5 + +precision. + +4. **CGLM_PRINT_MAX_TO_SHORT** 1e5 + +if a number is greater than this value then %g will be used, since this is shorten print you won't be able to see high precision. + +5. **CGLM_PRINT_COLOR** "\033[36m" +6. **CGLM_PRINT_COLOR_RESET** "\033[0m" + +You can disable colorful print output by defining **CGLM_PRINT_COLOR** and **CGLM_PRINT_COLOR_RESET** as empty macro. +Because some terminals may not support colors. diff --git a/cglm/docs/source/plane.rst b/cglm/docs/source/plane.rst new file mode 100644 index 0000000..b9afbed --- /dev/null +++ b/cglm/docs/source/plane.rst @@ -0,0 +1,33 @@ +.. default-domain:: C + +plane +================================================================================ + +Header: cglm/plane.h + +Plane extract functions are in frustum header and documented +in :doc:`frustum` page. + +**Definition of plane:** + +Plane equation: **Ax + By + Cz + D = 0** + +Plan is stored in **vec4** as **[A, B, C, D]**. (A, B, C) is normal and D is distance + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_plane_normalize` + + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_plane_normalize(vec4 plane) + + | normalizes a plane + + Parameters: + | *[in, out]* **plane** pnale to normalize diff --git a/cglm/docs/source/project.rst b/cglm/docs/source/project.rst new file mode 100644 index 0000000..161a2a7 --- /dev/null +++ b/cglm/docs/source/project.rst @@ -0,0 +1,119 @@ +.. default-domain:: C + +Project / UnProject +================================================================================ + +Header: cglm/project.h + +Viewport is required as *vec4* **[X, Y, Width, Height]** but this doesn't mean +that you should store it as **vec4**. You can convert your data representation +to vec4 before passing it to related functions. + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_unprojecti` +#. :c:func:`glm_unproject` +#. :c:func:`glm_project` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_unprojecti(vec3 pos, mat4 invMat, vec4 vp, vec3 dest) + + | maps the specified viewport coordinates into specified space [1] + the matrix should contain projection matrix. + + if you don't have ( and don't want to have ) an inverse matrix then use + :c:func:`glm_unproject` version. You may use existing inverse of matrix in somewhere + else, this is why **glm_unprojecti** exists to save inversion cost + + [1] space: + - if m = invProj: View Space + - if m = invViewProj: World Space + - if m = invMVP: Object Space + + You probably want to map the coordinates into object space + so use invMVP as m + + Computing viewProj: + + .. code-block:: c + + glm_mat4_mul(proj, view, viewProj); + glm_mat4_mul(viewProj, model, MVP); + glm_mat4_inv(viewProj, invMVP); + + Parameters: + | *[in]* **pos** point/position in viewport coordinates + | *[in]* **invMat** matrix (see brief) + | *[in]* **vp** viewport as [x, y, width, height] + | *[out]* **dest** unprojected coordinates + +.. c:function:: void glm_unproject(vec3 pos, mat4 m, vec4 vp, vec3 dest) + + | maps the specified viewport coordinates into specified space [1] + the matrix should contain projection matrix. + + this is same as :c:func:`glm_unprojecti` except this function get inverse matrix for + you. + + [1] space: + - if m = proj: View Space + - if m = viewProj: World Space + - if m = MVP: Object Space + + You probably want to map the coordinates into object space so use MVP as m + + Computing viewProj and MVP: + + .. code-block:: c + + glm_mat4_mul(proj, view, viewProj); + glm_mat4_mul(viewProj, model, MVP); + + Parameters: + | *[in]* **pos** point/position in viewport coordinates + | *[in]* **m** matrix (see brief) + | *[in]* **vp** viewport as [x, y, width, height] + | *[out]* **dest** unprojected coordinates + +.. c:function:: void glm_project(vec3 pos, mat4 m, vec4 vp, vec3 dest) + + | map object coordinates to window coordinates + + Computing MVP: + + .. code-block:: c + + glm_mat4_mul(proj, view, viewProj); + glm_mat4_mul(viewProj, model, MVP); + + Parameters: + | *[in]* **pos** object coordinates + | *[in]* **m** MVP matrix + | *[in]* **vp** viewport as [x, y, width, height] + | *[out]* **dest** projected coordinates + +.. c:function:: float glm_project_z(vec3 pos, mat4 m) + + | map object's z coordinate to window coordinates + + this is same as :c:func:`glm_project` except this function projects only Z coordinate + which reduces a few calculations and parameters. + + Computing MVP: + + .. code-block:: c + + glm_mat4_mul(proj, view, viewProj); + glm_mat4_mul(viewProj, model, MVP); + + Parameters: + | *[in]* **pos** object coordinates + | *[in]* **m** MVP matrix + + Returns: + projected z coordinate \ No newline at end of file diff --git a/cglm/docs/source/quat.rst b/cglm/docs/source/quat.rst new file mode 100644 index 0000000..eb6f987 --- /dev/null +++ b/cglm/docs/source/quat.rst @@ -0,0 +1,445 @@ +.. default-domain:: C + +quaternions +=========== + +Header: cglm/quat.h + + **Important:** *cglm* stores quaternion as **[x, y, z, w]** in memory + since **v0.4.0** it was **[w, x, y, z]** + before v0.4.0 ( **v0.3.5 and earlier** ). w is real part. + +What you can do with quaternions with existing functions is (Some of them): + +- You can rotate transform matrix using quaternion +- You can rotate vector using quaternion +- You can create view matrix using quaternion +- You can create a lookrotation (from source point to dest) + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. GLM_QUAT_IDENTITY_INIT +#. GLM_QUAT_IDENTITY + +Functions: + +1. :c:func:`glm_quat_identity` +#. :c:func:`glm_quat_identity_array` +#. :c:func:`glm_quat_init` +#. :c:func:`glm_quat` +#. :c:func:`glm_quatv` +#. :c:func:`glm_quat_copy` +#. :c:func:`glm_quat_from_vecs` +#. :c:func:`glm_quat_norm` +#. :c:func:`glm_quat_normalize` +#. :c:func:`glm_quat_normalize_to` +#. :c:func:`glm_quat_dot` +#. :c:func:`glm_quat_conjugate` +#. :c:func:`glm_quat_inv` +#. :c:func:`glm_quat_add` +#. :c:func:`glm_quat_sub` +#. :c:func:`glm_quat_real` +#. :c:func:`glm_quat_imag` +#. :c:func:`glm_quat_imagn` +#. :c:func:`glm_quat_imaglen` +#. :c:func:`glm_quat_angle` +#. :c:func:`glm_quat_axis` +#. :c:func:`glm_quat_mul` +#. :c:func:`glm_quat_mat4` +#. :c:func:`glm_quat_mat4t` +#. :c:func:`glm_quat_mat3` +#. :c:func:`glm_quat_mat3t` +#. :c:func:`glm_quat_lerp` +#. :c:func:`glm_quat_nlerp` +#. :c:func:`glm_quat_slerp` +#. :c:func:`glm_quat_slerp_longest` +#. :c:func:`glm_quat_look` +#. :c:func:`glm_quat_for` +#. :c:func:`glm_quat_forp` +#. :c:func:`glm_quat_rotatev` +#. :c:func:`glm_quat_rotate` +#. :c:func:`glm_quat_rotate_at` +#. :c:func:`glm_quat_rotate_atm` +#. :c:func:`glm_quat_make` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_quat_identity(versor q) + + | makes given quat to identity + + Parameters: + | *[in, out]* **q** quaternion + +.. c:function:: void glm_quat_identity_array(versor * __restrict q, size_t count) + + | make given quaternion array's each element identity quaternion + + Parameters: + | *[in, out]* **q** quat array (must be aligned (16) if alignment is not disabled) + | *[in]* **count** count of quaternions + +.. c:function:: void glm_quat_init(versor q, float x, float y, float z, float w) + + | inits quaternion with given values + + Parameters: + | *[out]* **q** quaternion + | *[in]* **x** imag.x + | *[in]* **y** imag.y + | *[in]* **z** imag.z + | *[in]* **w** w (real part) + +.. c:function:: void glm_quat(versor q, float angle, float x, float y, float z) + + | creates NEW quaternion with individual axis components + + | given axis will be normalized + + Parameters: + | *[out]* **q** quaternion + | *[in]* **angle** angle (radians) + | *[in]* **x** axis.x + | *[in]* **y** axis.y + | *[in]* **z** axis.z + +.. c:function:: void glm_quatv(versor q, float angle, vec3 axis) + + | creates NEW quaternion with axis vector + + | given axis will be normalized + + Parameters: + | *[out]* **q** quaternion + | *[in]* **angle** angle (radians) + | *[in]* **axis** axis (will be normalized) + +.. c:function:: void glm_quat_copy(versor q, versor dest) + + | copy quaternion to another one + + Parameters: + | *[in]* **q** source quaternion + | *[out]* **dest** destination quaternion + +.. c:function:: void glm_quat_from_vecs(vec3 a, vec3 b, versor dest) + + | compute unit quaternion needed to rotate a into b + + References: + * `Finding quaternion representing the rotation from one vector to another `_ + * `Quaternion from two vectors `_ + * `Angle between vectors `_ + + Parameters: + | *[in]* **a** unit vector + | *[in]* **b** unit vector + | *[in]* **dest** unit quaternion + +.. c:function:: float glm_quat_norm(versor q) + + | returns norm (magnitude) of quaternion + + Parameters: + | *[in]* **a** quaternion + + Returns: + norm (magnitude) + +.. c:function:: void glm_quat_normalize_to(versor q, versor dest) + + | normalize quaternion and store result in dest, original one will not be normalized + + Parameters: + | *[in]* **q** quaternion to normalize into + | *[out]* **dest** destination quaternion + +.. c:function:: void glm_quat_normalize(versor q) + + | normalize quaternion + + Parameters: + | *[in, out]* **q** quaternion + +.. c:function:: float glm_quat_dot(versor p, versor q) + + dot product of two quaternion + + Parameters: + | *[in]* **p** quaternion 1 + | *[in]* **q** quaternion 2 + + Returns: + dot product + +.. c:function:: void glm_quat_conjugate(versor q, versor dest) + + conjugate of quaternion + + Parameters: + | *[in]* **q** quaternion + | *[in]* **dest** conjugate + +.. c:function:: void glm_quat_inv(versor q, versor dest) + + inverse of non-zero quaternion + + Parameters: + | *[in]* **q** quaternion + | *[in]* **dest** inverse quaternion + +.. c:function:: void glm_quat_add(versor p, versor q, versor dest) + + add (componentwise) two quaternions and store result in dest + + Parameters: + | *[in]* **p** quaternion 1 + | *[in]* **q** quaternion 2 + | *[in]* **dest** result quaternion + +.. c:function:: void glm_quat_sub(versor p, versor q, versor dest) + + subtract (componentwise) two quaternions and store result in dest + + Parameters: + | *[in]* **p** quaternion 1 + | *[in]* **q** quaternion 2 + | *[in]* **dest** result quaternion + +.. c:function:: float glm_quat_real(versor q) + + returns real part of quaternion + + Parameters: + | *[in]* **q** quaternion + + Returns: + real part (quat.w) + +.. c:function:: void glm_quat_imag(versor q, vec3 dest) + + returns imaginary part of quaternion + + Parameters: + | *[in]* **q** quaternion + | *[out]* **dest** imag + +.. c:function:: void glm_quat_imagn(versor q, vec3 dest) + + returns normalized imaginary part of quaternion + + Parameters: + | *[in]* **q** quaternion + | *[out]* **dest** imag + +.. c:function:: float glm_quat_imaglen(versor q) + + returns length of imaginary part of quaternion + + Parameters: + | *[in]* **q** quaternion + + Returns: + norm of imaginary part + +.. c:function:: float glm_quat_angle(versor q) + + returns angle of quaternion + + Parameters: + | *[in]* **q** quaternion + + Returns: + angles of quat (radians) + +.. c:function:: void glm_quat_axis(versor q, versor dest) + + axis of quaternion + + Parameters: + | *[in]* **p** quaternion + | *[out]* **dest** axis of quaternion + +.. c:function:: void glm_quat_mul(versor p, versor q, versor dest) + + | multiplies two quaternion and stores result in dest + + | this is also called Hamilton Product + + | According to WikiPedia: + | The product of two rotation quaternions [clarification needed] will be + equivalent to the rotation q followed by the rotation p + + Parameters: + | *[in]* **p** quaternion 1 (first rotation) + | *[in]* **q** quaternion 2 (second rotation) + | *[out]* **dest** result quaternion + +.. c:function:: void glm_quat_mat4(versor q, mat4 dest) + + | convert quaternion to mat4 + + Parameters: + | *[in]* **q** quaternion + | *[out]* **dest** result matrix + +.. c:function:: void glm_quat_mat4t(versor q, mat4 dest) + + | convert quaternion to mat4 (transposed). This is transposed version of glm_quat_mat4 + + Parameters: + | *[in]* **q** quaternion + | *[out]* **dest** result matrix + +.. c:function:: void glm_quat_mat3(versor q, mat3 dest) + + | convert quaternion to mat3 + + Parameters: + | *[in]* **q** quaternion + | *[out]* **dest** result matrix + +.. c:function:: void glm_quat_mat3t(versor q, mat3 dest) + + | convert quaternion to mat3 (transposed). This is transposed version of glm_quat_mat3 + + Parameters: + | *[in]* **q** quaternion + | *[out]* **dest** result matrix + +.. c:function:: void glm_quat_lerp(versor from, versor to, float t, versor dest) + + | interpolates between two quaternions + | using spherical linear interpolation (LERP) + + Parameters: + | *[in]* **from** from + | *[in]* **to** to + | *[in]* **t** interpolant (amount) clamped between 0 and 1 + | *[out]* **dest** result quaternion + +.. c:function:: void glm_quat_nlerp(versor q, versor r, float t, versor dest) + + | interpolates between two quaternions + | taking the shortest rotation path using + | normalized linear interpolation (NLERP) + + | This is a cheaper alternative to slerp; most games use nlerp + | for animations as it visually makes little difference. + + References: + * `Understanding Slerp, Then Not Using it `_ + * `Lerp, Slerp and Nlerp `_ + + Parameters: + | *[in]* **from** from + | *[in]* **to** to + | *[in]* **t** interpolant (amount) clamped between 0 and 1 + | *[out]* **dest** result quaternion + +.. c:function:: void glm_quat_slerp(versor q, versor r, float t, versor dest) + + | interpolates between two quaternions + | using spherical linear interpolation (SLERP) + + Parameters: + | *[in]* **from** from + | *[in]* **to** to + | *[in]* **t** interpolant (amount) clamped between 0 and 1 + | *[out]* **dest** result quaternion + +.. c:function:: void glm_quat_slerp_longest(versor q, versor r, float t, versor dest) + + | interpolates between two quaternions + | using spherical linear interpolation (SLERP) and always takes the longest path + + Parameters: + | *[in]* **from** from + | *[in]* **to** to + | *[in]* **t** interpolant (amount) clamped between 0 and 1 + | *[out]* **dest** result quaternion + +.. c:function:: void glm_quat_look(vec3 eye, versor ori, mat4 dest) + + | creates view matrix using quaternion as camera orientation + + Parameters: + | *[in]* **eye** eye + | *[in]* **ori** orientation in world space as quaternion + | *[out]* **dest** result matrix + +.. c:function:: void glm_quat_for(vec3 dir, vec3 up, versor dest) + + | creates look rotation quaternion + + Parameters: + | *[in]* **dir** direction to look + | *[in]* **up** up vector + | *[out]* **dest** result matrix + +.. c:function:: void glm_quat_forp(vec3 from, vec3 to, vec3 up, versor dest) + + | creates look rotation quaternion using source and destination positions p suffix stands for position + + | this is similar to glm_quat_for except this computes direction for glm_quat_for for you. + + Parameters: + | *[in]* **from** source point + | *[in]* **to** destination point + | *[in]* **up** up vector + | *[out]* **dest** result matrix + +.. c:function:: void glm_quat_rotatev(versor q, vec3 v, vec3 dest) + + | crotate vector using using quaternion + + Parameters: + | *[in]* **q** quaternion + | *[in]* **v** vector to rotate + | *[out]* **dest** rotated vector + +.. c:function:: void glm_quat_rotate(mat4 m, versor q, mat4 dest) + + | rotate existing transform matrix using quaternion + + instead of passing identity matrix, consider to use quat_mat4 functions + + Parameters: + | *[in]* **m** existing transform matrix to rotate + | *[in]* **q** quaternion + | *[out]* **dest** rotated matrix/transform + +.. c:function:: void glm_quat_rotate_at(mat4 m, versor q, vec3 pivot) + + | rotate existing transform matrix using quaternion at pivot point + + Parameters: + | *[in, out]* **m** existing transform matrix to rotate + | *[in]* **q** quaternion + | *[in]* **pivot** pivot + +.. c:function:: void glm_quat_rotate_atm(mat4 m, versor q, vec3 pivot) + + | rotate NEW transform matrix using quaternion at pivot point + | this creates rotation matrix, it assumes you don't have a matrix + + | this should work faster than glm_quat_rotate_at because it reduces one glm_translate. + + Parameters: + | *[in, out]* **m** existing transform matrix to rotate + | *[in]* **q** quaternion + | *[in]* **pivot** pivot + +.. c:function:: void glm_quat_make(const float * __restrict src, versor dest) + + Create quaternion from pointer + + .. note:: **@src** must contain at least 4 elements. cglm store quaternions as [x, y, z, w]. + + Parameters: + | *[in]* **src** pointer to an array of floats + | *[out]* **dest** destination quaternion diff --git a/cglm/docs/source/ray.rst b/cglm/docs/source/ray.rst new file mode 100644 index 0000000..1538a1f --- /dev/null +++ b/cglm/docs/source/ray.rst @@ -0,0 +1,69 @@ +.. default-domain:: C + +ray +==== + +Header: cglm/ray.h + +This is for collision-checks used by ray-tracers and the like. + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_ray_triangle` +#. :c:func:`glm_ray_sphere` +#. :c:func:`glm_ray_at` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: bool glm_ray_triangle(vec3 origin, vec3 direction, vec3 v0, vec3 v1, vec3 v2, float *d) + + Möller–Trumbore ray-triangle intersection algorithm + + Parameters: + | *[in]* **origin** origin of ray + | *[in]* **direction** direction of ray + | *[in]* **v0** first vertex of triangle + | *[in]* **v1** second vertex of triangle + | *[in]* **v2** third vertex of triangle + | *[in, out]* **d** float pointer to save distance to intersection + | *[out]* **intersection** whether there is intersection + +.. c:function:: bool glm_ray_sphere(vec3 origin, vec3 dir, vec4 s, float * __restrict t1, float * __restrict t2) + + ray sphere intersection + + returns false if there is no intersection if true: + + - t1 > 0, t2 > 0: ray intersects the sphere at t1 and t2 both ahead of the origin + - t1 < 0, t2 > 0: ray starts inside the sphere, exits at t2 + - t1 < 0, t2 < 0: no intersection ahead of the ray ( returns false ) + - the caller can check if the intersection points (t1 and t2) fall within a + specific range (for example, tmin < t1, t2 < tmax) to determine if the + intersections are within a desired segment of the ray + + Parameters: + | *[in]* **origin** ray origin + | *[in]* **dir** normalized ray direction + | *[in]* **s** sphere [center.x, center.y, center.z, radii] + | *[out]* **t1** near point1 (closer to origin) + | *[out]* **t2** far point2 (farther from origin) + + Return: + | whether there is intersection + +.. c:function:: bool glm_ray_at(vec3 orig, vec3 dir, float t, vec3 point) + + point using t by 𝐏(𝑡)=𝐀+𝑡𝐛 + + Parameters: + | *[in]* **origin** ray origin + | *[in]* **dir** ray direction + | *[out]* **t** parameter + | *[out]* **point** point at t + + Return: + | point at t diff --git a/cglm/docs/source/sphere.rst b/cglm/docs/source/sphere.rst new file mode 100644 index 0000000..d4f57b6 --- /dev/null +++ b/cglm/docs/source/sphere.rst @@ -0,0 +1,74 @@ +.. default-domain:: C + +Sphere +================================================================================ + +Header: cglm/sphere.h + +**Definition of sphere:** + +Sphere Representation in cglm is *vec4*: **[center.x, center.y, center.z, radii]** + +You can call any vec3 function by passing sphere. Because first three elements +defines center of sphere. + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_sphere_radii` +#. :c:func:`glm_sphere_transform` +#. :c:func:`glm_sphere_merge` +#. :c:func:`glm_sphere_sphere` +#. :c:func:`glm_sphere_point` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: float glm_sphere_radii(vec4 s) + + | helper for getting sphere radius + + Parameters: + | *[in]* **s** sphere + + Returns: + returns radii + +.. c:function:: void glm_sphere_transform(vec4 s, mat4 m, vec4 dest) + + | apply transform to sphere, it is just wrapper for glm_mat4_mulv3 + + Parameters: + | *[in]* **s** sphere + | *[in]* **m** transform matrix + | *[out]* **dest** transformed sphere + +.. c:function:: void glm_sphere_merge(vec4 s1, vec4 s2, vec4 dest) + + | merges two spheres and creates a new one + + two sphere must be in same space, for instance if one in world space then + the other must be in world space too, not in local space. + + Parameters: + | *[in]* **s1** sphere 1 + | *[in]* **s2** sphere 2 + | *[out]* **dest** merged/extended sphere + +.. c:function:: bool glm_sphere_sphere(vec4 s1, vec4 s2) + + | check if two sphere intersects + + Parameters: + | *[in]* **s1** sphere + | *[in]* **s2** other sphere + +.. c:function:: bool glm_sphere_point(vec4 s, vec3 point) + + | check if sphere intersects with point + + Parameters: + | *[in]* **s** sphere + | *[in]* **point** point diff --git a/cglm/docs/source/sphinx-static/theme_overrides.css b/cglm/docs/source/sphinx-static/theme_overrides.css new file mode 100644 index 0000000..0952e2c --- /dev/null +++ b/cglm/docs/source/sphinx-static/theme_overrides.css @@ -0,0 +1,12 @@ +@media screen { + /* content column + * + * RTD theme's default is 800px as max width for the content, but we have + * tables with tons of columns, which need the full width of the view-port. + * + * Comment from yocto project theme_overrides.css + */ + + .wy-nav-content{ max-width: none; } + +} diff --git a/cglm/docs/source/troubleshooting.rst b/cglm/docs/source/troubleshooting.rst new file mode 100644 index 0000000..ea360c7 --- /dev/null +++ b/cglm/docs/source/troubleshooting.rst @@ -0,0 +1,99 @@ +.. default-domain:: C + +Troubleshooting +================================================================================ + +It is possible that you may sometimes get crashes or wrong results. +Follow these topics + +Memory Allocation: +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Recall that **cglm** does not alloc any memory on the heap. +cglm functions work like memcpy; they copy data from src, +make calculations, then copy the result to dest. + +You are responsible for allocation of **src** and **dest** parameters. + +Alignment: +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +**vec4** and **mat4** types require 16 byte alignment. +These types are marked with the align attribute to let the compiler know about this +requirement. + +Since MSVC (Windows) throws this error: + +**"formal parameter with requested alignment of 16 won't be aligned"** + +The alignment attribute has been commented out for MSVC + +.. code-block:: c + + #if defined(_MSC_VER) + # define CGLM_ALIGN(X) /* __declspec(align(X)) */ + #else + # define CGLM_ALIGN(X) __attribute((aligned(X))) + #endif. + +So MSVC may not know about alignment requirements when creating variables. +The interesting thing is that, if I remember correctly, Visual Studio 2017 +doesn't throw the above error. So we may uncomment that line for Visual Studio 2017, +you may do it yourself. + +**This MSVC issue is still in TODOs.** + +**UPDATE:** Starting with v0.4.5, cglm provides an option to disable the alignment requirement. +Also, alignment is disabled for older msvc versions by default. Now alignment is only required in Visual Studio 2017 version 15.6+ if the CGLM_ALL_UNALIGNED macro is not defined. + +Crashes, Invalid Memory Access: +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Most likely, you are trying to write to an invalid memory location. + +You may have used a wrong function for what you want to do. + +For example, you may have called a **glm_vec4_** function for a **vec3** data type. +It will try to write 32 bytes, but since **vec3** is 24 bytes, it should throw +a memory access error or exit the app without saying anything. + +**UPDATE - IMPORTANT:** + + | On MSVC or some other compilers, if alignment is enabled (default) then double check alignment requirements if you got a crash. + + | If you send GLM_VEC4_ONE or similar macros directly to a function, it may crash. + | Because the compiler may not apply alignment as defined on **typedef** to that macro while passing it (on stack) to a function. + +Wrong Results: +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Again, you may have used a wrong function. + +For instance if you use **glm_normalize()** or **glm_vec3_normalize()** for a **vec4**, +it will assume that the passed param is a **vec3**, and will normalize it for **vec3**. +Since you need a **vec4** to be normalized in your case, you will get wrong results. + +Accessing a vec4 type with vec3 functions is valid, you will not get any error, exception or crash. +You only get wrong results if you don't know what you are doing! + +So be careful, when your IDE (Xcode, Visual Studio ...) tries to autocomplete function names, READ IT :) + +**Also implementation may be wrong, please let us know by creating an issue on Github.** + +BAD_ACCESS : Thread 1: EXC_BAD_ACCESS (code=EXC_I386_GPFLT) or Similar Errors/Crashes +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +This is a similar issue with alignment. For instance if you compiled **cglm** with +AVX (**-mavx**, intentionally or not) and if you use **cglm** in an environment that doesn't +support AVX (or if AVX is disabled intentionally) e.g. environment that max support SSE2/3/4, +then you probably get **BAD ACCESS** or similar... + +Because if you compile **cglm** with AVX it aligns **mat4** with 32 byte boundary, +and your project aligns that as a 16 byte boundary... + +Check alignment, supported vector extension, or simd in **cglm** and linked projects... + +Other Issues? +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +**Please let us know by creating an issue on Github.** diff --git a/cglm/docs/source/util.rst b/cglm/docs/source/util.rst new file mode 100644 index 0000000..ca7715f --- /dev/null +++ b/cglm/docs/source/util.rst @@ -0,0 +1,182 @@ +.. default-domain:: C + +utils / helpers +================================================================================ + +Header: cglm/util.h + + + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_sign` +#. :c:func:`glm_signf` +#. :c:func:`glm_rad` +#. :c:func:`glm_deg` +#. :c:func:`glm_make_rad` +#. :c:func:`glm_make_deg` +#. :c:func:`glm_pow2` +#. :c:func:`glm_min` +#. :c:func:`glm_max` +#. :c:func:`glm_clamp` +#. :c:func:`glm_lerp` +#. :c:func:`glm_swapf` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: int glm_sign(int val) + + | returns sign of 32 bit integer as +1, -1, 0 + + | **Important**: It returns 0 for zero input + + Parameters: + | *[in]* **val** an integer + + Returns: + sign of given number + +.. c:function:: float glm_signf(float val) + + | returns sign of 32 bit integer as +1.0, -1.0, 0.0 + + | **Important**: It returns 0.0f for zero input + + Parameters: + | *[in]* **val** a float + + Returns: + sign of given number + +.. c:function:: float glm_rad(float deg) + + | convert degree to radians + + Parameters: + | *[in]* **deg** angle in degrees + +.. c:function:: float glm_deg(float rad) + + | convert radians to degree + + Parameters: + | *[in]* **rad** angle in radians + +.. c:function:: void glm_make_rad(float *degm) + + | convert existing degree to radians. this will override degrees value + + Parameters: + | *[in, out]* **deg** pointer to angle in degrees + +.. c:function:: void glm_make_deg(float *rad) + + | convert existing radians to degree. this will override radians value + + Parameters: + | *[in, out]* **rad** pointer to angle in radians + +.. c:function:: float glm_pow2(float x) + + | multiplies given parameter with itself = x * x or powf(x, 2) + + Parameters: + | *[in]* **x** value + + Returns: + square of a given number + +.. c:function:: float glm_min(float a, float b) + + | returns minimum of given two values + + Parameters: + | *[in]* **a** number 1 + | *[in]* **b** number 2 + + Returns: + minimum value + +.. c:function:: float glm_max(float a, float b) + + | returns maximum of given two values + + Parameters: + | *[in]* **a** number 1 + | *[in]* **b** number 2 + + Returns: + maximum value + +.. c:function:: void glm_clamp(float val, float minVal, float maxVal) + + constrain a value to lie between two further values + + Parameters: + | *[in]* **val** input value + | *[in]* **minVal** minimum value + | *[in]* **maxVal** maximum value + + Returns: + clamped value + +.. c:function:: float glm_lerp(float from, float to, float t) + + linear interpolation between two number + + | formula: from + s * (to - from) + + Parameters: + | *[in]* **from** from value + | *[in]* **to** to value + | *[in]* **t** interpolant (amount) clamped between 0 and 1 + + Returns: + interpolated value + +.. c:function:: bool glm_eq(float a, float b) + + check if two float equal with using EPSILON + + Parameters: + | *[in]* **a** a + | *[in]* **b** b + + Returns: + true if a and b are equal + +.. c:function:: float glm_percent(float from, float to, float current) + + percentage of current value between start and end value + + Parameters: + | *[in]* **from** from value + | *[in]* **to** to value + | *[in]* **current** value between from and to values + + Returns: + percentage of current value + +.. c:function:: float glm_percentc(float from, float to, float current) + + clamped percentage of current value between start and end value + + Parameters: + | *[in]* **from** from value + | *[in]* **to** to value + | *[in]* **current** value between from and to values + + Returns: + clamped normalized percent (0-100 in 0-1) + +.. c:function:: void glm_swapf(float *a, float *b) + + swap two float values + + Parameters: + | *[in]* **a** float 1 + | *[in]* **b** float 2 diff --git a/cglm/docs/source/vec2-ext.rst b/cglm/docs/source/vec2-ext.rst new file mode 100644 index 0000000..70ef4f6 --- /dev/null +++ b/cglm/docs/source/vec2-ext.rst @@ -0,0 +1,161 @@ +.. default-domain:: C + +vec2 extra +========== + +Header: cglm/vec2-ext.h + +There are some functions are in called in extra header. These are called extra +because they are not used like other functions in vec2.h in the future some of +these functions ma be moved to vec2 header. + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_vec2_fill` +#. :c:func:`glm_vec2_eq` +#. :c:func:`glm_vec2_eq_eps` +#. :c:func:`glm_vec2_eq_all` +#. :c:func:`glm_vec2_eqv` +#. :c:func:`glm_vec2_eqv_eps` +#. :c:func:`glm_vec2_max` +#. :c:func:`glm_vec2_min` +#. :c:func:`glm_vec2_isnan` +#. :c:func:`glm_vec2_isinf` +#. :c:func:`glm_vec2_isvalid` +#. :c:func:`glm_vec2_sign` +#. :c:func:`glm_vec2_abs` +#. :c:func:`glm_vec2_fract` +#. :c:func:`glm_vec2_floor` +#. :c:func:`glm_vec2_sqrt` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_vec2_fill(vec2 v, float val) + + fill a vector with specified value + + Parameters: + | *[in,out]* **dest** destination + | *[in]* **val** value + + +.. c:function:: bool glm_vec2_eq(vec2 v, float val) + + check if vector is equal to value (without epsilon) + + Parameters: + | *[in]* **v** vector + | *[in]* **val** value + +.. c:function:: bool glm_vec2_eq_eps(vec2 v, float val) + + check if vector is equal to value (with epsilon) + + Parameters: + | *[in]* **v** vector + | *[in]* **val** value + +.. c:function:: bool glm_vec2_eq_all(vec2 v) + + check if vectors members are equal (without epsilon) + + Parameters: + | *[in]* **v** vector + +.. c:function:: bool glm_vec2_eqv(vec2 v1, vec2 v2) + + check if vector is equal to another (without epsilon) vector + + Parameters: + | *[in]* **vec** vector 1 + | *[in]* **vec** vector 2 + +.. c:function:: bool glm_vec2_eqv_eps(vec2 v1, vec2 v2) + + check if vector is equal to another (with epsilon) + + Parameters: + | *[in]* **v1** vector1 + | *[in]* **v2** vector2 + +.. c:function:: float glm_vec2_max(vec2 v) + + max value of vector + + Parameters: + | *[in]* **v** vector + +.. c:function:: float glm_vec2_min(vec2 v) + + min value of vector + + Parameters: + | *[in]* **v** vector + +.. c:function:: bool glm_vec2_isnan(vec2 v) + + | check if one of items is NaN (not a number) + | you should only use this in DEBUG mode or very critical asserts + + Parameters: + | *[in]* **v** vector + +.. c:function:: bool glm_vec2_isinf(vec2 v) + + | check if one of items is INFINITY + | you should only use this in DEBUG mode or very critical asserts + + Parameters: + | *[in]* **v** vector + +.. c:function:: bool glm_vec2_isvalid(vec2 v) + + | check if all items are valid number + | you should only use this in DEBUG mode or very critical asserts + + Parameters: + | *[in]* **v** vector + +.. c:function:: void glm_vec2_sign(vec2 v, vec2 dest) + + get sign of 32 bit float as +1, -1, 0 + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** sign vector (only keeps signs as -1, 0, -1) + +.. c:function:: void glm_vec2_abs(vec2 v, vec2 dest) + + absolute value of each vector item + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec2_fract(vec2 v, vec2 dest) + + get fractional part of each vector item + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec2_floor(vec2 v, vec2 dest) + + floor value of each vector item + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec2_sqrt(vec2 v, vec2 dest) + + square root of each vector item + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination vector (sqrt(v)) diff --git a/cglm/docs/source/vec2.rst b/cglm/docs/source/vec2.rst new file mode 100644 index 0000000..e9e38ca --- /dev/null +++ b/cglm/docs/source/vec2.rst @@ -0,0 +1,424 @@ +.. default-domain:: C + +vec2 +==== + +Header: cglm/vec2.h + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. GLM_VEC2_ONE_INIT +#. GLM_VEC2_ZERO_INIT +#. GLM_VEC2_ONE +#. GLM_VEC2_ZERO + +Functions: + +1. :c:func:`glm_vec2` +#. :c:func:`glm_vec2_copy` +#. :c:func:`glm_vec2_zero` +#. :c:func:`glm_vec2_one` +#. :c:func:`glm_vec2_dot` +#. :c:func:`glm_vec2_cross` +#. :c:func:`glm_vec2_norm2` +#. :c:func:`glm_vec2_norm` +#. :c:func:`glm_vec2_add` +#. :c:func:`glm_vec2_adds` +#. :c:func:`glm_vec2_sub` +#. :c:func:`glm_vec2_subs` +#. :c:func:`glm_vec2_mul` +#. :c:func:`glm_vec2_scale` +#. :c:func:`glm_vec2_scale_as` +#. :c:func:`glm_vec2_div` +#. :c:func:`glm_vec2_divs` +#. :c:func:`glm_vec2_addadd` +#. :c:func:`glm_vec2_subadd` +#. :c:func:`glm_vec2_muladd` +#. :c:func:`glm_vec2_muladds` +#. :c:func:`glm_vec2_maxadd` +#. :c:func:`glm_vec2_minadd` +#. :c:func:`glm_vec2_negate` +#. :c:func:`glm_vec2_negate_to` +#. :c:func:`glm_vec2_normalize` +#. :c:func:`glm_vec2_normalize_to` +#. :c:func:`glm_vec2_rotate` +#. :c:func:`glm_vec2_center` +#. :c:func:`glm_vec2_distance2` +#. :c:func:`glm_vec2_distance` +#. :c:func:`glm_vec2_maxv` +#. :c:func:`glm_vec2_minv` +#. :c:func:`glm_vec2_clamp` +#. :c:func:`glm_vec2_lerp` +#. :c:func:`glm_vec2_make` +#. :c:func:`glm_vec2_reflect` +#. :c:func:`glm_vec2_refract` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_vec2(float * v, vec2 dest) + + init vec2 using vec3 or vec4 + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination + +.. c:function:: void glm_vec2_copy(vec2 a, vec2 dest) + + copy all members of [a] to [dest] + + Parameters: + | *[in]* **a** source + | *[out]* **dest** destination + +.. c:function:: void glm_vec2_zero(vec2 v) + + makes all members 0.0f (zero) + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: void glm_vec2_one(vec2 v) + + makes all members 1.0f (one) + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: float glm_vec2_dot(vec2 a, vec2 b) + + dot product of vec2 + + Parameters: + | *[in]* **a** vector1 + | *[in]* **b** vector2 + + Returns: + dot product + +.. c:function:: void glm_vec2_cross(vec2 a, vec2 b, vec2 d) + + cross product of two vector (RH) + + | ref: http://allenchou.net/2013/07/cross-product-of-2d-vectors/ + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** destination + + Returns: + Z component of cross product + +.. c:function:: float glm_vec2_norm2(vec2 v) + + norm * norm (magnitude) of vector + + we can use this func instead of calling norm * norm, because it would call + sqrtf function twice but with this func we can avoid func call, maybe this is + not good name for this func + + Parameters: + | *[in]* **v** vector + + Returns: + square of norm / magnitude + +.. c:function:: float glm_vec2_norm(vec2 vec) + + | euclidean norm (magnitude), also called L2 norm + | this will give magnitude of vector in euclidean space + + Parameters: + | *[in]* **vec** vector + +.. c:function:: void glm_vec2_add(vec2 a, vec2 b, vec2 dest) + + add a vector to b vector store result in dest + + Parameters: + | *[in]* **a** vector1 + | *[in]* **b** vector2 + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec2_adds(vec2 a, float s, vec2 dest) + + add scalar to v vector store result in dest (d = v + vec(s)) + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec2_sub(vec2 v1, vec2 v2, vec2 dest) + + subtract b vector from a vector store result in dest (d = v1 - v2) + + Parameters: + | *[in]* **a** vector1 + | *[in]* **b** vector2 + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec2_subs(vec2 v, float s, vec2 dest) + + subtract scalar from v vector store result in dest (d = v - vec(s)) + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec2_mul(vec2 a, vec2 b, vec2 d) + + multiply two vector (component-wise multiplication) + + Parameters: + | *[in]* **a** vector + | *[in]* **b** scalar + | *[out]* **d** result = (a[0] * b[0], a[1] * b[1], a[2] * b[2]) + +.. c:function:: void glm_vec2_scale(vec2 v, float s, vec2 dest) + + multiply/scale vec2 vector with scalar: result = v * s + + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec2_scale_as(vec2 v, float s, vec2 dest) + + make vec2 vector scale as specified: result = unit(v) * s + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec2_div(vec2 a, vec2 b, vec2 dest) + + div vector with another component-wise division: d = a / b + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** result = (a[0] / b[0], a[1] / b[1], a[2] / b[2]) + +.. c:function:: void glm_vec2_divs(vec2 v, float s, vec2 dest) + + div vector with scalar: d = v / s + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** result = (a[0] / s, a[1] / s, a[2] / s]) + +.. c:function:: void glm_vec2_addadd(vec2 a, vec2 b, vec2 dest) + + | add two vectors and add result to sum + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** dest += (a + b) + +.. c:function:: void glm_vec2_subadd(vec2 a, vec2 b, vec2 dest) + + | sub two vectors and add result to sum + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** dest += (a - b) + +.. c:function:: void glm_vec2_muladd(vec2 a, vec2 b, vec2 dest) + + | mul two vectors and add result to sum + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** dest += (a * b) + +.. c:function:: void glm_vec2_muladds(vec2 a, float s, vec2 dest) + + | mul vector with scalar and add result to sum + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector + | *[in]* **s** scalar + | *[out]* **dest** dest += (a * b) + +.. c:function:: void glm_vec2_maxadd(vec2 a, vec2 b, vec2 dest) + + | add max of two vector to result/dest + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** dest += (a * b) + +.. c:function:: void glm_vec2_minadd(vec2 a, vec2 b, vec2 dest) + + | add min of two vector to result/dest + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** dest += (a * b) + +.. c:function:: void glm_vec2_negate(vec2 v) + + negate vector components + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: void glm_vec2_negate_to(vec2 v, vec2 dest) + + negate vector components and store result in dest + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** negated vector + +.. c:function:: void glm_vec2_normalize(vec2 v) + + normalize vec2 and store result in same vec + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: void glm_vec2_normalize_to(vec2 vec, vec2 dest) + + normalize vec2 to dest + + Parameters: + | *[in]* **vec** source + | *[out]* **dest** destination + +.. c:function:: void glm_vec2_rotate(vec2 v, float angle, vec2 dest) + + rotate vec2 around axis by angle using Rodrigues' rotation formula + + Parameters: + | *[in]* **v** vector + | *[in]* **axis** axis vector + | *[out]* **dest** destination + +.. c:function:: void glm_vec2_center(vec2 v1, vec2 v2, vec2 dest) + + find center point of two vector + + Parameters: + | *[in]* **v1** vector1 + | *[in]* **v2** vector2 + | *[out]* **dest** center point + +.. c:function:: float glm_vec2_distance2(vec2 v1, vec2 v2) + + squared distance between two vectors + + Parameters: + | *[in]* **mat** vector1 + | *[in]* **row1** vector2 + + Returns: + | squared distance (distance * distance) + +.. c:function:: float glm_vec2_distance(vec2 v1, vec2 v2) + + distance between two vectors + + Parameters: + | *[in]* **mat** vector1 + | *[in]* **row1** vector2 + + Returns: + | distance + +.. c:function:: void glm_vec2_maxv(vec2 v1, vec2 v2, vec2 dest) + + max values of vectors + + Parameters: + | *[in]* **v1** vector1 + | *[in]* **v2** vector2 + | *[out]* **dest** destination + +.. c:function:: void glm_vec2_minv(vec2 v1, vec2 v2, vec2 dest) + + min values of vectors + + Parameters: + | *[in]* **v1** vector1 + | *[in]* **v2** vector2 + | *[out]* **dest** destination + +.. c:function:: void glm_vec2_clamp(vec2 v, float minVal, float maxVal) + + constrain a value to lie between two further values + + Parameters: + | *[in, out]* **v** vector + | *[in]* **minVal** minimum value + | *[in]* **maxVal** maximum value + +.. c:function:: void glm_vec2_lerp(vec2 from, vec2 to, float t, vec2 dest) + + linear interpolation between two vector + + | formula: from + s * (to - from) + + Parameters: + | *[in]* **from** from value + | *[in]* **to** to value + | *[in]* **t** interpolant (amount) clamped between 0 and 1 + | *[out]* **dest** destination + +.. c:function:: void glm_vec2_make(const float * __restrict src, vec2 dest) + + Create two dimensional vector from pointer + + .. note:: **@src** must contain at least 2 elements. + + Parameters: + | *[in]* **src** pointer to an array of floats + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec2_reflect(vec2 v, vec2 n, vec2 dest) + + Reflection vector using an incident ray and a surface normal + + Parameters: + | *[in]* **v** incident vector + | *[in]* **n** *❗️ normalized ❗️* normal vector + | *[out]* **dest** destination: reflection result + +.. c:function:: bool glm_vec2_refract(vec2 v, vec2 n, float eta, vec2 dest) + + Computes refraction vector for an incident vector and a surface normal. + + Calculates the refraction vector based on Snell's law. If total internal reflection + occurs (angle too great given eta), dest is set to zero and returns false. + Otherwise, computes refraction vector, stores it in dest, and returns true. + + Parameters: + | *[in]* **v** *❗️ normalized ❗️* incident vector + | *[in]* **n** *❗️ normalized ❗️* normal vector + | *[in]* **eta** ratio of indices of refraction (incident/transmitted) + | *[out]* **dest** refraction vector if refraction occurs; zero vector otherwise + + Returns: + returns true if refraction occurs; false if total internal reflection occurs. diff --git a/cglm/docs/source/vec3-ext.rst b/cglm/docs/source/vec3-ext.rst new file mode 100644 index 0000000..9668932 --- /dev/null +++ b/cglm/docs/source/vec3-ext.rst @@ -0,0 +1,178 @@ +.. default-domain:: C + +vec3 extra +========== + +Header: cglm/vec3-ext.h + +There are some functions are in called in extra header. These are called extra +because they are not used like other functions in vec3.h in the future some of +these functions ma be moved to vec3 header. + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_vec3_mulv` +#. :c:func:`glm_vec3_broadcast` +#. :c:func:`glm_vec3_eq` +#. :c:func:`glm_vec3_eq_eps` +#. :c:func:`glm_vec3_eq_all` +#. :c:func:`glm_vec3_eqv` +#. :c:func:`glm_vec3_eqv_eps` +#. :c:func:`glm_vec3_max` +#. :c:func:`glm_vec3_min` +#. :c:func:`glm_vec3_isnan` +#. :c:func:`glm_vec3_isinf` +#. :c:func:`glm_vec3_isvalid` +#. :c:func:`glm_vec3_sign` +#. :c:func:`glm_vec3_abs` +#. :c:func:`glm_vec3_fract` +#. :c:func:`glm_vec3_floor` +#. :c:func:`glm_vec3_sqrt` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_vec3_mulv(vec3 a, vec3 b, vec3 d) + + multiplies individual items + + Parameters: + | *[in]* **a** vec1 + | *[in]* **b** vec2 + | *[out]* **d** destination (v1[0] * v2[0], v1[1] * v2[1], v1[2] * v2[2]) + +.. c:function:: void glm_vec3_broadcast(float val, vec3 d) + + fill a vector with specified value + + Parameters: + | *[in]* **val** value + | *[out]* **dest** destination + +.. c:function:: void glm_vec3_fill(vec3 v, float val) + + fill a vector with specified value + + Parameters: + | *[out]* **v** vector + | *[in]* **val** value + +.. c:function:: bool glm_vec3_eq(vec3 v, float val) + + check if vector is equal to value (without epsilon) + + Parameters: + | *[in]* **v** vector + | *[in]* **val** value + +.. c:function:: bool glm_vec3_eq_eps(vec3 v, float val) + + check if vector is equal to value (with epsilon) + + Parameters: + | *[in]* **v** vector + | *[in]* **val** value + +.. c:function:: bool glm_vec3_eq_all(vec3 v) + + check if vectors members are equal (without epsilon) + + Parameters: + | *[in]* **v** vector + +.. c:function:: bool glm_vec3_eqv(vec3 v1, vec3 v2) + + check if vector is equal to another (without epsilon) vector + + Parameters: + | *[in]* **vec** vector 1 + | *[in]* **vec** vector 2 + +.. c:function:: bool glm_vec3_eqv_eps(vec3 v1, vec3 v2) + + check if vector is equal to another (with epsilon) + + Parameters: + | *[in]* **v1** vector1 + | *[in]* **v2** vector2 + +.. c:function:: float glm_vec3_max(vec3 v) + + max value of vector + + Parameters: + | *[in]* **v** vector + +.. c:function:: float glm_vec3_min(vec3 v) + + min value of vector + + Parameters: + | *[in]* **v** vector + +.. c:function:: bool glm_vec3_isnan(vec3 v) + + | check if one of items is NaN (not a number) + | you should only use this in DEBUG mode or very critical asserts + + Parameters: + | *[in]* **v** vector + +.. c:function:: bool glm_vec3_isinf(vec3 v) + + | check if one of items is INFINITY + | you should only use this in DEBUG mode or very critical asserts + + Parameters: + | *[in]* **v** vector + +.. c:function:: bool glm_vec3_isvalid(vec3 v) + + | check if all items are valid number + | you should only use this in DEBUG mode or very critical asserts + + Parameters: + | *[in]* **v** vector + +.. c:function:: void glm_vec3_sign(vec3 v, vec3 dest) + + get sign of 32 bit float as +1, -1, 0 + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** sign vector (only keeps signs as -1, 0, -1) + +.. c:function:: void glm_vec3_abs(vec3 v, vec3 dest) + + absolute value of each vector item + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec3_fract(vec3 v, vec3 dest) + + fractional part of each vector item + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec3_floor(vec3 v, vec3 dest) + + floor of each vector item + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec3_sqrt(vec3 v, vec3 dest) + + square root of each vector item + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination vector (sqrt(v)) diff --git a/cglm/docs/source/vec3.rst b/cglm/docs/source/vec3.rst new file mode 100644 index 0000000..0b989b8 --- /dev/null +++ b/cglm/docs/source/vec3.rst @@ -0,0 +1,554 @@ +.. default-domain:: C + +vec3 +==== + +Header: cglm/vec3.h + + **Important:** *cglm* was used **glm_vec_** namespace for vec3 functions until + **v0.5.0**, since **v0.5.0** cglm uses **glm_vec3_** namespace for vec3. + + Also `glm_vec3_flipsign` has been renamed to `glm_vec3_negate` + +We mostly use vectors in graphics math, to make writing code faster +and easy to read, some *vec3* functions are aliased in global namespace. +For instance :c:func:`glm_dot` is alias of :c:func:`glm_vec3_dot`, +alias means inline wrapper here. There is no call version of alias functions + +There are also functions for rotating *vec3* vector. **_m4**, **_m3** prefixes +rotate *vec3* with matrix. + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. glm_vec3_dup(v, dest) +#. GLM_VEC3_ONE_INIT +#. GLM_VEC3_ZERO_INIT +#. GLM_VEC3_ONE +#. GLM_VEC3_ZERO +#. GLM_YUP +#. GLM_ZUP +#. GLM_XUP + +Functions: + +1. :c:func:`glm_vec3` +#. :c:func:`glm_vec3_copy` +#. :c:func:`glm_vec3_zero` +#. :c:func:`glm_vec3_one` +#. :c:func:`glm_vec3_dot` +#. :c:func:`glm_vec3_norm2` +#. :c:func:`glm_vec3_norm` +#. :c:func:`glm_vec3_add` +#. :c:func:`glm_vec3_adds` +#. :c:func:`glm_vec3_sub` +#. :c:func:`glm_vec3_subs` +#. :c:func:`glm_vec3_mul` +#. :c:func:`glm_vec3_scale` +#. :c:func:`glm_vec3_scale_as` +#. :c:func:`glm_vec3_div` +#. :c:func:`glm_vec3_divs` +#. :c:func:`glm_vec3_addadd` +#. :c:func:`glm_vec3_subadd` +#. :c:func:`glm_vec3_muladd` +#. :c:func:`glm_vec3_muladds` +#. :c:func:`glm_vec3_maxadd` +#. :c:func:`glm_vec3_minadd` +#. :c:func:`glm_vec3_flipsign` +#. :c:func:`glm_vec3_flipsign_to` +#. :c:func:`glm_vec3_inv` +#. :c:func:`glm_vec3_inv_to` +#. :c:func:`glm_vec3_negate` +#. :c:func:`glm_vec3_negate_to` +#. :c:func:`glm_vec3_normalize` +#. :c:func:`glm_vec3_normalize_to` +#. :c:func:`glm_vec3_cross` +#. :c:func:`glm_vec3_crossn` +#. :c:func:`glm_vec3_distance2` +#. :c:func:`glm_vec3_distance` +#. :c:func:`glm_vec3_angle` +#. :c:func:`glm_vec3_rotate` +#. :c:func:`glm_vec3_rotate_m4` +#. :c:func:`glm_vec3_rotate_m3` +#. :c:func:`glm_vec3_proj` +#. :c:func:`glm_vec3_center` +#. :c:func:`glm_vec3_maxv` +#. :c:func:`glm_vec3_minv` +#. :c:func:`glm_vec3_ortho` +#. :c:func:`glm_vec3_clamp` +#. :c:func:`glm_vec3_lerp` +#. :c:func:`glm_vec3_make` +#. :c:func:`glm_vec3_faceforward` +#. :c:func:`glm_vec3_reflect` +#. :c:func:`glm_vec3_refract` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_vec3(vec4 v4, vec3 dest) + + init vec3 using vec4 + + Parameters: + | *[in]* **v4** vector4 + | *[out]* **dest** destination + +.. c:function:: void glm_vec3_copy(vec3 a, vec3 dest) + + copy all members of [a] to [dest] + + Parameters: + | *[in]* **a** source + | *[out]* **dest** destination + +.. c:function:: void glm_vec3_zero(vec3 v) + + makes all members 0.0f (zero) + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: void glm_vec3_one(vec3 v) + + makes all members 1.0f (one) + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: float glm_vec3_dot(vec3 a, vec3 b) + + dot product of vec3 + + Parameters: + | *[in]* **a** vector1 + | *[in]* **b** vector2 + + Returns: + dot product + +.. c:function:: void glm_vec3_cross(vec3 a, vec3 b, vec3 d) + + cross product of two vector (RH) + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** destination + +.. c:function:: void glm_vec3_crossn(vec3 a, vec3 b, vec3 dest) + + cross product of two vector (RH) and normalize the result + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** destination + +.. c:function:: float glm_vec3_norm2(vec3 v) + + norm * norm (magnitude) of vector + + we can use this func instead of calling norm * norm, because it would call + sqrtf function twice but with this func we can avoid func call, maybe this is + not good name for this func + + Parameters: + | *[in]* **v** vector + + Returns: + square of norm / magnitude + +.. c:function:: float glm_vec3_norm(vec3 vec) + + | euclidean norm (magnitude), also called L2 norm + | this will give magnitude of vector in euclidean space + + Parameters: + | *[in]* **vec** vector + +.. c:function:: void glm_vec3_add(vec3 a, vec3 b, vec3 dest) + + add a vector to b vector store result in dest + + Parameters: + | *[in]* **a** vector1 + | *[in]* **b** vector2 + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec3_adds(vec3 a, float s, vec3 dest) + + add scalar to v vector store result in dest (d = v + vec(s)) + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec3_sub(vec3 v1, vec3 v2, vec3 dest) + + subtract b vector from a vector store result in dest (d = v1 - v2) + + Parameters: + | *[in]* **a** vector1 + | *[in]* **b** vector2 + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec3_subs(vec3 v, float s, vec3 dest) + + subtract scalar from v vector store result in dest (d = v - vec(s)) + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec3_mul(vec3 a, vec3 b, vec3 d) + + multiply two vector (component-wise multiplication) + + Parameters: + | *[in]* **a** vector + | *[in]* **b** scalar + | *[out]* **d** result = (a[0] * b[0], a[1] * b[1], a[2] * b[2]) + +.. c:function:: void glm_vec3_scale(vec3 v, float s, vec3 dest) + + multiply/scale vec3 vector with scalar: result = v * s + + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec3_scale_as(vec3 v, float s, vec3 dest) + + make vec3 vector scale as specified: result = unit(v) * s + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec3_div(vec3 a, vec3 b, vec3 dest) + + div vector with another component-wise division: d = a / b + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** result = (a[0] / b[0], a[1] / b[1], a[2] / b[2]) + +.. c:function:: void glm_vec3_divs(vec3 v, float s, vec3 dest) + + div vector with scalar: d = v / s + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** result = (a[0] / s, a[1] / s, a[2] / s]) + +.. c:function:: void glm_vec3_addadd(vec3 a, vec3 b, vec3 dest) + + | add two vectors and add result to sum + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** dest += (a + b) + +.. c:function:: void glm_vec3_subadd(vec3 a, vec3 b, vec3 dest) + + | sub two vectors and add result to sum + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** dest += (a - b) + +.. c:function:: void glm_vec3_muladd(vec3 a, vec3 b, vec3 dest) + + | mul two vectors and add result to sum + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** dest += (a * b) + +.. c:function:: void glm_vec3_muladds(vec3 a, float s, vec3 dest) + + | mul vector with scalar and add result to sum + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector + | *[in]* **s** scalar + | *[out]* **dest** dest += (a * b) + +.. c:function:: void glm_vec3_maxadd(vec3 a, vec3 b, vec3 dest) + + | add max of two vector to result/dest + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** dest += (a * b) + +.. c:function:: void glm_vec3_minadd(vec3 a, vec3 b, vec3 dest) + + | add min of two vector to result/dest + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** dest += (a * b) + +.. c:function:: void glm_vec3_flipsign(vec3 v) + + **DEPRECATED!** + + use :c:func:`glm_vec3_negate` + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: void glm_vec3_flipsign_to(vec3 v, vec3 dest) + + **DEPRECATED!** + + use :c:func:`glm_vec3_negate_to` + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** negated vector + +.. c:function:: void glm_vec3_inv(vec3 v) + + **DEPRECATED!** + + use :c:func:`glm_vec3_negate` + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: void glm_vec3_inv_to(vec3 v, vec3 dest) + + **DEPRECATED!** + + use :c:func:`glm_vec3_negate_to` + + Parameters: + | *[in]* **v** source + | *[out]* **dest** destination + +.. c:function:: void glm_vec3_negate(vec3 v) + + negate vector components + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: void glm_vec3_negate_to(vec3 v, vec3 dest) + + negate vector components and store result in dest + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** negated vector + +.. c:function:: void glm_vec3_normalize(vec3 v) + + normalize vec3 and store result in same vec + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: void glm_vec3_normalize_to(vec3 vec, vec3 dest) + + normalize vec3 to dest + + Parameters: + | *[in]* **vec** source + | *[out]* **dest** destination + +.. c:function:: float glm_vec3_angle(vec3 v1, vec3 v2) + + angle between two vector + + Parameters: + | *[in]* **v1** vector1 + | *[in]* **v2** vector2 + + Return: + | angle as radians + +.. c:function:: void glm_vec3_rotate(vec3 v, float angle, vec3 axis) + + rotate vec3 around axis by angle using Rodrigues' rotation formula + + Parameters: + | *[in, out]* **v** vector + | *[in]* **axis** axis vector (will be normalized) + | *[in]* **angle** angle (radians) + +.. c:function:: void glm_vec3_rotate_m4(mat4 m, vec3 v, vec3 dest) + + apply rotation matrix to vector + + Parameters: + | *[in]* **m** affine matrix or rot matrix + | *[in]* **v** vector + | *[out]* **dest** rotated vector + +.. c:function:: void glm_vec3_rotate_m3(mat3 m, vec3 v, vec3 dest) + + apply rotation matrix to vector + + Parameters: + | *[in]* **m** affine matrix or rot matrix + | *[in]* **v** vector + | *[out]* **dest** rotated vector + +.. c:function:: void glm_vec3_proj(vec3 a, vec3 b, vec3 dest) + + project a vector onto b vector + + Parameters: + | *[in]* **a** vector1 + | *[in]* **b** vector2 + | *[out]* **dest** projected vector + +.. c:function:: void glm_vec3_center(vec3 v1, vec3 v2, vec3 dest) + + find center point of two vector + + Parameters: + | *[in]* **v1** vector1 + | *[in]* **v2** vector2 + | *[out]* **dest** center point + +.. c:function:: float glm_vec3_distance2(vec3 v1, vec3 v2) + + squared distance between two vectors + + Parameters: + | *[in]* **v1** vector1 + | *[in]* **v2** vector2 + + Returns: + | squared distance (distance * distance) + +.. c:function:: float glm_vec3_distance(vec3 v1, vec3 v2) + + distance between two vectors + + Parameters: + | *[in]* **v1** vector1 + | *[in]* **v2** vector2 + + Returns: + | distance + +.. c:function:: void glm_vec3_maxv(vec3 v1, vec3 v2, vec3 dest) + + max values of vectors + + Parameters: + | *[in]* **v1** vector1 + | *[in]* **v2** vector2 + | *[out]* **dest** destination + +.. c:function:: void glm_vec3_minv(vec3 v1, vec3 v2, vec3 dest) + + min values of vectors + + Parameters: + | *[in]* **v1** vector1 + | *[in]* **v2** vector2 + | *[out]* **dest** destination + +.. c:function:: void glm_vec3_ortho(vec3 v, vec3 dest) + + possible orthogonal/perpendicular vector + + References: + * `On picking an orthogonal vector (and combing coconuts) `_ + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** orthogonal/perpendicular vector + +.. c:function:: void glm_vec3_clamp(vec3 v, float minVal, float maxVal) + + constrain a value to lie between two further values + + Parameters: + | *[in, out]* **v** vector + | *[in]* **minVal** minimum value + | *[in]* **maxVal** maximum value + +.. c:function:: void glm_vec3_lerp(vec3 from, vec3 to, float t, vec3 dest) + + linear interpolation between two vector + + | formula: from + s * (to - from) + + Parameters: + | *[in]* **from** from value + | *[in]* **to** to value + | *[in]* **t** interpolant (amount) clamped between 0 and 1 + | *[out]* **dest** destination + +.. c:function:: void glm_vec3_make(const float * __restrict src, vec3 dest) + + Create three dimensional vector from pointer + + .. note::: **@src** must contain at least 3 elements. + + Parameters: + | *[in]* **src** pointer to an array of floats + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec3_faceforward(vec3 n, vec3 v, vec3 nref, vec3 dest) + + A vector pointing in the same direction as another + + Parameters: + | *[in]* **n** vector to orient + | *[in]* **v** incident vector + | *[in]* **nref** reference vector + | *[out]* **dest** destination: oriented vector, pointing away from the surface. + +.. c:function:: void glm_vec3_reflect(vec3 v, vec3 n, vec3 dest) + + Reflection vector using an incident ray and a surface normal + + Parameters: + | *[in]* **v** incident vector + | *[in]* **n** *❗️ normalized ❗️* normal vector + | *[out]* **dest** destination: reflection result + +.. c:function:: bool glm_vec3_refract(vec3 v, vec3 n, float eta, vec3 dest) + + + Computes refraction vector for an incident vector and a surface normal. + + Calculates the refraction vector based on Snell's law. If total internal reflection + occurs (angle too great given eta), dest is set to zero and returns false. + Otherwise, computes refraction vector, stores it in dest, and returns true. + + Parameters: + | *[in]* **v** *❗️ normalized ❗️* incident vector + | *[in]* **n** *❗️ normalized ❗️* normal vector + | *[in]* **eta** ratio of indices of refraction (incident/transmitted) + | *[out]* **dest** refraction vector if refraction occurs; zero vector otherwise + + Returns: + returns true if refraction occurs; false if total internal reflection occurs. diff --git a/cglm/docs/source/vec4-ext.rst b/cglm/docs/source/vec4-ext.rst new file mode 100644 index 0000000..bf7eca2 --- /dev/null +++ b/cglm/docs/source/vec4-ext.rst @@ -0,0 +1,171 @@ +.. default-domain:: C + +vec4 extra +========== + +Header: cglm/vec4-ext.h + +There are some functions are in called in extra header. These are called extra +because they are not used like other functions in vec4.h in the future some of +these functions ma be moved to vec4 header. + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Functions: + +1. :c:func:`glm_vec4_mulv` +#. :c:func:`glm_vec4_broadcast` +#. :c:func:`glm_vec4_eq` +#. :c:func:`glm_vec4_eq_eps` +#. :c:func:`glm_vec4_eq_all` +#. :c:func:`glm_vec4_eqv` +#. :c:func:`glm_vec4_eqv_eps` +#. :c:func:`glm_vec4_max` +#. :c:func:`glm_vec4_min` +#. :c:func:`glm_vec4_isnan` +#. :c:func:`glm_vec4_isinf` +#. :c:func:`glm_vec4_isvalid` +#. :c:func:`glm_vec4_sign` +#. :c:func:`glm_vec4_abs` +#. :c:func:`glm_vec4_fract` +#. :c:func:`glm_vec4_floor` +#. :c:func:`glm_vec4_sqrt` + + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_vec4_mulv(vec4 a, vec4 b, vec4 d) + + multiplies individual items + + Parameters: + | *[in]* **a** vec1 + | *[in]* **b** vec2 + | *[out]* **d** destination + +.. c:function:: void glm_vec4_broadcast(float val, vec4 d) + + fill a vector with specified value + + Parameters: + | *[in]* **val** value + | *[out]* **dest** destination + +.. c:function:: bool glm_vec4_eq(vec4 v, float val) + + check if vector is equal to value (without epsilon) + + Parameters: + | *[in]* **v** vector + | *[in]* **val** value + +.. c:function:: bool glm_vec4_eq_eps(vec4 v, float val) + + check if vector is equal to value (with epsilon) + + Parameters: + | *[in]* **v** vector + | *[in]* **val** value + +.. c:function:: bool glm_vec4_eq_all(vec4 v) + + check if vectors members are equal (without epsilon) + + Parameters: + | *[in]* **v** vector + +.. c:function:: bool glm_vec4_eqv(vec4 v1, vec4 v2) + + check if vector is equal to another (without epsilon) vector + + Parameters: + | *[in]* **vec** vector 1 + | *[in]* **vec** vector 2 + +.. c:function:: bool glm_vec4_eqv_eps(vec4 v1, vec4 v2) + + check if vector is equal to another (with epsilon) + + Parameters: + | *[in]* **v1** vector1 + | *[in]* **v2** vector2 + +.. c:function:: float glm_vec4_max(vec4 v) + + max value of vector + + Parameters: + | *[in]* **v** vector + +.. c:function:: float glm_vec4_min(vec4 v) + + min value of vector + + Parameters: + | *[in]* **v** vector + +.. c:function:: bool glm_vec4_isnan(vec4 v) + + | check if one of items is NaN (not a number) + | you should only use this in DEBUG mode or very critical asserts + + Parameters: + | *[in]* **v** vector + +.. c:function:: bool glm_vec4_isinf(vec4 v) + + | check if one of items is INFINITY + | you should only use this in DEBUG mode or very critical asserts + + Parameters: + | *[in]* **v** vector + +.. c:function:: bool glm_vec4_isvalid(vec4 v) + + | check if all items are valid number + | you should only use this in DEBUG mode or very critical asserts + + Parameters: + | *[in]* **v** vector + +.. c:function:: void glm_vec4_sign(vec4 v, vec4 dest) + + get sign of 32 bit float as +1, -1, 0 + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** sign vector (only keeps signs as -1, 0, -1) + +.. c:function:: void glm_vec4_abs(vec4 v, vec4 dest) + + absolute value of each vector item + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination vector (abs(v)) + +.. c:function:: void glm_vec4_fract(vec4 v, vec4 dest) + + fractional part of each vector item + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination vector (fract(v)) + +.. c:function:: void glm_vec4_floor(vec4 v, vec4 dest) + + floor of each vector item + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination vector (floor(v)) + +.. c:function:: void glm_vec4_sqrt(vec4 v, vec4 dest) + + square root of each vector item + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** destination vector (sqrt(v)) diff --git a/cglm/docs/source/vec4.rst b/cglm/docs/source/vec4.rst new file mode 100644 index 0000000..9035a3d --- /dev/null +++ b/cglm/docs/source/vec4.rst @@ -0,0 +1,458 @@ +.. default-domain:: C + +vec4 +==== + +Header: cglm/vec4.h + +Table of contents (click to go): +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Macros: + +1. glm_vec4_dup3(v, dest) +#. glm_vec4_dup(v, dest) +#. GLM_VEC4_ONE_INIT +#. GLM_VEC4_BLACK_INIT +#. GLM_VEC4_ZERO_INIT +#. GLM_VEC4_ONE +#. GLM_VEC4_BLACK +#. GLM_VEC4_ZERO + +Functions: + +1. :c:func:`glm_vec4` +#. :c:func:`glm_vec4_copy3` +#. :c:func:`glm_vec4_copy` +#. :c:func:`glm_vec4_ucopy` +#. :c:func:`glm_vec4_zero` +#. :c:func:`glm_vec4_one` +#. :c:func:`glm_vec4_dot` +#. :c:func:`glm_vec4_norm2` +#. :c:func:`glm_vec4_norm` +#. :c:func:`glm_vec4_add` +#. :c:func:`glm_vec4_adds` +#. :c:func:`glm_vec4_sub` +#. :c:func:`glm_vec4_subs` +#. :c:func:`glm_vec4_mul` +#. :c:func:`glm_vec4_scale` +#. :c:func:`glm_vec4_scale_as` +#. :c:func:`glm_vec4_div` +#. :c:func:`glm_vec4_divs` +#. :c:func:`glm_vec4_addadd` +#. :c:func:`glm_vec4_subadd` +#. :c:func:`glm_vec4_muladd` +#. :c:func:`glm_vec4_muladds` +#. :c:func:`glm_vec4_maxadd` +#. :c:func:`glm_vec4_minadd` +#. :c:func:`glm_vec4_flipsign` +#. :c:func:`glm_vec4_flipsign_to` +#. :c:func:`glm_vec4_inv` +#. :c:func:`glm_vec4_inv_to` +#. :c:func:`glm_vec4_negate` +#. :c:func:`glm_vec4_negate_to` +#. :c:func:`glm_vec4_normalize` +#. :c:func:`glm_vec4_normalize_to` +#. :c:func:`glm_vec4_distance` +#. :c:func:`glm_vec4_maxv` +#. :c:func:`glm_vec4_minv` +#. :c:func:`glm_vec4_clamp` +#. :c:func:`glm_vec4_lerp` +#. :c:func:`glm_vec4_cubic` +#. :c:func:`glm_vec4_make` +#. :c:func:`glm_vec4_reflect` +#. :c:func:`glm_vec4_refract` + +Functions documentation +~~~~~~~~~~~~~~~~~~~~~~~ + +.. c:function:: void glm_vec4(vec3 v3, float last, vec4 dest) + + init vec4 using vec3, since you are initializing vec4 with vec3 + you need to set last item. cglm could set it zero but making it parameter + gives more control + + Parameters: + | *[in]* **v3** vector4 + | *[in]* **last** last item of vec4 + | *[out]* **dest** destination + +.. c:function:: void glm_vec4_copy3(vec4 a, vec3 dest) + + copy first 3 members of [a] to [dest] + + Parameters: + | *[in]* **a** source + | *[out]* **dest** destination + +.. c:function:: void glm_vec4_copy(vec4 v, vec4 dest) + + copy all members of [a] to [dest] + + Parameters: + | *[in]* **v** source + | *[in]* **dest** destination + +.. c:function:: void glm_vec4_ucopy(vec4 v, vec4 dest) + + copy all members of [a] to [dest] + + | alignment is not required + + Parameters: + | *[in]* **v** source + | *[in]* **dest** destination + +.. c:function:: void glm_vec4_zero(vec4 v) + + makes all members zero + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: void glm_vec4_one(vec4 v) + + makes all members one + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: float glm_vec4_dot(vec4 a, vec4 b) + + dot product of vec4 + + Parameters: + | *[in]* **a** vector1 + | *[in]* **b** vector2 + + Returns: + dot product + +.. c:function:: float glm_vec4_norm2(vec4 v) + + norm * norm (magnitude) of vector + + we can use this func instead of calling norm * norm, because it would call + sqrtf function twice but with this func we can avoid func call, maybe this is + not good name for this func + + Parameters: + | *[in]* **v** vector + + Returns: + square of norm / magnitude + +.. c:function:: float glm_vec4_norm(vec4 vec) + + | euclidean norm (magnitude), also called L2 norm + | this will give magnitude of vector in euclidean space + + Parameters: + | *[in]* **vec** vector + +.. c:function:: void glm_vec4_add(vec4 a, vec4 b, vec4 dest) + + add a vector to b vector store result in dest + + Parameters: + | *[in]* **a** vector1 + | *[in]* **b** vector2 + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec4_adds(vec4 v, float s, vec4 dest) + + add scalar to v vector store result in dest (d = v + vec(s)) + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec4_sub(vec4 a, vec4 b, vec4 dest) + + subtract b vector from a vector store result in dest (d = v1 - v2) + + Parameters: + | *[in]* **a** vector1 + | *[in]* **b** vector2 + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec4_subs(vec4 v, float s, vec4 dest) + + subtract scalar from v vector store result in dest (d = v - vec(s)) + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec4_mul(vec4 a, vec4 b, vec4 d) + + multiply two vector (component-wise multiplication) + + Parameters: + | *[in]* **a** vector1 + | *[in]* **b** vector2 + | *[out]* **dest** result = (a[0] * b[0], a[1] * b[1], a[2] * b[2], a[3] * b[3]) + +.. c:function:: void glm_vec4_scale(vec4 v, float s, vec4 dest) + + multiply/scale vec4 vector with scalar: result = v * s + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec4_scale_as(vec4 v, float s, vec4 dest) + + make vec4 vector scale as specified: result = unit(v) * s + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** destination vector + +.. c:function:: void glm_vec4_div(vec4 a, vec4 b, vec4 dest) + + div vector with another component-wise division: d = v1 / v2 + + Parameters: + | *[in]* **a** vector1 + | *[in]* **b** vector2 + | *[out]* **dest** result = (a[0] / b[0], a[1] / b[1], a[2] / b[2], a[3] / b[3]) + +.. c:function:: void glm_vec4_divs(vec4 v, float s, vec4 dest) + + div vector with scalar: d = v / s + + Parameters: + | *[in]* **v** vector + | *[in]* **s** scalar + | *[out]* **dest** result = (a[0] / s, a[1] / s, a[2] / s, a[3] / s) + +.. c:function:: void glm_vec4_addadd(vec4 a, vec4 b, vec4 dest) + + | add two vectors and add result to sum + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** dest += (a + b) + +.. c:function:: void glm_vec4_subadd(vec4 a, vec4 b, vec4 dest) + + | sub two vectors and add result to sum + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** dest += (a - b) + +.. c:function:: void glm_vec4_muladd(vec4 a, vec4 b, vec4 dest) + + | mul two vectors and add result to sum + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** dest += (a * b) + +.. c:function:: void glm_vec4_muladds(vec4 a, float s, vec4 dest) + + | mul vector with scalar and add result to sum + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector + | *[in]* **s** scalar + | *[out]* **dest** dest += (a * b) + +.. c:function:: void glm_vec4_maxadd(vec4 a, vec4 b, vec4 dest) + + | add max of two vector to result/dest + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** dest += (a * b) + +.. c:function:: void glm_vec4_minadd(vec4 a, vec4 b, vec4 dest) + + | add min of two vector to result/dest + | it applies += operator so dest must be initialized + + Parameters: + | *[in]* **a** vector 1 + | *[in]* **b** vector 2 + | *[out]* **dest** dest += (a * b) + +.. c:function:: void glm_vec4_flipsign(vec4 v) + + **DEPRECATED!** + + use :c:func:`glm_vec4_negate` + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: void glm_vec4_flipsign_to(vec4 v, vec4 dest) + + **DEPRECATED!** + + use :c:func:`glm_vec4_negate_to` + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** negated vector + +.. c:function:: void glm_vec4_inv(vec4 v) + + **DEPRECATED!** + + use :c:func:`glm_vec4_negate` + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: void glm_vec4_inv_to(vec4 v, vec4 dest) + + **DEPRECATED!** + + use :c:func:`glm_vec4_negate_to` + + Parameters: + | *[in]* **v** source + | *[out]* **dest** destination + +.. c:function:: void glm_vec4_negate(vec4 v) + + negate vector components + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: void glm_vec4_negate_to(vec4 v, vec4 dest) + + negate vector components and store result in dest + + Parameters: + | *[in]* **v** vector + | *[out]* **dest** negated vector + +.. c:function:: void glm_vec4_normalize(vec4 v) + + normalize vec4 and store result in same vec + + Parameters: + | *[in, out]* **v** vector + +.. c:function:: void glm_vec4_normalize_to(vec4 vec, vec4 dest) + + normalize vec4 to dest + + Parameters: + | *[in]* **vec** source + | *[out]* **dest** destination + +.. c:function:: float glm_vec4_distance(vec4 v1, vec4 v2) + + distance between two vectors + + Parameters: + | *[in]* **mat** vector1 + | *[in]* **row1** vector2 + + Returns: + | distance + +.. c:function:: void glm_vec4_maxv(vec4 v1, vec4 v2, vec4 dest) + + max values of vectors + + Parameters: + | *[in]* **v1** vector1 + | *[in]* **v2** vector2 + | *[out]* **dest** destination + +.. c:function:: void glm_vec4_minv(vec4 v1, vec4 v2, vec4 dest) + + min values of vectors + + Parameters: + | *[in]* **v1** vector1 + | *[in]* **v2** vector2 + | *[out]* **dest** destination + +.. c:function:: void glm_vec4_clamp(vec4 v, float minVal, float maxVal) + + constrain a value to lie between two further values + + Parameters: + | *[in, out]* **v** vector + | *[in]* **minVal** minimum value + | *[in]* **maxVal** maximum value + +.. c:function:: void glm_vec4_lerp(vec4 from, vec4 to, float t, vec4 dest) + + linear interpolation between two vector + + | formula: from + s * (to - from) + + Parameters: + | *[in]* **from** from value + | *[in]* **to** to value + | *[in]* **t** interpolant (amount) clamped between 0 and 1 + | *[out]* **dest** destination + +.. c:function:: void glm_vec4_cubic(float s, vec4 dest) + + helper to fill vec4 as [S^3, S^2, S, 1] + + Parameters: + | *[in]* **s** parameter + | *[out]* **dest** destination + +.. c:function:: void glm_vec4_make(const float * __restrict src, vec4 dest) + + Create four dimensional vector from pointer + + .. note:: **@src** must contain at least 4 elements. + + Parameters: + | *[in]* **src** pointer to an array of floats + | *[out]* **dest** destination vector + +.. c:function:: bool glm_vec4_reflect(vec4 v, vec4 n, vec4 dest) + + Reflection vector using an incident ray and a surface normal + + Parameters: + | *[in]* **v** incident vector + | *[in]* **n** *❗️ normalized ❗️* normal vector + | *[out]* **dest** destination: reflection result + +.. c:function:: bool glm_vec4_refract(vec4 v, vec4 n, float eta, vec4 dest) + + computes refraction vector for an incident vector and a surface normal. + + Calculates the refraction vector based on Snell's law. If total internal reflection + occurs (angle too great given eta), dest is set to zero and returns false. + Otherwise, computes refraction vector, stores it in dest, and returns true. + + This implementation does not explicitly preserve the 'w' component of the + incident vector 'I' in the output 'dest', users requiring the preservation of + the 'w' component should manually adjust 'dest' after calling this function. + + Parameters: + | *[in]* **v** *❗️ normalized ❗️* incident vector + | *[in]* **n** *❗️ normalized ❗️* normal vector + | *[in]* **eta** ratio of indices of refraction (incident/transmitted) + | *[out]* **dest** refraction vector if refraction occurs; zero vector otherwise + + Returns: + returns true if refraction occurs; false if total internal reflection occurs. diff --git a/cglm/docs/source/version.rst b/cglm/docs/source/version.rst new file mode 100644 index 0000000..7e662f9 --- /dev/null +++ b/cglm/docs/source/version.rst @@ -0,0 +1,15 @@ +.. default-domain:: C + +version +================================================================================ + +Header: cglm/version.h + +**cglm** uses semantic versioning (http://semver.org) which is MAJOR.MINOR.PATCH + +| **CGLM_VERSION_MAJOR** is major number of the version. +| **CGLM_VERSION_MINOR** is minor number of the version. +| **CGLM_VERSION_PATCH** is patch number of the version. + +every release increases these numbers. You can check existing version by +including `cglm/version.h` diff --git a/cglm/include/cglm/aabb2d.h b/cglm/include/cglm/aabb2d.h new file mode 100644 index 0000000..6369d08 --- /dev/null +++ b/cglm/include/cglm/aabb2d.h @@ -0,0 +1,270 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_aabb2d_h +#define cglm_aabb2d_h + +#include "common.h" +#include "vec2.h" +#include "util.h" + +/* DEPRECATED! use _diag */ +#define glm_aabb2d_size(aabb) glm_aabb2d_diag(aabb) + +/*! + * @brief make [aabb] zero + * + * @param[in, out] aabb aabb + */ +CGLM_INLINE +void +glm_aabb2d_zero(vec2 aabb[2]) { + glm_vec2_zero(aabb[0]); + glm_vec2_zero(aabb[1]); +} + +/*! + * @brief copy all members of [aabb] to [dest] + * + * @param[in] aabb source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_aabb2d_copy(vec2 aabb[2], vec2 dest[2]) { + glm_vec2_copy(aabb[0], dest[0]); + glm_vec2_copy(aabb[1], dest[1]); +} + +/*! + * @brief apply transform to Axis-Aligned Bounding aabb + * + * @param[in] aabb bounding aabb + * @param[in] m transform matrix + * @param[out] dest transformed bounding aabb + */ +CGLM_INLINE +void +glm_aabb2d_transform(vec2 aabb[2], mat3 m, vec2 dest[2]) { + vec2 v[2], xa, xb, ya, yb; + + glm_vec2_scale(m[0], aabb[0][0], xa); + glm_vec2_scale(m[0], aabb[1][0], xb); + + glm_vec2_scale(m[1], aabb[0][1], ya); + glm_vec2_scale(m[1], aabb[1][1], yb); + + /* translation + min(xa, xb) + min(ya, yb) */ + glm_vec2(m[2], v[0]); + glm_vec2_minadd(xa, xb, v[0]); + glm_vec2_minadd(ya, yb, v[0]); + + /* translation + max(xa, xb) + max(ya, yb) */ + glm_vec2(m[2], v[1]); + glm_vec2_maxadd(xa, xb, v[1]); + glm_vec2_maxadd(ya, yb, v[1]); + + glm_vec2_copy(v[0], dest[0]); + glm_vec2_copy(v[1], dest[1]); +} + +/*! + * @brief merges two AABB bounding aabb and creates new one + * + * two aabb must be in same space, if one of aabb is in different space then + * you should consider to convert it's space by glm_aabb_space + * + * @param[in] aabb1 bounding aabb 1 + * @param[in] aabb2 bounding aabb 2 + * @param[out] dest merged bounding aabb + */ +CGLM_INLINE +void +glm_aabb2d_merge(vec2 aabb1[2], vec2 aabb2[2], vec2 dest[2]) { + dest[0][0] = glm_min(aabb1[0][0], aabb2[0][0]); + dest[0][1] = glm_min(aabb1[0][1], aabb2[0][1]); + + dest[1][0] = glm_max(aabb1[1][0], aabb2[1][0]); + dest[1][1] = glm_max(aabb1[1][1], aabb2[1][1]); +} + +/*! + * @brief crops a bounding aabb with another one. + * + * this could be useful for getting a baabb which fits with view frustum and + * object bounding aabbes. In this case you crop view frustum aabb with objects + * aabb + * + * @param[in] aabb bounding aabb 1 + * @param[in] cropAabb crop aabb + * @param[out] dest cropped bounding aabb + */ +CGLM_INLINE +void +glm_aabb2d_crop(vec2 aabb[2], vec2 cropAabb[2], vec2 dest[2]) { + dest[0][0] = glm_max(aabb[0][0], cropAabb[0][0]); + dest[0][1] = glm_max(aabb[0][1], cropAabb[0][1]); + + dest[1][0] = glm_min(aabb[1][0], cropAabb[1][0]); + dest[1][1] = glm_min(aabb[1][1], cropAabb[1][1]); +} + +/*! + * @brief crops a bounding aabb with another one. + * + * this could be useful for getting a baabb which fits with view frustum and + * object bounding aabbes. In this case you crop view frustum aabb with objects + * aabb + * + * @param[in] aabb bounding aabb + * @param[in] cropAabb crop aabb + * @param[in] clampAabb minimum aabb + * @param[out] dest cropped bounding aabb + */ +CGLM_INLINE +void +glm_aabb2d_crop_until(vec2 aabb[2], + vec2 cropAabb[2], + vec2 clampAabb[2], + vec2 dest[2]) { + glm_aabb2d_crop(aabb, cropAabb, dest); + glm_aabb2d_merge(clampAabb, dest, dest); +} + +/*! + * @brief invalidate AABB min and max values + * + * @param[in, out] aabb bounding aabb + */ +CGLM_INLINE +void +glm_aabb2d_invalidate(vec2 aabb[2]) { + glm_vec2_fill(aabb[0], FLT_MAX); + glm_vec2_fill(aabb[1], -FLT_MAX); +} + +/*! + * @brief check if AABB is valid or not + * + * @param[in] aabb bounding aabb + */ +CGLM_INLINE +bool +glm_aabb2d_isvalid(vec2 aabb[2]) { + return glm_vec2_max(aabb[0]) != FLT_MAX + && glm_vec2_min(aabb[1]) != -FLT_MAX; +} + +/*! + * @brief distance between of min and max + * + * @param[in] aabb bounding aabb + */ +CGLM_INLINE +float +glm_aabb2d_diag(vec2 aabb[2]) { + return glm_vec2_distance(aabb[0], aabb[1]); +} + +/*! + * @brief size of aabb + * + * @param[in] aabb bounding aabb + * @param[out] dest size + */ +CGLM_INLINE +void +glm_aabb2d_sizev(vec2 aabb[2], vec2 dest) { + glm_vec2_sub(aabb[1], aabb[0], dest); +} + +/*! + * @brief radius of sphere which surrounds AABB + * + * @param[in] aabb bounding aabb + */ +CGLM_INLINE +float +glm_aabb2d_radius(vec2 aabb[2]) { + return glm_aabb2d_diag(aabb) * 0.5f; +} + +/*! + * @brief computes center point of AABB + * + * @param[in] aabb bounding aabb + * @param[out] dest center of bounding aabb + */ +CGLM_INLINE +void +glm_aabb2d_center(vec2 aabb[2], vec2 dest) { + glm_vec2_center(aabb[0], aabb[1], dest); +} + +/*! + * @brief check if two AABB intersects + * + * @param[in] aabb bounding aabb + * @param[in] other other bounding aabb + */ +CGLM_INLINE +bool +glm_aabb2d_aabb(vec2 aabb[2], vec2 other[2]) { + return (aabb[0][0] <= other[1][0] && aabb[1][0] >= other[0][0]) + && (aabb[0][1] <= other[1][1] && aabb[1][1] >= other[0][1]); +} + +/*! + * @brief check if AABB intersects with a circle + * + * Circle Representation in cglm: [center.x, center.y, radii] + * + * @param[in] aabb solid bounding aabb + * @param[in] c solid circle + */ +CGLM_INLINE +bool +glm_aabb2d_circle(vec2 aabb[2], vec3 c) { + float dmin; + int a, b; + + a = (c[0] < aabb[0][0]) + (c[0] > aabb[1][0]); + b = (c[1] < aabb[0][1]) + (c[1] > aabb[1][1]); + + dmin = glm_pow2((c[0] - aabb[!(a - 1)][0]) * (a != 0)) + + glm_pow2((c[1] - aabb[!(b - 1)][1]) * (b != 0)); + + return dmin <= glm_pow2(c[2]); +} + +/*! + * @brief check if point is inside of AABB + * + * @param[in] aabb bounding aabb + * @param[in] point point + */ +CGLM_INLINE +bool +glm_aabb2d_point(vec2 aabb[2], vec2 point) { + return (point[0] >= aabb[0][0] && point[0] <= aabb[1][0]) + && (point[1] >= aabb[0][1] && point[1] <= aabb[1][1]); +} + +/*! + * @brief check if AABB contains other AABB + * + * @param[in] aabb bounding aabb + * @param[in] other other bounding aabb + */ +CGLM_INLINE +bool +glm_aabb2d_contains(vec2 aabb[2], vec2 other[2]) { + return (aabb[0][0] <= other[0][0] && aabb[1][0] >= other[1][0]) + && (aabb[0][1] <= other[0][1] && aabb[1][1] >= other[1][1]); +} + +#endif /* cglm_aabb2d_h */ diff --git a/cglm/include/cglm/affine-mat.h b/cglm/include/cglm/affine-mat.h new file mode 100644 index 0000000..c22c0e0 --- /dev/null +++ b/cglm/include/cglm/affine-mat.h @@ -0,0 +1,189 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_mul(mat4 m1, mat4 m2, mat4 dest); + CGLM_INLINE void glm_mul_rot(mat4 m1, mat4 m2, mat4 dest); + CGLM_INLINE void glm_inv_tr(mat4 mat); + */ + +#ifndef cglm_affine_mat_h +#define cglm_affine_mat_h + +#include "common.h" +#include "mat4.h" +#include "mat3.h" + +#ifdef CGLM_SSE_FP +# include "simd/sse2/affine.h" +#endif + +#ifdef CGLM_AVX_FP +# include "simd/avx/affine.h" +#endif + +#ifdef CGLM_NEON_FP +# include "simd/neon/affine.h" +#endif + +#ifdef CGLM_SIMD_WASM +# include "simd/wasm/affine.h" +#endif + +/*! + * @brief this is similar to glm_mat4_mul but specialized to affine transform + * + * Matrix format should be: + * R R R X + * R R R Y + * R R R Z + * 0 0 0 W + * + * this reduces some multiplications. It should be faster than mat4_mul. + * if you are not sure about matrix format then DON'T use this! use mat4_mul + * + * @param[in] m1 affine matrix 1 + * @param[in] m2 affine matrix 2 + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_mul(mat4 m1, mat4 m2, mat4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glm_mul_wasm(m1, m2, dest); +#elif defined(__AVX__) + glm_mul_avx(m1, m2, dest); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glm_mul_sse2(m1, m2, dest); +#elif defined(CGLM_NEON_FP) + glm_mul_neon(m1, m2, dest); +#else + float a00 = m1[0][0], a01 = m1[0][1], a02 = m1[0][2], a03 = m1[0][3], + a10 = m1[1][0], a11 = m1[1][1], a12 = m1[1][2], a13 = m1[1][3], + a20 = m1[2][0], a21 = m1[2][1], a22 = m1[2][2], a23 = m1[2][3], + a30 = m1[3][0], a31 = m1[3][1], a32 = m1[3][2], a33 = m1[3][3], + + b00 = m2[0][0], b01 = m2[0][1], b02 = m2[0][2], + b10 = m2[1][0], b11 = m2[1][1], b12 = m2[1][2], + b20 = m2[2][0], b21 = m2[2][1], b22 = m2[2][2], + b30 = m2[3][0], b31 = m2[3][1], b32 = m2[3][2], b33 = m2[3][3]; + + dest[0][0] = a00 * b00 + a10 * b01 + a20 * b02; + dest[0][1] = a01 * b00 + a11 * b01 + a21 * b02; + dest[0][2] = a02 * b00 + a12 * b01 + a22 * b02; + dest[0][3] = a03 * b00 + a13 * b01 + a23 * b02; + + dest[1][0] = a00 * b10 + a10 * b11 + a20 * b12; + dest[1][1] = a01 * b10 + a11 * b11 + a21 * b12; + dest[1][2] = a02 * b10 + a12 * b11 + a22 * b12; + dest[1][3] = a03 * b10 + a13 * b11 + a23 * b12; + + dest[2][0] = a00 * b20 + a10 * b21 + a20 * b22; + dest[2][1] = a01 * b20 + a11 * b21 + a21 * b22; + dest[2][2] = a02 * b20 + a12 * b21 + a22 * b22; + dest[2][3] = a03 * b20 + a13 * b21 + a23 * b22; + + dest[3][0] = a00 * b30 + a10 * b31 + a20 * b32 + a30 * b33; + dest[3][1] = a01 * b30 + a11 * b31 + a21 * b32 + a31 * b33; + dest[3][2] = a02 * b30 + a12 * b31 + a22 * b32 + a32 * b33; + dest[3][3] = a03 * b30 + a13 * b31 + a23 * b32 + a33 * b33; +#endif +} + +/*! + * @brief this is similar to glm_mat4_mul but specialized to affine transform + * + * Right Matrix format should be: + * R R R 0 + * R R R 0 + * R R R 0 + * 0 0 0 1 + * + * this reduces some multiplications. It should be faster than mat4_mul. + * if you are not sure about matrix format then DON'T use this! use mat4_mul + * + * @param[in] m1 affine matrix 1 + * @param[in] m2 affine matrix 2 + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_mul_rot(mat4 m1, mat4 m2, mat4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glm_mul_rot_wasm(m1, m2, dest); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glm_mul_rot_sse2(m1, m2, dest); +#elif defined(CGLM_NEON_FP) + glm_mul_rot_neon(m1, m2, dest); +#else + float a00 = m1[0][0], a01 = m1[0][1], a02 = m1[0][2], a03 = m1[0][3], + a10 = m1[1][0], a11 = m1[1][1], a12 = m1[1][2], a13 = m1[1][3], + a20 = m1[2][0], a21 = m1[2][1], a22 = m1[2][2], a23 = m1[2][3], + a30 = m1[3][0], a31 = m1[3][1], a32 = m1[3][2], a33 = m1[3][3], + + b00 = m2[0][0], b01 = m2[0][1], b02 = m2[0][2], + b10 = m2[1][0], b11 = m2[1][1], b12 = m2[1][2], + b20 = m2[2][0], b21 = m2[2][1], b22 = m2[2][2]; + + dest[0][0] = a00 * b00 + a10 * b01 + a20 * b02; + dest[0][1] = a01 * b00 + a11 * b01 + a21 * b02; + dest[0][2] = a02 * b00 + a12 * b01 + a22 * b02; + dest[0][3] = a03 * b00 + a13 * b01 + a23 * b02; + + dest[1][0] = a00 * b10 + a10 * b11 + a20 * b12; + dest[1][1] = a01 * b10 + a11 * b11 + a21 * b12; + dest[1][2] = a02 * b10 + a12 * b11 + a22 * b12; + dest[1][3] = a03 * b10 + a13 * b11 + a23 * b12; + + dest[2][0] = a00 * b20 + a10 * b21 + a20 * b22; + dest[2][1] = a01 * b20 + a11 * b21 + a21 * b22; + dest[2][2] = a02 * b20 + a12 * b21 + a22 * b22; + dest[2][3] = a03 * b20 + a13 * b21 + a23 * b22; + + dest[3][0] = a30; + dest[3][1] = a31; + dest[3][2] = a32; + dest[3][3] = a33; +#endif +} + +/*! + * @brief inverse orthonormal rotation + translation matrix (ridig-body) + * + * @code + * X = | R T | X' = | R' -R'T | + * | 0 1 | | 0 1 | + * @endcode + * + * @param[in,out] mat matrix + */ +CGLM_INLINE +void +glm_inv_tr(mat4 mat) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glm_inv_tr_wasm(mat); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glm_inv_tr_sse2(mat); +#elif defined(CGLM_NEON_FP) + glm_inv_tr_neon(mat); +#else + CGLM_ALIGN_MAT mat3 r; + CGLM_ALIGN(8) vec3 t; + + /* rotate */ + glm_mat4_pick3t(mat, r); + glm_mat4_ins3(r, mat); + + /* translate */ + glm_mat3_mulv(r, mat[3], t); + glm_vec3_negate(t); + glm_vec3_copy(t, mat[3]); +#endif +} + +#endif /* cglm_affine_mat_h */ diff --git a/cglm/include/cglm/affine-post.h b/cglm/include/cglm/affine-post.h new file mode 100644 index 0000000..3e297e6 --- /dev/null +++ b/cglm/include/cglm/affine-post.h @@ -0,0 +1,247 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_affine_post_h +#define cglm_affine_post_h + +/* + Functions: + CGLM_INLINE void glm_translated_to(mat4 m, vec3 v, mat4 dest); + CGLM_INLINE void glm_translated(mat4 m, vec3 v); + CGLM_INLINE void glm_translated_x(mat4 m, float to); + CGLM_INLINE void glm_translated_y(mat4 m, float to); + CGLM_INLINE void glm_translated_z(mat4 m, float to); + CGLM_INLINE void glm_rotated_x(mat4 m, float angle, mat4 dest); + CGLM_INLINE void glm_rotated_y(mat4 m, float angle, mat4 dest); + CGLM_INLINE void glm_rotated_z(mat4 m, float angle, mat4 dest); + CGLM_INLINE void glm_rotated(mat4 m, float angle, vec3 axis); + CGLM_INLINE void glm_rotated_at(mat4 m, vec3 pivot, float angle, vec3 axis); + CGLM_INLINE void glm_spinned(mat4 m, float angle, vec3 axis); + */ + +#include "common.h" +#include "util.h" +#include "vec3.h" +#include "vec4.h" +#include "mat4.h" +#include "affine-mat.h" + +/*! + * @brief translate existing transform matrix by v vector + * and stores result in same matrix + * + * this is POST transform, applies to existing transform as last transform + * + * @param[in, out] m affine transform + * @param[in] v translate vector [x, y, z] + */ +CGLM_INLINE +void +glm_translated(mat4 m, vec3 v) { + glm_vec3_add(m[3], v, m[3]); +} + +/*! + * @brief translate existing transform matrix by v vector + * and store result in dest + * + * source matrix will remain same + * + * this is POST transform, applies to existing transform as last transform + * + * @param[in] m affine transform + * @param[in] v translate vector [x, y, z] + * @param[out] dest translated matrix + */ +CGLM_INLINE +void +glm_translated_to(mat4 m, vec3 v, mat4 dest) { + glm_mat4_copy(m, dest); + glm_translated(dest, v); +} + +/*! + * @brief translate existing transform matrix by x factor + * + * this is POST transform, applies to existing transform as last transform + * + * @param[in, out] m affine transform + * @param[in] x x factor + */ +CGLM_INLINE +void +glm_translated_x(mat4 m, float x) { + m[3][0] += x; +} + +/*! + * @brief translate existing transform matrix by y factor + * + * this is POST transform, applies to existing transform as last transform + * + * @param[in, out] m affine transform + * @param[in] y y factor + */ +CGLM_INLINE +void +glm_translated_y(mat4 m, float y) { + m[3][1] += y; +} + +/*! + * @brief translate existing transform matrix by z factor + * + * this is POST transform, applies to existing transform as last transform + * + * @param[in, out] m affine transform + * @param[in] z z factor + */ +CGLM_INLINE +void +glm_translated_z(mat4 m, float z) { + m[3][2] += z; +} + +/*! + * @brief rotate existing transform matrix around X axis by angle + * and store result in dest + * + * this is POST transform, applies to existing transform as last transform + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @param[out] dest rotated matrix + */ +CGLM_INLINE +void +glm_rotated_x(mat4 m, float angle, mat4 dest) { + CGLM_ALIGN_MAT mat4 t = GLM_MAT4_IDENTITY_INIT; + float c, s; + + c = cosf(angle); + s = sinf(angle); + + t[1][1] = c; + t[1][2] = s; + t[2][1] = -s; + t[2][2] = c; + + glm_mul_rot(t, m, dest); +} + +/*! + * @brief rotate existing transform matrix around Y axis by angle + * and store result in dest + * + * this is POST transform, applies to existing transform as last transform + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @param[out] dest rotated matrix + */ +CGLM_INLINE +void +glm_rotated_y(mat4 m, float angle, mat4 dest) { + CGLM_ALIGN_MAT mat4 t = GLM_MAT4_IDENTITY_INIT; + float c, s; + + c = cosf(angle); + s = sinf(angle); + + t[0][0] = c; + t[0][2] = -s; + t[2][0] = s; + t[2][2] = c; + + glm_mul_rot(t, m, dest); +} + +/*! + * @brief rotate existing transform matrix around Z axis by angle + * and store result in dest + * + * this is POST transform, applies to existing transform as last transform + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @param[out] dest rotated matrix + */ +CGLM_INLINE +void +glm_rotated_z(mat4 m, float angle, mat4 dest) { + CGLM_ALIGN_MAT mat4 t = GLM_MAT4_IDENTITY_INIT; + float c, s; + + c = cosf(angle); + s = sinf(angle); + + t[0][0] = c; + t[0][1] = s; + t[1][0] = -s; + t[1][1] = c; + + glm_mul_rot(t, m, dest); +} + +/*! + * @brief rotate existing transform matrix around given axis by angle + * + * this is POST transform, applies to existing transform as last transform + * + * @param[in, out] m affine transform + * @param[in] angle angle (radians) + * @param[in] axis axis + */ +CGLM_INLINE +void +glm_rotated(mat4 m, float angle, vec3 axis) { + CGLM_ALIGN_MAT mat4 rot; + glm_rotate_make(rot, angle, axis); + glm_mul_rot(rot, m, m); +} + +/*! + * @brief rotate existing transform + * around given axis by angle at given pivot point (rotation center) + * + * this is POST transform, applies to existing transform as last transform + * + * @param[in, out] m affine transform + * @param[in] pivot rotation center + * @param[in] angle angle (radians) + * @param[in] axis axis + */ +CGLM_INLINE +void +glm_rotated_at(mat4 m, vec3 pivot, float angle, vec3 axis) { + CGLM_ALIGN(8) vec3 pivotInv; + + glm_vec3_negate_to(pivot, pivotInv); + + glm_translated(m, pivot); + glm_rotated(m, angle, axis); + glm_translated(m, pivotInv); +} + +/*! + * @brief rotate existing transform matrix around given axis by angle around self (doesn't affected by position) + * + * this is POST transform, applies to existing transform as last transform + * + * @param[in, out] m affine transform + * @param[in] angle angle (radians) + * @param[in] axis axis + */ +CGLM_INLINE +void +glm_spinned(mat4 m, float angle, vec3 axis) { + CGLM_ALIGN_MAT mat4 rot; + glm_rotate_atm(rot, m[3], angle, axis); + glm_mat4_mul(rot, m, m); +} + +#endif /* cglm_affine_post_h */ diff --git a/cglm/include/cglm/affine-pre.h b/cglm/include/cglm/affine-pre.h new file mode 100644 index 0000000..2fa77f7 --- /dev/null +++ b/cglm/include/cglm/affine-pre.h @@ -0,0 +1,304 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_affine_pre_h +#define cglm_affine_pre_h + +/* + Functions: + CGLM_INLINE void glm_translate_to(mat4 m, vec3 v, mat4 dest); + CGLM_INLINE void glm_translate(mat4 m, vec3 v); + CGLM_INLINE void glm_translate_x(mat4 m, float to); + CGLM_INLINE void glm_translate_y(mat4 m, float to); + CGLM_INLINE void glm_translate_z(mat4 m, float to); + CGLM_INLINE void glm_rotate_x(mat4 m, float angle, mat4 dest); + CGLM_INLINE void glm_rotate_y(mat4 m, float angle, mat4 dest); + CGLM_INLINE void glm_rotate_z(mat4 m, float angle, mat4 dest); + CGLM_INLINE void glm_rotate(mat4 m, float angle, vec3 axis); + CGLM_INLINE void glm_rotate_at(mat4 m, vec3 pivot, float angle, vec3 axis); + CGLM_INLINE void glm_rotate_atm(mat4 m, vec3 pivot, float angle, vec3 axis); + CGLM_INLINE void glm_spin(mat4 m, float angle, vec3 axis); + */ + +#include "common.h" +#include "util.h" +#include "vec3.h" +#include "vec4.h" +#include "mat4.h" +#include "affine-mat.h" + +/*! + * @brief translate existing transform matrix by v vector + * and stores result in same matrix + * + * @param[in, out] m affine transform + * @param[in] v translate vector [x, y, z] + */ +CGLM_INLINE +void +glm_translate(mat4 m, vec3 v) { +#if defined(CGLM_SIMD) + glmm_128 m0, m1, m2, m3; + + m0 = glmm_load(m[0]); + m1 = glmm_load(m[1]); + m2 = glmm_load(m[2]); + m3 = glmm_load(m[3]); + + glmm_store(m[3], + glmm_fmadd(m0, glmm_set1(v[0]), + glmm_fmadd(m1, glmm_set1(v[1]), + glmm_fmadd(m2, glmm_set1(v[2]), m3)))); +#else + glm_vec4_muladds(m[0], v[0], m[3]); + glm_vec4_muladds(m[1], v[1], m[3]); + glm_vec4_muladds(m[2], v[2], m[3]); +#endif +} + +/*! + * @brief translate existing transform matrix by v vector + * and store result in dest + * + * source matrix will remain same + * + * @param[in] m affine transform + * @param[in] v translate vector [x, y, z] + * @param[out] dest translated matrix + */ +CGLM_INLINE +void +glm_translate_to(mat4 m, vec3 v, mat4 dest) { + glm_mat4_copy(m, dest); + glm_translate(dest, v); +} + +/*! + * @brief translate existing transform matrix by x factor + * + * @param[in, out] m affine transform + * @param[in] x x factor + */ +CGLM_INLINE +void +glm_translate_x(mat4 m, float x) { +#if defined(CGLM_SIMD) + glmm_store(m[3], glmm_fmadd(glmm_load(m[0]), glmm_set1(x), glmm_load(m[3]))); +#else + vec4 v1; + glm_vec4_scale(m[0], x, v1); + glm_vec4_add(v1, m[3], m[3]); +#endif +} + +/*! + * @brief translate existing transform matrix by y factor + * + * @param[in, out] m affine transform + * @param[in] y y factor + */ +CGLM_INLINE +void +glm_translate_y(mat4 m, float y) { +#if defined(CGLM_SIMD) + glmm_store(m[3], glmm_fmadd(glmm_load(m[1]), glmm_set1(y), glmm_load(m[3]))); +#else + vec4 v1; + glm_vec4_scale(m[1], y, v1); + glm_vec4_add(v1, m[3], m[3]); +#endif +} + +/*! + * @brief translate existing transform matrix by z factor + * + * @param[in, out] m affine transform + * @param[in] z z factor + */ +CGLM_INLINE +void +glm_translate_z(mat4 m, float z) { +#if defined(CGLM_SIMD) + glmm_store(m[3], glmm_fmadd(glmm_load(m[2]), glmm_set1(z), glmm_load(m[3]))); +#else + vec4 v1; + glm_vec4_scale(m[2], z, v1); + glm_vec4_add(v1, m[3], m[3]); +#endif +} + +/*! + * @brief rotate existing transform matrix around X axis by angle + * and store result in dest + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @param[out] dest rotated matrix + */ +CGLM_INLINE +void +glm_rotate_x(mat4 m, float angle, mat4 dest) { + CGLM_ALIGN_MAT mat4 t = GLM_MAT4_IDENTITY_INIT; + float c, s; + + c = cosf(angle); + s = sinf(angle); + + t[1][1] = c; + t[1][2] = s; + t[2][1] = -s; + t[2][2] = c; + + glm_mul_rot(m, t, dest); +} + +/*! + * @brief rotate existing transform matrix around Y axis by angle + * and store result in dest + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @param[out] dest rotated matrix + */ +CGLM_INLINE +void +glm_rotate_y(mat4 m, float angle, mat4 dest) { + CGLM_ALIGN_MAT mat4 t = GLM_MAT4_IDENTITY_INIT; + float c, s; + + c = cosf(angle); + s = sinf(angle); + + t[0][0] = c; + t[0][2] = -s; + t[2][0] = s; + t[2][2] = c; + + glm_mul_rot(m, t, dest); +} + +/*! + * @brief rotate existing transform matrix around Z axis by angle + * and store result in dest + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @param[out] dest rotated matrix + */ +CGLM_INLINE +void +glm_rotate_z(mat4 m, float angle, mat4 dest) { + CGLM_ALIGN_MAT mat4 t = GLM_MAT4_IDENTITY_INIT; + float c, s; + + c = cosf(angle); + s = sinf(angle); + + t[0][0] = c; + t[0][1] = s; + t[1][0] = -s; + t[1][1] = c; + + glm_mul_rot(m, t, dest); +} + +/*! + * @brief rotate existing transform matrix + * around given axis by angle at ORIGIN (0,0,0) + * + * **❗️IMPORTANT ❗️** + * + * If you need to rotate object around itself e.g. center of object or at + * some point [of object] then `glm_rotate_at()` would be better choice to do so. + * + * Even if object's model transform is identity, rotation may not be around + * center of object if object does not lay out at ORIGIN perfectly. + * + * Using `glm_rotate_at()` with center of bounding shape ( AABB, Sphere ... ) + * would be an easy option to rotate around object if object is not at origin. + * + * One another option to rotate around itself at any point is `glm_spin()` + * which is perfect if only rotating around model position is desired e.g. not + * specific point on model for instance center of geometry or center of mass, + * again if geometry is not perfectly centered at origin at identity transform, + * rotation may not be around geometry. + * + * @param[in, out] m affine transform + * @param[in] angle angle (radians) + * @param[in] axis axis + */ +CGLM_INLINE +void +glm_rotate(mat4 m, float angle, vec3 axis) { + CGLM_ALIGN_MAT mat4 rot; + glm_rotate_make(rot, angle, axis); + glm_mul_rot(m, rot, m); +} + +/*! + * @brief rotate existing transform + * around given axis by angle at given pivot point (rotation center) + * + * @param[in, out] m affine transform + * @param[in] pivot rotation center + * @param[in] angle angle (radians) + * @param[in] axis axis + */ +CGLM_INLINE +void +glm_rotate_at(mat4 m, vec3 pivot, float angle, vec3 axis) { + CGLM_ALIGN(8) vec3 pivotInv; + + glm_vec3_negate_to(pivot, pivotInv); + + glm_translate(m, pivot); + glm_rotate(m, angle, axis); + glm_translate(m, pivotInv); +} + +/*! + * @brief creates NEW rotation matrix by angle and axis at given point + * + * this creates rotation matrix, it assumes you don't have a matrix + * + * this should work faster than glm_rotate_at because it reduces + * one glm_translate. + * + * @param[out] m affine transform + * @param[in] pivot rotation center + * @param[in] angle angle (radians) + * @param[in] axis axis + */ +CGLM_INLINE +void +glm_rotate_atm(mat4 m, vec3 pivot, float angle, vec3 axis) { + CGLM_ALIGN(8) vec3 pivotInv; + + glm_vec3_negate_to(pivot, pivotInv); + + glm_translate_make(m, pivot); + glm_rotate(m, angle, axis); + glm_translate(m, pivotInv); +} + +/*! + * @brief rotate existing transform matrix + * around given axis by angle around self (doesn't affected by position) + * + * @param[in, out] m affine transform + * @param[in] angle angle (radians) + * @param[in] axis axis + */ +CGLM_INLINE +void +glm_spin(mat4 m, float angle, vec3 axis) { + CGLM_ALIGN_MAT mat4 rot; + glm_rotate_atm(rot, m[3], angle, axis); + glm_mat4_mul(m, rot, m); +} + +#endif /* cglm_affine_pre_h */ diff --git a/cglm/include/cglm/affine.h b/cglm/include/cglm/affine.h new file mode 100644 index 0000000..2c608f7 --- /dev/null +++ b/cglm/include/cglm/affine.h @@ -0,0 +1,238 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_translate_to(mat4 m, vec3 v, mat4 dest); + CGLM_INLINE void glm_translate(mat4 m, vec3 v); + CGLM_INLINE void glm_translate_x(mat4 m, float to); + CGLM_INLINE void glm_translate_y(mat4 m, float to); + CGLM_INLINE void glm_translate_z(mat4 m, float to); + CGLM_INLINE void glm_translate_make(mat4 m, vec3 v); + CGLM_INLINE void glm_scale_to(mat4 m, vec3 v, mat4 dest); + CGLM_INLINE void glm_scale_make(mat4 m, vec3 v); + CGLM_INLINE void glm_scale(mat4 m, vec3 v); + CGLM_INLINE void glm_scale_uni(mat4 m, float s); + CGLM_INLINE void glm_rotate_x(mat4 m, float angle, mat4 dest); + CGLM_INLINE void glm_rotate_y(mat4 m, float angle, mat4 dest); + CGLM_INLINE void glm_rotate_z(mat4 m, float angle, mat4 dest); + CGLM_INLINE void glm_rotate_make(mat4 m, float angle, vec3 axis); + CGLM_INLINE void glm_rotate(mat4 m, float angle, vec3 axis); + CGLM_INLINE void glm_rotate_at(mat4 m, vec3 pivot, float angle, vec3 axis); + CGLM_INLINE void glm_rotate_atm(mat4 m, vec3 pivot, float angle, vec3 axis); + CGLM_INLINE void glm_spin(mat4 m, float angle, vec3 axis); + CGLM_INLINE void glm_decompose_scalev(mat4 m, vec3 s); + CGLM_INLINE bool glm_uniscaled(mat4 m); + CGLM_INLINE void glm_decompose_rs(mat4 m, mat4 r, vec3 s); + CGLM_INLINE void glm_decompose(mat4 m, vec4 t, mat4 r, vec3 s); + */ + +#ifndef cglm_affine_h +#define cglm_affine_h + +#include "common.h" +#include "util.h" +#include "vec3.h" +#include "vec4.h" +#include "mat4.h" +#include "affine-mat.h" + +/*! + * @brief creates NEW translate transform matrix by v vector + * + * @param[out] m affine transform + * @param[in] v translate vector [x, y, z] + */ +CGLM_INLINE +void +glm_translate_make(mat4 m, vec3 v) { + glm_mat4_identity(m); + glm_vec3_copy(v, m[3]); +} + +/*! + * @brief scale existing transform matrix by v vector + * and store result in dest + * + * @param[in] m affine transform + * @param[in] v scale vector [x, y, z] + * @param[out] dest scaled matrix + */ +CGLM_INLINE +void +glm_scale_to(mat4 m, vec3 v, mat4 dest) { + glm_vec4_scale(m[0], v[0], dest[0]); + glm_vec4_scale(m[1], v[1], dest[1]); + glm_vec4_scale(m[2], v[2], dest[2]); + + glm_vec4_copy(m[3], dest[3]); +} + +/*! + * @brief creates NEW scale matrix by v vector + * + * @param[out] m affine transform + * @param[in] v scale vector [x, y, z] + */ +CGLM_INLINE +void +glm_scale_make(mat4 m, vec3 v) { + glm_mat4_identity(m); + m[0][0] = v[0]; + m[1][1] = v[1]; + m[2][2] = v[2]; +} + +/*! + * @brief scales existing transform matrix by v vector + * and stores result in same matrix + * + * @param[in, out] m affine transform + * @param[in] v scale vector [x, y, z] + */ +CGLM_INLINE +void +glm_scale(mat4 m, vec3 v) { + glm_scale_to(m, v, m); +} + +/*! + * @brief applies uniform scale to existing transform matrix v = [s, s, s] + * and stores result in same matrix + * + * @param[in, out] m affine transform + * @param[in] s scale factor + */ +CGLM_INLINE +void +glm_scale_uni(mat4 m, float s) { + CGLM_ALIGN(8) vec3 v = { s, s, s }; + glm_scale_to(m, v, m); +} + +/*! + * @brief creates NEW rotation matrix by angle and axis + * + * axis will be normalized so you don't need to normalize it + * + * @param[out] m affine transform + * @param[in] angle angle (radians) + * @param[in] axis axis + */ +CGLM_INLINE +void +glm_rotate_make(mat4 m, float angle, vec3 axis) { + CGLM_ALIGN(8) vec3 axisn, v, vs; + float c; + + c = cosf(angle); + + glm_vec3_normalize_to(axis, axisn); + glm_vec3_scale(axisn, 1.0f - c, v); + glm_vec3_scale(axisn, sinf(angle), vs); + + glm_vec3_scale(axisn, v[0], m[0]); + glm_vec3_scale(axisn, v[1], m[1]); + glm_vec3_scale(axisn, v[2], m[2]); + + m[0][0] += c; m[1][0] -= vs[2]; m[2][0] += vs[1]; + m[0][1] += vs[2]; m[1][1] += c; m[2][1] -= vs[0]; + m[0][2] -= vs[1]; m[1][2] += vs[0]; m[2][2] += c; + + m[0][3] = m[1][3] = m[2][3] = m[3][0] = m[3][1] = m[3][2] = 0.0f; + m[3][3] = 1.0f; +} + +/*! + * @brief decompose scale vector + * + * @param[in] m affine transform + * @param[out] s scale vector (Sx, Sy, Sz) + */ +CGLM_INLINE +void +glm_decompose_scalev(mat4 m, vec3 s) { + s[0] = glm_vec3_norm(m[0]); + s[1] = glm_vec3_norm(m[1]); + s[2] = glm_vec3_norm(m[2]); +} + +/*! + * @brief returns true if matrix is uniform scaled. This is helpful for + * creating normal matrix. + * + * @param[in] m m + * + * @return boolean + */ +CGLM_INLINE +bool +glm_uniscaled(mat4 m) { + CGLM_ALIGN(8) vec3 s; + glm_decompose_scalev(m, s); + return glm_vec3_eq_all(s); +} + +/*! + * @brief decompose rotation matrix (mat4) and scale vector [Sx, Sy, Sz] + * DON'T pass projected matrix here + * + * @param[in] m affine transform + * @param[out] r rotation matrix + * @param[out] s scale matrix + */ +CGLM_INLINE +void +glm_decompose_rs(mat4 m, mat4 r, vec3 s) { + CGLM_ALIGN(16) vec4 t = {0.0f, 0.0f, 0.0f, 1.0f}; + CGLM_ALIGN(8) vec3 v; + + glm_vec4_copy(m[0], r[0]); + glm_vec4_copy(m[1], r[1]); + glm_vec4_copy(m[2], r[2]); + glm_vec4_copy(t, r[3]); + + s[0] = glm_vec3_norm(m[0]); + s[1] = glm_vec3_norm(m[1]); + s[2] = glm_vec3_norm(m[2]); + + glm_vec4_scale(r[0], 1.0f/s[0], r[0]); + glm_vec4_scale(r[1], 1.0f/s[1], r[1]); + glm_vec4_scale(r[2], 1.0f/s[2], r[2]); + + /* Note from Apple Open Source (assume that the matrix is orthonormal): + check for a coordinate system flip. If the determinant + is -1, then negate the matrix and the scaling factors. */ + glm_vec3_cross(m[0], m[1], v); + if (glm_vec3_dot(v, m[2]) < 0.0f) { + glm_vec4_negate(r[0]); + glm_vec4_negate(r[1]); + glm_vec4_negate(r[2]); + glm_vec3_negate(s); + } +} + +/*! + * @brief decompose affine transform, TODO: extract shear factors. + * DON'T pass projected matrix here + * + * @param[in] m affine transform + * @param[out] t translation vector + * @param[out] r rotation matrix (mat4) + * @param[out] s scaling vector [X, Y, Z] + */ +CGLM_INLINE +void +glm_decompose(mat4 m, vec4 t, mat4 r, vec3 s) { + glm_vec4_copy(m[3], t); + glm_decompose_rs(m, r, s); +} + +#include "affine-pre.h" +#include "affine-post.h" + +#endif /* cglm_affine_h */ diff --git a/cglm/include/cglm/affine2d.h b/cglm/include/cglm/affine2d.h new file mode 100644 index 0000000..0dcf50a --- /dev/null +++ b/cglm/include/cglm/affine2d.h @@ -0,0 +1,268 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_translate2d(mat3 m, vec2 v) + CGLM_INLINE void glm_translate2d_to(mat3 m, vec2 v, mat3 dest) + CGLM_INLINE void glm_translate2d_x(mat3 m, float x) + CGLM_INLINE void glm_translate2d_y(mat3 m, float y) + CGLM_INLINE void glm_translate2d_make(mat3 m, vec2 v) + CGLM_INLINE void glm_scale2d_to(mat3 m, vec2 v, mat3 dest) + CGLM_INLINE void glm_scale2d_make(mat3 m, vec2 v) + CGLM_INLINE void glm_scale2d(mat3 m, vec2 v) + CGLM_INLINE void glm_scale2d_uni(mat3 m, float s) + CGLM_INLINE void glm_rotate2d_make(mat3 m, float angle) + CGLM_INLINE void glm_rotate2d(mat3 m, float angle) + CGLM_INLINE void glm_rotate2d_to(mat3 m, float angle, mat3 dest) + */ + +#ifndef cglm_affine2d_h +#define cglm_affine2d_h + +#include "common.h" +#include "util.h" +#include "vec2.h" +#include "mat3.h" + +/*! + * @brief translate existing 2d transform matrix by v vector + * and stores result in same matrix + * + * @param[in, out] m affine transform + * @param[in] v translate vector [x, y] + */ +CGLM_INLINE +void +glm_translate2d(mat3 m, vec2 v) { + m[2][0] = m[0][0] * v[0] + m[1][0] * v[1] + m[2][0]; + m[2][1] = m[0][1] * v[0] + m[1][1] * v[1] + m[2][1]; + m[2][2] = m[0][2] * v[0] + m[1][2] * v[1] + m[2][2]; +} + +/*! + * @brief translate existing 2d transform matrix by v vector + * and store result in dest + * + * source matrix will remain same + * + * @param[in] m affine transform + * @param[in] v translate vector [x, y] + * @param[out] dest translated matrix + */ +CGLM_INLINE +void +glm_translate2d_to(mat3 m, vec2 v, mat3 dest) { + glm_mat3_copy(m, dest); + glm_translate2d(dest, v); +} + +/*! + * @brief translate existing 2d transform matrix by x factor + * + * @param[in, out] m affine transform + * @param[in] x x factor + */ +CGLM_INLINE +void +glm_translate2d_x(mat3 m, float x) { + m[2][0] = m[0][0] * x + m[2][0]; + m[2][1] = m[0][1] * x + m[2][1]; + m[2][2] = m[0][2] * x + m[2][2]; +} + +/*! + * @brief translate existing 2d transform matrix by y factor + * + * @param[in, out] m affine transform + * @param[in] y y factor + */ +CGLM_INLINE +void +glm_translate2d_y(mat3 m, float y) { + m[2][0] = m[1][0] * y + m[2][0]; + m[2][1] = m[1][1] * y + m[2][1]; + m[2][2] = m[1][2] * y + m[2][2]; +} + +/*! + * @brief creates NEW translate 2d transform matrix by v vector + * + * @param[out] m affine transform + * @param[in] v translate vector [x, y] + */ +CGLM_INLINE +void +glm_translate2d_make(mat3 m, vec2 v) { + glm_mat3_identity(m); + m[2][0] = v[0]; + m[2][1] = v[1]; +} + +/*! + * @brief scale existing 2d transform matrix by v vector + * and store result in dest + * + * @param[in] m affine transform + * @param[in] v scale vector [x, y] + * @param[out] dest scaled matrix + */ +CGLM_INLINE +void +glm_scale2d_to(mat3 m, vec2 v, mat3 dest) { + dest[0][0] = m[0][0] * v[0]; + dest[0][1] = m[0][1] * v[0]; + dest[0][2] = m[0][2] * v[0]; + + dest[1][0] = m[1][0] * v[1]; + dest[1][1] = m[1][1] * v[1]; + dest[1][2] = m[1][2] * v[1]; + + dest[2][0] = m[2][0]; + dest[2][1] = m[2][1]; + dest[2][2] = m[2][2]; +} + +/*! + * @brief creates NEW 2d scale matrix by v vector + * + * @param[out] m affine transform + * @param[in] v scale vector [x, y] + */ +CGLM_INLINE +void +glm_scale2d_make(mat3 m, vec2 v) { + glm_mat3_identity(m); + m[0][0] = v[0]; + m[1][1] = v[1]; +} + +/*! + * @brief scales existing 2d transform matrix by v vector + * and stores result in same matrix + * + * @param[in, out] m affine transform + * @param[in] v scale vector [x, y] + */ +CGLM_INLINE +void +glm_scale2d(mat3 m, vec2 v) { + m[0][0] = m[0][0] * v[0]; + m[0][1] = m[0][1] * v[0]; + m[0][2] = m[0][2] * v[0]; + + m[1][0] = m[1][0] * v[1]; + m[1][1] = m[1][1] * v[1]; + m[1][2] = m[1][2] * v[1]; +} + +/*! + * @brief applies uniform scale to existing 2d transform matrix v = [s, s] + * and stores result in same matrix + * + * @param[in, out] m affine transform + * @param[in] s scale factor + */ +CGLM_INLINE +void +glm_scale2d_uni(mat3 m, float s) { + m[0][0] = m[0][0] * s; + m[0][1] = m[0][1] * s; + m[0][2] = m[0][2] * s; + + m[1][0] = m[1][0] * s; + m[1][1] = m[1][1] * s; + m[1][2] = m[1][2] * s; +} + +/*! + * @brief creates NEW rotation matrix by angle around Z axis + * + * @param[out] m affine transform + * @param[in] angle angle (radians) + */ +CGLM_INLINE +void +glm_rotate2d_make(mat3 m, float angle) { + float c, s; + + s = sinf(angle); + c = cosf(angle); + + m[0][0] = c; + m[0][1] = s; + m[0][2] = 0; + + m[1][0] = -s; + m[1][1] = c; + m[1][2] = 0; + + m[2][0] = 0.0f; + m[2][1] = 0.0f; + m[2][2] = 1.0f; +} + +/*! + * @brief rotate existing 2d transform matrix around Z axis by angle + * and store result in same matrix + * + * @param[in, out] m affine transform + * @param[in] angle angle (radians) + */ +CGLM_INLINE +void +glm_rotate2d(mat3 m, float angle) { + float m00 = m[0][0], m10 = m[1][0], + m01 = m[0][1], m11 = m[1][1], + m02 = m[0][2], m12 = m[1][2]; + float c, s; + + s = sinf(angle); + c = cosf(angle); + + m[0][0] = m00 * c + m10 * s; + m[0][1] = m01 * c + m11 * s; + m[0][2] = m02 * c + m12 * s; + + m[1][0] = m00 * -s + m10 * c; + m[1][1] = m01 * -s + m11 * c; + m[1][2] = m02 * -s + m12 * c; +} + +/*! + * @brief rotate existing 2d transform matrix around Z axis by angle + * and store result in dest + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_rotate2d_to(mat3 m, float angle, mat3 dest) { + float m00 = m[0][0], m10 = m[1][0], + m01 = m[0][1], m11 = m[1][1], + m02 = m[0][2], m12 = m[1][2]; + float c, s; + + s = sinf(angle); + c = cosf(angle); + + dest[0][0] = m00 * c + m10 * s; + dest[0][1] = m01 * c + m11 * s; + dest[0][2] = m02 * c + m12 * s; + + dest[1][0] = m00 * -s + m10 * c; + dest[1][1] = m01 * -s + m11 * c; + dest[1][2] = m02 * -s + m12 * c; + + dest[2][0] = m[2][0]; + dest[2][1] = m[2][1]; + dest[2][2] = m[2][2]; +} + +#endif /* cglm_affine2d_h */ diff --git a/cglm/include/cglm/applesimd.h b/cglm/include/cglm/applesimd.h new file mode 100644 index 0000000..3608bb3 --- /dev/null +++ b/cglm/include/cglm/applesimd.h @@ -0,0 +1,95 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_applesimd_h +#define cglm_applesimd_h +#if defined(__APPLE__) \ + && defined(SIMD_COMPILER_HAS_REQUIRED_FEATURES) \ + && defined(SIMD_BASE) \ + && defined(SIMD_TYPES) \ + && defined(SIMD_VECTOR_TYPES) + +#include "common.h" + +/*! +* @brief converts mat4 to Apple's simd type simd_float4x4 +* @return simd_float4x4 +*/ +CGLM_INLINE +simd_float4x4 +glm_mat4_applesimd(mat4 m) { + simd_float4x4 t; + + t.columns[0][0] = m[0][0]; + t.columns[0][1] = m[0][1]; + t.columns[0][2] = m[0][2]; + t.columns[0][3] = m[0][3]; + + t.columns[1][0] = m[1][0]; + t.columns[1][1] = m[1][1]; + t.columns[1][2] = m[1][2]; + t.columns[1][3] = m[1][3]; + + t.columns[2][0] = m[2][0]; + t.columns[2][1] = m[2][1]; + t.columns[2][2] = m[2][2]; + t.columns[2][3] = m[2][3]; + + t.columns[3][0] = m[3][0]; + t.columns[3][1] = m[3][1]; + t.columns[3][2] = m[3][2]; + t.columns[3][3] = m[3][3]; + + return t; +} + +/*! +* @brief converts mat3 to Apple's simd type simd_float3x3 +* @return simd_float3x3 +*/ +CGLM_INLINE +simd_float3x3 +glm_mat3_applesimd(mat3 m) { + simd_float3x3 t; + + t.columns[0][0] = m[0][0]; + t.columns[0][1] = m[0][1]; + t.columns[0][2] = m[0][2]; + + t.columns[1][0] = m[1][0]; + t.columns[1][1] = m[1][1]; + t.columns[1][2] = m[1][2]; + + t.columns[2][0] = m[2][0]; + t.columns[2][1] = m[2][1]; + t.columns[2][2] = m[2][2]; + + return t; +} + +/*! +* @brief converts vec4 to Apple's simd type simd_float4 +* @return simd_float4 +*/ +CGLM_INLINE +simd_float4 +glm_vec4_applesimd(vec4 v) { + return (simd_float4){v[0], v[1], v[2], v[3]}; +} + +/*! +* @brief converts vec3 to Apple's simd type simd_float3 +* @return v +*/ +CGLM_INLINE +simd_float3 +glm_vec3_applesimd(vec3 v) { + return (simd_float3){v[0], v[1], v[2]}; +} + +#endif +#endif /* cglm_applesimd_h */ diff --git a/cglm/include/cglm/bezier.h b/cglm/include/cglm/bezier.h new file mode 100644 index 0000000..a6e5f8a --- /dev/null +++ b/cglm/include/cglm/bezier.h @@ -0,0 +1,154 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_bezier_h +#define cglm_bezier_h + +#include "common.h" + +#define GLM_BEZIER_MAT_INIT {{-1.0f, 3.0f, -3.0f, 1.0f}, \ + { 3.0f, -6.0f, 3.0f, 0.0f}, \ + {-3.0f, 3.0f, 0.0f, 0.0f}, \ + { 1.0f, 0.0f, 0.0f, 0.0f}} +#define GLM_HERMITE_MAT_INIT {{ 2.0f, -3.0f, 0.0f, 1.0f}, \ + {-2.0f, 3.0f, 0.0f, 0.0f}, \ + { 1.0f, -2.0f, 1.0f, 0.0f}, \ + { 1.0f, -1.0f, 0.0f, 0.0f}} +/* for C only */ +#define GLM_BEZIER_MAT ((mat4)GLM_BEZIER_MAT_INIT) +#define GLM_HERMITE_MAT ((mat4)GLM_HERMITE_MAT_INIT) + +#define CGLM_DECASTEL_EPS 1e-9f +#define CGLM_DECASTEL_MAX 1000 +#define CGLM_DECASTEL_SMALL 1e-20f + +/*! + * @brief cubic bezier interpolation + * + * Formula: + * B(s) = P0*(1-s)^3 + 3*C0*s*(1-s)^2 + 3*C1*s^2*(1-s) + P1*s^3 + * + * similar result using matrix: + * B(s) = glm_smc(t, GLM_BEZIER_MAT, (vec4){p0, c0, c1, p1}) + * + * glm_eq(glm_smc(...), glm_bezier(...)) should return TRUE + * + * @param[in] s parameter between 0 and 1 + * @param[in] p0 begin point + * @param[in] c0 control point 1 + * @param[in] c1 control point 2 + * @param[in] p1 end point + * + * @return B(s) + */ +CGLM_INLINE +float +glm_bezier(float s, float p0, float c0, float c1, float p1) { + float x, xx, ss, xs3, a; + + x = 1.0f - s; + xx = x * x; + ss = s * s; + xs3 = (s - ss) * 3.0f; + a = p0 * xx + c0 * xs3; + + return a + s * (c1 * xs3 + p1 * ss - a); +} + +/*! + * @brief cubic hermite interpolation + * + * Formula: + * H(s) = P0*(2*s^3 - 3*s^2 + 1) + T0*(s^3 - 2*s^2 + s) + * + P1*(-2*s^3 + 3*s^2) + T1*(s^3 - s^2) + * + * similar result using matrix: + * H(s) = glm_smc(t, GLM_HERMITE_MAT, (vec4){p0, p1, c0, c1}) + * + * glm_eq(glm_smc(...), glm_hermite(...)) should return TRUE + * + * @param[in] s parameter between 0 and 1 + * @param[in] p0 begin point + * @param[in] t0 tangent 1 + * @param[in] t1 tangent 2 + * @param[in] p1 end point + * + * @return H(s) + */ +CGLM_INLINE +float +glm_hermite(float s, float p0, float t0, float t1, float p1) { + float ss, d, a, b, c, e, f; + + ss = s * s; + a = ss + ss; + c = a + ss; + b = a * s; + d = s * ss; + f = d - ss; + e = b - c; + + return p0 * (e + 1.0f) + t0 * (f - ss + s) + t1 * f - p1 * e; +} + +/*! + * @brief iterative way to solve cubic equation + * + * @param[in] prm parameter between 0 and 1 + * @param[in] p0 begin point + * @param[in] c0 control point 1 + * @param[in] c1 control point 2 + * @param[in] p1 end point + * + * @return parameter to use in cubic equation + */ +CGLM_INLINE +float +glm_decasteljau(float prm, float p0, float c0, float c1, float p1) { + float u, v, a, b, c, d, e, f; + int i; + + if (prm - p0 < CGLM_DECASTEL_SMALL) + return 0.0f; + + if (p1 - prm < CGLM_DECASTEL_SMALL) + return 1.0f; + + u = 0.0f; + v = 1.0f; + + for (i = 0; i < CGLM_DECASTEL_MAX; i++) { + /* de Casteljau Subdivision */ + a = (p0 + c0) * 0.5f; + b = (c0 + c1) * 0.5f; + c = (c1 + p1) * 0.5f; + d = (a + b) * 0.5f; + e = (b + c) * 0.5f; + f = (d + e) * 0.5f; /* this one is on the curve! */ + + /* The curve point is close enough to our wanted t */ + if (fabsf(f - prm) < CGLM_DECASTEL_EPS) + return glm_clamp_zo((u + v) * 0.5f); + + /* dichotomy */ + if (f < prm) { + p0 = f; + c0 = e; + c1 = c; + u = (u + v) * 0.5f; + } else { + c0 = a; + c1 = d; + p1 = f; + v = (u + v) * 0.5f; + } + } + + return glm_clamp_zo((u + v) * 0.5f); +} + +#endif /* cglm_bezier_h */ diff --git a/cglm/include/cglm/box.h b/cglm/include/cglm/box.h new file mode 100644 index 0000000..8bba678 --- /dev/null +++ b/cglm/include/cglm/box.h @@ -0,0 +1,281 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_box_h +#define cglm_box_h + +#include "common.h" +#include "vec3.h" +#include "vec4.h" +#include "util.h" + +/*! + * @brief apply transform to Axis-Aligned Bounding Box + * + * @param[in] box bounding box + * @param[in] m transform matrix + * @param[out] dest transformed bounding box + */ +CGLM_INLINE +void +glm_aabb_transform(vec3 box[2], mat4 m, vec3 dest[2]) { + vec3 v[2], xa, xb, ya, yb, za, zb; + + glm_vec3_scale(m[0], box[0][0], xa); + glm_vec3_scale(m[0], box[1][0], xb); + + glm_vec3_scale(m[1], box[0][1], ya); + glm_vec3_scale(m[1], box[1][1], yb); + + glm_vec3_scale(m[2], box[0][2], za); + glm_vec3_scale(m[2], box[1][2], zb); + + /* translation + min(xa, xb) + min(ya, yb) + min(za, zb) */ + glm_vec3(m[3], v[0]); + glm_vec3_minadd(xa, xb, v[0]); + glm_vec3_minadd(ya, yb, v[0]); + glm_vec3_minadd(za, zb, v[0]); + + /* translation + max(xa, xb) + max(ya, yb) + max(za, zb) */ + glm_vec3(m[3], v[1]); + glm_vec3_maxadd(xa, xb, v[1]); + glm_vec3_maxadd(ya, yb, v[1]); + glm_vec3_maxadd(za, zb, v[1]); + + glm_vec3_copy(v[0], dest[0]); + glm_vec3_copy(v[1], dest[1]); +} + +/*! + * @brief merges two AABB bounding box and creates new one + * + * two box must be in same space, if one of box is in different space then + * you should consider to convert it's space by glm_box_space + * + * @param[in] box1 bounding box 1 + * @param[in] box2 bounding box 2 + * @param[out] dest merged bounding box + */ +CGLM_INLINE +void +glm_aabb_merge(vec3 box1[2], vec3 box2[2], vec3 dest[2]) { + dest[0][0] = glm_min(box1[0][0], box2[0][0]); + dest[0][1] = glm_min(box1[0][1], box2[0][1]); + dest[0][2] = glm_min(box1[0][2], box2[0][2]); + + dest[1][0] = glm_max(box1[1][0], box2[1][0]); + dest[1][1] = glm_max(box1[1][1], box2[1][1]); + dest[1][2] = glm_max(box1[1][2], box2[1][2]); +} + +/*! + * @brief crops a bounding box with another one. + * + * this could be useful for getting a bbox which fits with view frustum and + * object bounding boxes. In this case you crop view frustum box with objects + * box + * + * @param[in] box bounding box 1 + * @param[in] cropBox crop box + * @param[out] dest cropped bounding box + */ +CGLM_INLINE +void +glm_aabb_crop(vec3 box[2], vec3 cropBox[2], vec3 dest[2]) { + dest[0][0] = glm_max(box[0][0], cropBox[0][0]); + dest[0][1] = glm_max(box[0][1], cropBox[0][1]); + dest[0][2] = glm_max(box[0][2], cropBox[0][2]); + + dest[1][0] = glm_min(box[1][0], cropBox[1][0]); + dest[1][1] = glm_min(box[1][1], cropBox[1][1]); + dest[1][2] = glm_min(box[1][2], cropBox[1][2]); +} + +/*! + * @brief crops a bounding box with another one. + * + * this could be useful for getting a bbox which fits with view frustum and + * object bounding boxes. In this case you crop view frustum box with objects + * box + * + * @param[in] box bounding box + * @param[in] cropBox crop box + * @param[in] clampBox minimum box + * @param[out] dest cropped bounding box + */ +CGLM_INLINE +void +glm_aabb_crop_until(vec3 box[2], + vec3 cropBox[2], + vec3 clampBox[2], + vec3 dest[2]) { + glm_aabb_crop(box, cropBox, dest); + glm_aabb_merge(clampBox, dest, dest); +} + +/*! + * @brief check if AABB intersects with frustum planes + * + * this could be useful for frustum culling using AABB. + * + * OPTIMIZATION HINT: + * if planes order is similar to LEFT, RIGHT, BOTTOM, TOP, NEAR, FAR + * then this method should run even faster because it would only use two + * planes if object is not inside the two planes + * fortunately cglm extracts planes as this order! just pass what you got! + * + * @param[in] box bounding box + * @param[in] planes frustum planes + */ +CGLM_INLINE +bool +glm_aabb_frustum(vec3 box[2], vec4 planes[6]) { + float *p, dp; + int i; + + for (i = 0; i < 6; i++) { + p = planes[i]; + dp = p[0] * box[p[0] > 0.0f][0] + + p[1] * box[p[1] > 0.0f][1] + + p[2] * box[p[2] > 0.0f][2]; + + if (dp < -p[3]) + return false; + } + + return true; +} + +/*! + * @brief invalidate AABB min and max values + * + * @param[in, out] box bounding box + */ +CGLM_INLINE +void +glm_aabb_invalidate(vec3 box[2]) { + glm_vec3_broadcast(FLT_MAX, box[0]); + glm_vec3_broadcast(-FLT_MAX, box[1]); +} + +/*! + * @brief check if AABB is valid or not + * + * @param[in] box bounding box + */ +CGLM_INLINE +bool +glm_aabb_isvalid(vec3 box[2]) { + return glm_vec3_max(box[0]) != FLT_MAX + && glm_vec3_min(box[1]) != -FLT_MAX; +} + +/*! + * @brief distance between of min and max + * + * @param[in] box bounding box + */ +CGLM_INLINE +float +glm_aabb_size(vec3 box[2]) { + return glm_vec3_distance(box[0], box[1]); +} + +/*! + * @brief radius of sphere which surrounds AABB + * + * @param[in] box bounding box + */ +CGLM_INLINE +float +glm_aabb_radius(vec3 box[2]) { + return glm_aabb_size(box) * 0.5f; +} + +/*! + * @brief computes center point of AABB + * + * @param[in] box bounding box + * @param[out] dest center of bounding box + */ +CGLM_INLINE +void +glm_aabb_center(vec3 box[2], vec3 dest) { + glm_vec3_center(box[0], box[1], dest); +} + +/*! + * @brief check if two AABB intersects + * + * @param[in] box bounding box + * @param[in] other other bounding box + */ +CGLM_INLINE +bool +glm_aabb_aabb(vec3 box[2], vec3 other[2]) { + return (box[0][0] <= other[1][0] && box[1][0] >= other[0][0]) + && (box[0][1] <= other[1][1] && box[1][1] >= other[0][1]) + && (box[0][2] <= other[1][2] && box[1][2] >= other[0][2]); +} + +/*! + * @brief check if AABB intersects with sphere + * + * https://github.com/erich666/GraphicsGems/blob/master/gems/BoxSphere.c + * Solid Box - Solid Sphere test. + * + * Sphere Representation in cglm: [center.x, center.y, center.z, radii] + * + * @param[in] box solid bounding box + * @param[in] s solid sphere + */ +CGLM_INLINE +bool +glm_aabb_sphere(vec3 box[2], vec4 s) { + float dmin; + int a, b, c; + + a = (s[0] < box[0][0]) + (s[0] > box[1][0]); + b = (s[1] < box[0][1]) + (s[1] > box[1][1]); + c = (s[2] < box[0][2]) + (s[2] > box[1][2]); + + dmin = glm_pow2((s[0] - box[!(a - 1)][0]) * (a != 0)) + + glm_pow2((s[1] - box[!(b - 1)][1]) * (b != 0)) + + glm_pow2((s[2] - box[!(c - 1)][2]) * (c != 0)); + + return dmin <= glm_pow2(s[3]); +} + +/*! + * @brief check if point is inside of AABB + * + * @param[in] box bounding box + * @param[in] point point + */ +CGLM_INLINE +bool +glm_aabb_point(vec3 box[2], vec3 point) { + return (point[0] >= box[0][0] && point[0] <= box[1][0]) + && (point[1] >= box[0][1] && point[1] <= box[1][1]) + && (point[2] >= box[0][2] && point[2] <= box[1][2]); +} + +/*! + * @brief check if AABB contains other AABB + * + * @param[in] box bounding box + * @param[in] other other bounding box + */ +CGLM_INLINE +bool +glm_aabb_contains(vec3 box[2], vec3 other[2]) { + return (box[0][0] <= other[0][0] && box[1][0] >= other[1][0]) + && (box[0][1] <= other[0][1] && box[1][1] >= other[1][1]) + && (box[0][2] <= other[0][2] && box[1][2] >= other[1][2]); +} + +#endif /* cglm_box_h */ diff --git a/cglm/include/cglm/call.h b/cglm/include/cglm/call.h new file mode 100644 index 0000000..165f502 --- /dev/null +++ b/cglm/include/cglm/call.h @@ -0,0 +1,51 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_call_h +#define cglm_call_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "cglm.h" +#include "call/vec2.h" +#include "call/vec3.h" +#include "call/vec4.h" +#include "call/ivec2.h" +#include "call/ivec3.h" +#include "call/ivec4.h" +#include "call/mat2.h" +#include "call/mat2x3.h" +#include "call/mat2x4.h" +#include "call/mat3.h" +#include "call/mat3x2.h" +#include "call/mat3x4.h" +#include "call/mat4.h" +#include "call/mat4x2.h" +#include "call/mat4x3.h" +#include "call/affine.h" +#include "call/cam.h" +#include "call/quat.h" +#include "call/euler.h" +#include "call/plane.h" +#include "call/noise.h" +#include "call/frustum.h" +#include "call/aabb2d.h" +#include "call/box.h" +#include "call/io.h" +#include "call/project.h" +#include "call/sphere.h" +#include "call/ease.h" +#include "call/curve.h" +#include "call/bezier.h" +#include "call/ray.h" +#include "call/affine2d.h" + +#ifdef __cplusplus +} +#endif +#endif /* cglm_call_h */ diff --git a/cglm/include/cglm/call/aabb2d.h b/cglm/include/cglm/call/aabb2d.h new file mode 100644 index 0000000..e6f36a0 --- /dev/null +++ b/cglm/include/cglm/call/aabb2d.h @@ -0,0 +1,89 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_aabb2d_h +#define cglmc_aabb2d_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +/* DEPRECATED! use _diag */ +#define glmc_aabb2d_size(aabb) glmc_aabb2d_diag(aabb) + +CGLM_EXPORT +void +glmc_aabb2d_zero(vec2 aabb[2]); + +CGLM_EXPORT +void +glmc_aabb2d_copy(vec2 aabb[2], vec2 dest[2]); + +CGLM_EXPORT +void +glmc_aabb2d_transform(vec2 aabb[2], mat3 m, vec2 dest[2]); + +CGLM_EXPORT +void +glmc_aabb2d_merge(vec2 aabb1[2], vec2 aabb2[2], vec2 dest[2]); + +CGLM_EXPORT +void +glmc_aabb2d_crop(vec2 aabb[2], vec2 cropAabb[2], vec2 dest[2]); + +CGLM_EXPORT +void +glmc_aabb2d_crop_until(vec2 aabb[2], + vec2 cropAabb[2], + vec2 clampAabb[2], + vec2 dest[2]); + +CGLM_EXPORT +void +glmc_aabb2d_invalidate(vec2 aabb[2]); + +CGLM_EXPORT +bool +glmc_aabb2d_isvalid(vec2 aabb[2]); + +CGLM_EXPORT +float +glmc_aabb2d_diag(vec2 aabb[2]); + +CGLM_EXPORT +void +glmc_aabb2d_sizev(vec2 aabb[2], vec2 dest); + +CGLM_EXPORT +float +glmc_aabb2d_radius(vec2 aabb[2]); + +CGLM_EXPORT +void +glmc_aabb2d_center(vec2 aabb[2], vec2 dest); + +CGLM_EXPORT +bool +glmc_aabb2d_aabb(vec2 aabb[2], vec2 other[2]); + +CGLM_EXPORT +bool +glmc_aabb2d_point(vec2 aabb[2], vec2 point); + +CGLM_EXPORT +bool +glmc_aabb2d_contains(vec2 aabb[2], vec2 other[2]); + +CGLM_EXPORT +bool +glmc_aabb2d_circle(vec2 aabb[2], vec3 s); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_aabb2d_h */ diff --git a/cglm/include/cglm/call/affine.h b/cglm/include/cglm/call/affine.h new file mode 100644 index 0000000..52b8501 --- /dev/null +++ b/cglm/include/cglm/call/affine.h @@ -0,0 +1,167 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_affine_h +#define cglmc_affine_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_translate_make(mat4 m, vec3 v); + +CGLM_EXPORT +void +glmc_translate_to(mat4 m, vec3 v, mat4 dest); + +CGLM_EXPORT +void +glmc_translate(mat4 m, vec3 v); + +CGLM_EXPORT +void +glmc_translate_x(mat4 m, float to); + +CGLM_EXPORT +void +glmc_translate_y(mat4 m, float to); + +CGLM_EXPORT +void +glmc_translate_z(mat4 m, float to); + +CGLM_EXPORT +void +glmc_scale_make(mat4 m, vec3 v); + +CGLM_EXPORT +void +glmc_scale_to(mat4 m, vec3 v, mat4 dest); + +CGLM_EXPORT +void +glmc_scale(mat4 m, vec3 v); + +CGLM_EXPORT +void +glmc_scale_uni(mat4 m, float s); + +CGLM_EXPORT +void +glmc_rotate_x(mat4 m, float rad, mat4 dest); + +CGLM_EXPORT +void +glmc_rotate_y(mat4 m, float rad, mat4 dest); + +CGLM_EXPORT +void +glmc_rotate_z(mat4 m, float rad, mat4 dest); + +CGLM_EXPORT +void +glmc_rotate_make(mat4 m, float angle, vec3 axis); + +CGLM_EXPORT +void +glmc_rotate(mat4 m, float angle, vec3 axis); + +CGLM_EXPORT +void +glmc_rotate_at(mat4 m, vec3 pivot, float angle, vec3 axis); + +CGLM_EXPORT +void +glmc_rotate_atm(mat4 m, vec3 pivot, float angle, vec3 axis); + +CGLM_EXPORT +void +glmc_spin(mat4 m, float angle, vec3 axis); + +CGLM_EXPORT +void +glmc_decompose_scalev(mat4 m, vec3 s); + +CGLM_EXPORT +bool +glmc_uniscaled(mat4 m); + +CGLM_EXPORT +void +glmc_decompose_rs(mat4 m, mat4 r, vec3 s); + +CGLM_EXPORT +void +glmc_decompose(mat4 m, vec4 t, mat4 r, vec3 s); + +/* affine-post */ + +CGLM_EXPORT +void +glmc_translated(mat4 m, vec3 v); + +CGLM_EXPORT +void +glmc_translated_to(mat4 m, vec3 v, mat4 dest); + +CGLM_EXPORT +void +glmc_translated_x(mat4 m, float x); + +CGLM_EXPORT +void +glmc_translated_y(mat4 m, float y); + +CGLM_EXPORT +void +glmc_translated_z(mat4 m, float z); + +CGLM_EXPORT +void +glmc_rotated_x(mat4 m, float angle, mat4 dest); + +CGLM_EXPORT +void +glmc_rotated_y(mat4 m, float angle, mat4 dest); + +CGLM_EXPORT +void +glmc_rotated_z(mat4 m, float angle, mat4 dest); + +CGLM_EXPORT +void +glmc_rotated(mat4 m, float angle, vec3 axis); + +CGLM_EXPORT +void +glmc_rotated_at(mat4 m, vec3 pivot, float angle, vec3 axis); + +CGLM_EXPORT +void +glmc_spinned(mat4 m, float angle, vec3 axis); + +/* affine-mat */ + +CGLM_EXPORT +void +glmc_mul(mat4 m1, mat4 m2, mat4 dest); + +CGLM_EXPORT +void +glmc_mul_rot(mat4 m1, mat4 m2, mat4 dest); + +CGLM_EXPORT +void +glmc_inv_tr(mat4 mat); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_affine_h */ diff --git a/cglm/include/cglm/call/affine2d.h b/cglm/include/cglm/call/affine2d.h new file mode 100644 index 0000000..e1b9462 --- /dev/null +++ b/cglm/include/cglm/call/affine2d.h @@ -0,0 +1,67 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_affine2d_h +#define cglmc_affine2d_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_translate2d_make(mat3 m, vec2 v); + +CGLM_EXPORT +void +glmc_translate2d_to(mat3 m, vec2 v, mat3 dest); + +CGLM_EXPORT +void +glmc_translate2d(mat3 m, vec2 v); + +CGLM_EXPORT +void +glmc_translate2d_x(mat3 m, float to); + +CGLM_EXPORT +void +glmc_translate2d_y(mat3 m, float to); + +CGLM_EXPORT +void +glmc_scale2d_to(mat3 m, vec2 v, mat3 dest); + +CGLM_EXPORT +void +glmc_scale2d_make(mat3 m, vec2 v); + +CGLM_EXPORT +void +glmc_scale2d(mat3 m, vec2 v); + +CGLM_EXPORT +void +glmc_scale2d_uni(mat3 m, float s); + +CGLM_EXPORT +void +glmc_rotate2d_make(mat3 m, float angle); + +CGLM_EXPORT +void +glmc_rotate2d(mat3 m, float angle); + +CGLM_EXPORT +void +glmc_rotate2d_to(mat3 m, float angle, mat3 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_affine2d_h */ diff --git a/cglm/include/cglm/call/bezier.h b/cglm/include/cglm/call/bezier.h new file mode 100644 index 0000000..a6a0eb4 --- /dev/null +++ b/cglm/include/cglm/call/bezier.h @@ -0,0 +1,31 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_bezier_h +#define cglmc_bezier_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +float +glmc_bezier(float s, float p0, float c0, float c1, float p1); + +CGLM_EXPORT +float +glmc_hermite(float s, float p0, float t0, float t1, float p1); + +CGLM_EXPORT +float +glmc_decasteljau(float prm, float p0, float c0, float c1, float p1); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_bezier_h */ diff --git a/cglm/include/cglm/call/box.h b/cglm/include/cglm/call/box.h new file mode 100644 index 0000000..3617eed --- /dev/null +++ b/cglm/include/cglm/call/box.h @@ -0,0 +1,78 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_box_h +#define cglmc_box_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_aabb_transform(vec3 box[2], mat4 m, vec3 dest[2]); + +CGLM_EXPORT +void +glmc_aabb_merge(vec3 box1[2], vec3 box2[2], vec3 dest[2]); + +CGLM_EXPORT +void +glmc_aabb_crop(vec3 box[2], vec3 cropBox[2], vec3 dest[2]); + +CGLM_EXPORT +void +glmc_aabb_crop_until(vec3 box[2], + vec3 cropBox[2], + vec3 clampBox[2], + vec3 dest[2]); + +CGLM_EXPORT +bool +glmc_aabb_frustum(vec3 box[2], vec4 planes[6]); + +CGLM_EXPORT +void +glmc_aabb_invalidate(vec3 box[2]); + +CGLM_EXPORT +bool +glmc_aabb_isvalid(vec3 box[2]); + +CGLM_EXPORT +float +glmc_aabb_size(vec3 box[2]); + +CGLM_EXPORT +float +glmc_aabb_radius(vec3 box[2]); + +CGLM_EXPORT +void +glmc_aabb_center(vec3 box[2], vec3 dest); + +CGLM_EXPORT +bool +glmc_aabb_aabb(vec3 box[2], vec3 other[2]); + +CGLM_EXPORT +bool +glmc_aabb_point(vec3 box[2], vec3 point); + +CGLM_EXPORT +bool +glmc_aabb_contains(vec3 box[2], vec3 other[2]); + +CGLM_EXPORT +bool +glmc_aabb_sphere(vec3 box[2], vec4 s); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_box_h */ diff --git a/cglm/include/cglm/call/cam.h b/cglm/include/cglm/call/cam.h new file mode 100644 index 0000000..d9567ec --- /dev/null +++ b/cglm/include/cglm/call/cam.h @@ -0,0 +1,133 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_cam_h +#define cglmc_cam_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_frustum(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest); + +CGLM_EXPORT +void +glmc_ortho(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_aabb(vec3 box[2], mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_aabb_p(vec3 box[2], float padding, mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_aabb_pz(vec3 box[2], float padding, mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_default(float aspect, mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_default_s(float aspect, float size, mat4 dest); + +CGLM_EXPORT +void +glmc_perspective(float fovy, float aspect, float nearZ, float farZ, mat4 dest); + +CGLM_EXPORT +void +glmc_persp_move_far(mat4 proj, float deltaFar); + +CGLM_EXPORT +void +glmc_perspective_default(float aspect, mat4 dest); + +CGLM_EXPORT +void +glmc_perspective_resize(float aspect, mat4 proj); + +CGLM_EXPORT +void +glmc_lookat(vec3 eye, vec3 center, vec3 up, mat4 dest); + +CGLM_EXPORT +void +glmc_look(vec3 eye, vec3 dir, vec3 up, mat4 dest); + +CGLM_EXPORT +void +glmc_look_anyup(vec3 eye, vec3 dir, mat4 dest); + +CGLM_EXPORT +void +glmc_persp_decomp(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ, + float * __restrict top, + float * __restrict bottom, + float * __restrict left, + float * __restrict right); + +CGLM_EXPORT +void +glmc_persp_decompv(mat4 proj, float dest[6]); + +CGLM_EXPORT +void +glmc_persp_decomp_x(mat4 proj, + float * __restrict left, + float * __restrict right); + +CGLM_EXPORT +void +glmc_persp_decomp_y(mat4 proj, + float * __restrict top, + float * __restrict bottom); + +CGLM_EXPORT +void +glmc_persp_decomp_z(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ); + +CGLM_EXPORT +void +glmc_persp_decomp_far(mat4 proj, float * __restrict farZ); + +CGLM_EXPORT +void +glmc_persp_decomp_near(mat4 proj, float * __restrict nearZ); + +CGLM_EXPORT +float +glmc_persp_fovy(mat4 proj); + +CGLM_EXPORT +float +glmc_persp_aspect(mat4 proj); + +CGLM_EXPORT +void +glmc_persp_sizes(mat4 proj, float fovy, vec4 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_cam_h */ diff --git a/cglm/include/cglm/call/clipspace/ortho_lh_no.h b/cglm/include/cglm/call/clipspace/ortho_lh_no.h new file mode 100644 index 0000000..3e26fa9 --- /dev/null +++ b/cglm/include/cglm/call/clipspace/ortho_lh_no.h @@ -0,0 +1,46 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_ortho_lh_no_h +#define cglmc_ortho_lh_no_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../../cglm.h" + +CGLM_EXPORT +void +glmc_ortho_lh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_aabb_lh_no(vec3 box[2], mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_aabb_p_lh_no(vec3 box[2], float padding, mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_aabb_pz_lh_no(vec3 box[2], float padding, mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_default_lh_no(float aspect, mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_default_s_lh_no(float aspect, float size, mat4 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_ortho_lh_no_h */ diff --git a/cglm/include/cglm/call/clipspace/ortho_lh_zo.h b/cglm/include/cglm/call/clipspace/ortho_lh_zo.h new file mode 100644 index 0000000..dc4c610 --- /dev/null +++ b/cglm/include/cglm/call/clipspace/ortho_lh_zo.h @@ -0,0 +1,46 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_ortho_lh_zo_h +#define cglmc_ortho_lh_zo_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../../cglm.h" + +CGLM_EXPORT +void +glmc_ortho_lh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_aabb_lh_zo(vec3 box[2], mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_aabb_p_lh_zo(vec3 box[2], float padding, mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_aabb_pz_lh_zo(vec3 box[2], float padding, mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_default_lh_zo(float aspect, mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_default_s_lh_zo(float aspect, float size, mat4 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_ortho_lh_zo_h */ diff --git a/cglm/include/cglm/call/clipspace/ortho_rh_no.h b/cglm/include/cglm/call/clipspace/ortho_rh_no.h new file mode 100644 index 0000000..dbba497 --- /dev/null +++ b/cglm/include/cglm/call/clipspace/ortho_rh_no.h @@ -0,0 +1,46 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_ortho_rh_no_h +#define cglmc_ortho_rh_no_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../../cglm.h" + +CGLM_EXPORT +void +glmc_ortho_rh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_aabb_rh_no(vec3 box[2], mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_aabb_p_rh_no(vec3 box[2], float padding, mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_aabb_pz_rh_no(vec3 box[2], float padding, mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_default_rh_no(float aspect, mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_default_s_rh_no(float aspect, float size, mat4 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_ortho_rh_no_h */ diff --git a/cglm/include/cglm/call/clipspace/ortho_rh_zo.h b/cglm/include/cglm/call/clipspace/ortho_rh_zo.h new file mode 100644 index 0000000..e79ae83 --- /dev/null +++ b/cglm/include/cglm/call/clipspace/ortho_rh_zo.h @@ -0,0 +1,46 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_ortho_rh_zo_h +#define cglmc_ortho_rh_zo_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../../cglm.h" + +CGLM_EXPORT +void +glmc_ortho_rh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_aabb_rh_zo(vec3 box[2], mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_aabb_p_rh_zo(vec3 box[2], float padding, mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_aabb_pz_rh_zo(vec3 box[2], float padding, mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_default_rh_zo(float aspect, mat4 dest); + +CGLM_EXPORT +void +glmc_ortho_default_s_rh_zo(float aspect, float size, mat4 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_ortho_rh_zo_h */ diff --git a/cglm/include/cglm/call/clipspace/persp_lh_no.h b/cglm/include/cglm/call/clipspace/persp_lh_no.h new file mode 100644 index 0000000..4bdbcfe --- /dev/null +++ b/cglm/include/cglm/call/clipspace/persp_lh_no.h @@ -0,0 +1,87 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_persp_lh_no_h +#define cglmc_persp_lh_no_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../../cglm.h" + +CGLM_EXPORT +void +glmc_frustum_lh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest); + +CGLM_EXPORT +void +glmc_perspective_lh_no(float fovy, + float aspect, + float nearVal, + float farVal, + mat4 dest); + +CGLM_EXPORT +void +glmc_persp_move_far_lh_no(mat4 proj, float deltaFar); + +CGLM_EXPORT +void +glmc_persp_decomp_lh_no(mat4 proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right); + +CGLM_EXPORT +void +glmc_persp_decompv_lh_no(mat4 proj, float dest[6]); + +CGLM_EXPORT +void +glmc_persp_decomp_x_lh_no(mat4 proj, + float * __restrict left, + float * __restrict right); + +CGLM_EXPORT +void +glmc_persp_decomp_y_lh_no(mat4 proj, + float * __restrict top, + float * __restrict bottom); + +CGLM_EXPORT +void +glmc_persp_decomp_z_lh_no(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ); + +CGLM_EXPORT +void +glmc_persp_decomp_far_lh_no(mat4 proj, float * __restrict farZ); + +CGLM_EXPORT +void +glmc_persp_decomp_near_lh_no(mat4 proj, float * __restrict nearZ); + +CGLM_EXPORT +void +glmc_persp_sizes_lh_no(mat4 proj, float fovy, vec4 dest); + +CGLM_EXPORT +float +glmc_persp_fovy_lh_no(mat4 proj); + +CGLM_EXPORT +float +glmc_persp_aspect_lh_no(mat4 proj); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_persp_lh_no_h */ diff --git a/cglm/include/cglm/call/clipspace/persp_lh_zo.h b/cglm/include/cglm/call/clipspace/persp_lh_zo.h new file mode 100644 index 0000000..53c2c1c --- /dev/null +++ b/cglm/include/cglm/call/clipspace/persp_lh_zo.h @@ -0,0 +1,87 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_persp_lh_zo_h +#define cglmc_persp_lh_zo_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../../cglm.h" + +CGLM_EXPORT +void +glmc_frustum_lh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest); + +CGLM_EXPORT +void +glmc_perspective_lh_zo(float fovy, + float aspect, + float nearVal, + float farVal, + mat4 dest); + +CGLM_EXPORT +void +glmc_persp_move_far_lh_zo(mat4 proj, float deltaFar); + +CGLM_EXPORT +void +glmc_persp_decomp_lh_zo(mat4 proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right); + +CGLM_EXPORT +void +glmc_persp_decompv_lh_zo(mat4 proj, float dest[6]); + +CGLM_EXPORT +void +glmc_persp_decomp_x_lh_zo(mat4 proj, + float * __restrict left, + float * __restrict right); + +CGLM_EXPORT +void +glmc_persp_decomp_y_lh_zo(mat4 proj, + float * __restrict top, + float * __restrict bottom); + +CGLM_EXPORT +void +glmc_persp_decomp_z_lh_zo(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ); + +CGLM_EXPORT +void +glmc_persp_decomp_far_lh_zo(mat4 proj, float * __restrict farZ); + +CGLM_EXPORT +void +glmc_persp_decomp_near_lh_zo(mat4 proj, float * __restrict nearZ); + +CGLM_EXPORT +void +glmc_persp_sizes_lh_zo(mat4 proj, float fovy, vec4 dest); + +CGLM_EXPORT +float +glmc_persp_fovy_lh_zo(mat4 proj); + +CGLM_EXPORT +float +glmc_persp_aspect_lh_zo(mat4 proj); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_persp_lh_zo_h */ diff --git a/cglm/include/cglm/call/clipspace/persp_rh_no.h b/cglm/include/cglm/call/clipspace/persp_rh_no.h new file mode 100644 index 0000000..9c0d65d --- /dev/null +++ b/cglm/include/cglm/call/clipspace/persp_rh_no.h @@ -0,0 +1,87 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_persp_rh_no_h +#define cglmc_persp_rh_no_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../../cglm.h" + +CGLM_EXPORT +void +glmc_frustum_rh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest); + +CGLM_EXPORT +void +glmc_perspective_rh_no(float fovy, + float aspect, + float nearVal, + float farVal, + mat4 dest); + +CGLM_EXPORT +void +glmc_persp_move_far_rh_no(mat4 proj, float deltaFar); + +CGLM_EXPORT +void +glmc_persp_decomp_rh_no(mat4 proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right); + +CGLM_EXPORT +void +glmc_persp_decompv_rh_no(mat4 proj, float dest[6]); + +CGLM_EXPORT +void +glmc_persp_decomp_x_rh_no(mat4 proj, + float * __restrict left, + float * __restrict right); + +CGLM_EXPORT +void +glmc_persp_decomp_y_rh_no(mat4 proj, + float * __restrict top, + float * __restrict bottom); + +CGLM_EXPORT +void +glmc_persp_decomp_z_rh_no(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ); + +CGLM_EXPORT +void +glmc_persp_decomp_far_rh_no(mat4 proj, float * __restrict farZ); + +CGLM_EXPORT +void +glmc_persp_decomp_near_rh_no(mat4 proj, float * __restrict nearZ); + +CGLM_EXPORT +void +glmc_persp_sizes_rh_no(mat4 proj, float fovy, vec4 dest); + +CGLM_EXPORT +float +glmc_persp_fovy_rh_no(mat4 proj); + +CGLM_EXPORT +float +glmc_persp_aspect_rh_no(mat4 proj); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_persp_rh_no_h */ diff --git a/cglm/include/cglm/call/clipspace/persp_rh_zo.h b/cglm/include/cglm/call/clipspace/persp_rh_zo.h new file mode 100644 index 0000000..718d4ad --- /dev/null +++ b/cglm/include/cglm/call/clipspace/persp_rh_zo.h @@ -0,0 +1,87 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_persp_rh_zo_h +#define cglmc_persp_rh_zo_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../../cglm.h" + +CGLM_EXPORT +void +glmc_frustum_rh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest); + +CGLM_EXPORT +void +glmc_perspective_rh_zo(float fovy, + float aspect, + float nearVal, + float farVal, + mat4 dest); + +CGLM_EXPORT +void +glmc_persp_move_far_rh_zo(mat4 proj, float deltaFar); + +CGLM_EXPORT +void +glmc_persp_decomp_rh_zo(mat4 proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right); + +CGLM_EXPORT +void +glmc_persp_decompv_rh_zo(mat4 proj, float dest[6]); + +CGLM_EXPORT +void +glmc_persp_decomp_x_rh_zo(mat4 proj, + float * __restrict left, + float * __restrict right); + +CGLM_EXPORT +void +glmc_persp_decomp_y_rh_zo(mat4 proj, + float * __restrict top, + float * __restrict bottom); + +CGLM_EXPORT +void +glmc_persp_decomp_z_rh_zo(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ); + +CGLM_EXPORT +void +glmc_persp_decomp_far_rh_zo(mat4 proj, float * __restrict farZ); + +CGLM_EXPORT +void +glmc_persp_decomp_near_rh_zo(mat4 proj, float * __restrict nearZ); + +CGLM_EXPORT +void +glmc_persp_sizes_rh_zo(mat4 proj, float fovy, vec4 dest); + +CGLM_EXPORT +float +glmc_persp_fovy_rh_zo(mat4 proj); + +CGLM_EXPORT +float +glmc_persp_aspect_rh_zo(mat4 proj); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_persp_rh_zo_h */ diff --git a/cglm/include/cglm/call/clipspace/project_no.h b/cglm/include/cglm/call/clipspace/project_no.h new file mode 100644 index 0000000..3cba860 --- /dev/null +++ b/cglm/include/cglm/call/clipspace/project_no.h @@ -0,0 +1,31 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_project_no_h +#define cglmc_project_no_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../../cglm.h" + +CGLM_EXPORT +void +glmc_unprojecti_no(vec3 pos, mat4 invMat, vec4 vp, vec3 dest); + +CGLM_EXPORT +void +glmc_project_no(vec3 pos, mat4 m, vec4 vp, vec3 dest); + +CGLM_EXPORT +float +glmc_project_z_no(vec3 pos, mat4 m); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_project_no_h */ diff --git a/cglm/include/cglm/call/clipspace/project_zo.h b/cglm/include/cglm/call/clipspace/project_zo.h new file mode 100644 index 0000000..d2a6c62 --- /dev/null +++ b/cglm/include/cglm/call/clipspace/project_zo.h @@ -0,0 +1,31 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_project_zo_h +#define cglmc_project_zo_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../../cglm.h" + +CGLM_EXPORT +void +glmc_unprojecti_zo(vec3 pos, mat4 invMat, vec4 vp, vec3 dest); + +CGLM_EXPORT +void +glmc_project_zo(vec3 pos, mat4 m, vec4 vp, vec3 dest); + +CGLM_EXPORT +float +glmc_project_z_zo(vec3 pos, mat4 m); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_project_zo_h */ diff --git a/cglm/include/cglm/call/clipspace/view_lh_no.h b/cglm/include/cglm/call/clipspace/view_lh_no.h new file mode 100644 index 0000000..3b58c84 --- /dev/null +++ b/cglm/include/cglm/call/clipspace/view_lh_no.h @@ -0,0 +1,31 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_view_lh_no_h +#define cglmc_view_lh_no_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../../cglm.h" + +CGLM_EXPORT +void +glmc_lookat_lh_no(vec3 eye, vec3 center, vec3 up, mat4 dest); + +CGLM_EXPORT +void +glmc_look_lh_no(vec3 eye, vec3 dir, vec3 up, mat4 dest); + +CGLM_EXPORT +void +glmc_look_anyup_lh_no(vec3 eye, vec3 dir, mat4 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_view_lh_no_h */ diff --git a/cglm/include/cglm/call/clipspace/view_lh_zo.h b/cglm/include/cglm/call/clipspace/view_lh_zo.h new file mode 100644 index 0000000..c877367 --- /dev/null +++ b/cglm/include/cglm/call/clipspace/view_lh_zo.h @@ -0,0 +1,31 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_view_lh_zo_h +#define cglmc_view_lh_zo_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../../cglm.h" + +CGLM_EXPORT +void +glmc_lookat_lh_zo(vec3 eye, vec3 center, vec3 up, mat4 dest); + +CGLM_EXPORT +void +glmc_look_lh_zo(vec3 eye, vec3 dir, vec3 up, mat4 dest); + +CGLM_EXPORT +void +glmc_look_anyup_lh_zo(vec3 eye, vec3 dir, mat4 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_view_lh_zo_h */ diff --git a/cglm/include/cglm/call/clipspace/view_rh_no.h b/cglm/include/cglm/call/clipspace/view_rh_no.h new file mode 100644 index 0000000..6303dbf --- /dev/null +++ b/cglm/include/cglm/call/clipspace/view_rh_no.h @@ -0,0 +1,31 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_view_rh_no_h +#define cglmc_view_rh_no_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../../cglm.h" + +CGLM_EXPORT +void +glmc_lookat_rh_no(vec3 eye, vec3 center, vec3 up, mat4 dest); + +CGLM_EXPORT +void +glmc_look_rh_no(vec3 eye, vec3 dir, vec3 up, mat4 dest); + +CGLM_EXPORT +void +glmc_look_anyup_rh_no(vec3 eye, vec3 dir, mat4 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_view_rh_no_h */ diff --git a/cglm/include/cglm/call/clipspace/view_rh_zo.h b/cglm/include/cglm/call/clipspace/view_rh_zo.h new file mode 100644 index 0000000..00b8707 --- /dev/null +++ b/cglm/include/cglm/call/clipspace/view_rh_zo.h @@ -0,0 +1,31 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_view_rh_zo_h +#define cglmc_view_rh_zo_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../../cglm.h" + +CGLM_EXPORT +void +glmc_lookat_rh_zo(vec3 eye, vec3 center, vec3 up, mat4 dest); + +CGLM_EXPORT +void +glmc_look_rh_zo(vec3 eye, vec3 dir, vec3 up, mat4 dest); + +CGLM_EXPORT +void +glmc_look_anyup_rh_zo(vec3 eye, vec3 dir, mat4 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_view_rh_zo_h */ diff --git a/cglm/include/cglm/call/curve.h b/cglm/include/cglm/call/curve.h new file mode 100644 index 0000000..061fdb9 --- /dev/null +++ b/cglm/include/cglm/call/curve.h @@ -0,0 +1,23 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_curve_h +#define cglmc_curve_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +float +glmc_smc(float s, mat4 m, vec4 c); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_curve_h */ diff --git a/cglm/include/cglm/call/ease.h b/cglm/include/cglm/call/ease.h new file mode 100644 index 0000000..87e39ca --- /dev/null +++ b/cglm/include/cglm/call/ease.h @@ -0,0 +1,143 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_ease_h +#define cglmc_ease_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +float +glmc_ease_linear(float t); + +CGLM_EXPORT +float +glmc_ease_sine_in(float t); + +CGLM_EXPORT +float +glmc_ease_sine_out(float t); + +CGLM_EXPORT +float +glmc_ease_sine_inout(float t); + +CGLM_EXPORT +float +glmc_ease_quad_in(float t); + +CGLM_EXPORT +float +glmc_ease_quad_out(float t); + +CGLM_EXPORT +float +glmc_ease_quad_inout(float t); + +CGLM_EXPORT +float +glmc_ease_cubic_in(float t); + +CGLM_EXPORT +float +glmc_ease_cubic_out(float t); + +CGLM_EXPORT +float +glmc_ease_cubic_inout(float t); + +CGLM_EXPORT +float +glmc_ease_quart_in(float t); + +CGLM_EXPORT +float +glmc_ease_quart_out(float t); + +CGLM_EXPORT +float +glmc_ease_quart_inout(float t); + +CGLM_EXPORT +float +glmc_ease_quint_in(float t); + +CGLM_EXPORT +float +glmc_ease_quint_out(float t); + +CGLM_EXPORT +float +glmc_ease_quint_inout(float t); + +CGLM_EXPORT +float +glmc_ease_exp_in(float t); + +CGLM_EXPORT +float +glmc_ease_exp_out(float t); + +CGLM_EXPORT +float +glmc_ease_exp_inout(float t); + +CGLM_EXPORT +float +glmc_ease_circ_in(float t); + +CGLM_EXPORT +float +glmc_ease_circ_out(float t); + +CGLM_EXPORT +float +glmc_ease_circ_inout(float t); + +CGLM_EXPORT +float +glmc_ease_back_in(float t); + +CGLM_EXPORT +float +glmc_ease_back_out(float t); + +CGLM_EXPORT +float +glmc_ease_back_inout(float t); + +CGLM_EXPORT +float +glmc_ease_elast_in(float t); + +CGLM_EXPORT +float +glmc_ease_elast_out(float t); + +CGLM_EXPORT +float +glmc_ease_elast_inout(float t); + +CGLM_EXPORT +float +glmc_ease_bounce_out(float t); + +CGLM_EXPORT +float +glmc_ease_bounce_in(float t); + +CGLM_EXPORT +float +glmc_ease_bounce_inout(float t); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_ease_h */ diff --git a/cglm/include/cglm/call/euler.h b/cglm/include/cglm/call/euler.h new file mode 100644 index 0000000..182bcbb --- /dev/null +++ b/cglm/include/cglm/call/euler.h @@ -0,0 +1,80 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_euler_h +#define cglmc_euler_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_euler_angles(mat4 m, vec3 dest); + +CGLM_EXPORT +void +glmc_euler(vec3 angles, mat4 dest); + +CGLM_EXPORT +void +glmc_euler_xyz(vec3 angles, mat4 dest); + +CGLM_EXPORT +void +glmc_euler_zyx(vec3 angles, mat4 dest); + +CGLM_EXPORT +void +glmc_euler_zxy(vec3 angles, mat4 dest); + +CGLM_EXPORT +void +glmc_euler_xzy(vec3 angles, mat4 dest); + +CGLM_EXPORT +void +glmc_euler_yzx(vec3 angles, mat4 dest); + +CGLM_EXPORT +void +glmc_euler_yxz(vec3 angles, mat4 dest); + +CGLM_EXPORT +void +glmc_euler_by_order(vec3 angles, glm_euler_seq axis, mat4 dest); + +CGLM_EXPORT +void +glmc_euler_xyz_quat(vec3 angles, versor dest); + +CGLM_EXPORT +void +glmc_euler_xzy_quat(vec3 angles, versor dest); + +CGLM_EXPORT +void +glmc_euler_yxz_quat(vec3 angles, versor dest); + +CGLM_EXPORT +void +glmc_euler_yzx_quat(vec3 angles, versor dest); + +CGLM_EXPORT +void +glmc_euler_zxy_quat(vec3 angles, versor dest); + +CGLM_EXPORT +void +glmc_euler_zyx_quat(vec3 angles, versor dest); + + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_euler_h */ diff --git a/cglm/include/cglm/call/frustum.h b/cglm/include/cglm/call/frustum.h new file mode 100644 index 0000000..6b4facb --- /dev/null +++ b/cglm/include/cglm/call/frustum.h @@ -0,0 +1,41 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_frustum_h +#define cglmc_frustum_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_frustum_planes(mat4 m, vec4 dest[6]); + +CGLM_EXPORT +void +glmc_frustum_corners(mat4 invMat, vec4 dest[8]); + +CGLM_EXPORT +void +glmc_frustum_center(vec4 corners[8], vec4 dest); + +CGLM_EXPORT +void +glmc_frustum_box(vec4 corners[8], mat4 m, vec3 box[2]); + +CGLM_EXPORT +void +glmc_frustum_corners_at(vec4 corners[8], + float splitDist, + float farDist, + vec4 planeCorners[4]); +#ifdef __cplusplus +} +#endif +#endif /* cglmc_frustum_h */ diff --git a/cglm/include/cglm/call/io.h b/cglm/include/cglm/call/io.h new file mode 100644 index 0000000..19ea06f --- /dev/null +++ b/cglm/include/cglm/call/io.h @@ -0,0 +1,45 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_io_h +#define cglmc_io_h + +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_mat4_print(mat4 matrix, + FILE * __restrict ostream); + +CGLM_EXPORT +void +glmc_mat3_print(mat3 matrix, + FILE * __restrict ostream); + +CGLM_EXPORT +void +glmc_vec4_print(vec4 vec, + FILE * __restrict ostream); + +CGLM_EXPORT +void +glmc_vec3_print(vec3 vec, + FILE * __restrict ostream); + +CGLM_EXPORT +void +glmc_versor_print(versor vec, + FILE * __restrict ostream); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_io_h */ diff --git a/cglm/include/cglm/call/ivec2.h b/cglm/include/cglm/call/ivec2.h new file mode 100644 index 0000000..82f70eb --- /dev/null +++ b/cglm/include/cglm/call/ivec2.h @@ -0,0 +1,179 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_ivec2_h +#define cglmc_ivec2_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_ivec2(int * __restrict v, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_copy(ivec2 a, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_zero(ivec2 v); + +CGLM_EXPORT +void +glmc_ivec2_one(ivec2 v); + +CGLM_EXPORT +int +glmc_ivec2_dot(ivec2 a, ivec2 b); + +CGLM_EXPORT +int +glmc_ivec2_cross(ivec2 a, ivec2 b); + +CGLM_EXPORT +void +glmc_ivec2_add(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_adds(ivec2 v, int s, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_sub(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_subs(ivec2 v, int s, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_mul(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_scale(ivec2 v, int s, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_div(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_divs(ivec2 v, int s, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_mod(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_addadd(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_addadds(ivec2 a, int s, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_subadd(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_subadds(ivec2 a, int s, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_muladd(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_muladds(ivec2 a, int s, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_maxadd(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_minadd(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_subsub(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_subsubs(ivec2 a, int s, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_addsub(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_addsubs(ivec2 a, int s, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_mulsub(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_mulsubs(ivec2 a, int s, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_maxsub(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_minsub(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +int +glmc_ivec2_distance2(ivec2 a, ivec2 b); + +CGLM_EXPORT +float +glmc_ivec2_distance(ivec2 a, ivec2 b); + +CGLM_EXPORT +void +glmc_ivec2_fill(ivec2 v, int val); + +CGLM_EXPORT +bool +glmc_ivec2_eq(ivec2 v, int val); + +CGLM_EXPORT +bool +glmc_ivec2_eqv(ivec2 a, ivec2 b); + +CGLM_EXPORT +void +glmc_ivec2_maxv(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_minv(ivec2 a, ivec2 b, ivec2 dest); + +CGLM_EXPORT +void +glmc_ivec2_clamp(ivec2 v, int minVal, int maxVal); + +CGLM_EXPORT +void +glmc_ivec2_abs(ivec2 v, ivec2 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_ivec2_h */ diff --git a/cglm/include/cglm/call/ivec3.h b/cglm/include/cglm/call/ivec3.h new file mode 100644 index 0000000..a6cec53 --- /dev/null +++ b/cglm/include/cglm/call/ivec3.h @@ -0,0 +1,183 @@ +/* + * Copyright (c);, Recep Aslantas. + * + * MIT License (MIT);, http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_ivec3_h +#define cglmc_ivec3_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_ivec3(ivec4 v4, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_copy(ivec3 a, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_zero(ivec3 v); + +CGLM_EXPORT +void +glmc_ivec3_one(ivec3 v); + +CGLM_EXPORT +int +glmc_ivec3_dot(ivec3 a, ivec3 b); + +CGLM_EXPORT +int +glmc_ivec3_norm2(ivec3 v); + +CGLM_EXPORT +int +glmc_ivec3_norm(ivec3 v); + +CGLM_EXPORT +void +glmc_ivec3_add(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_adds(ivec3 v, int s, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_sub(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_subs(ivec3 v, int s, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_mul(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_scale(ivec3 v, int s, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_div(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_divs(ivec3 v, int s, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_mod(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_addadd(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_addadds(ivec3 a, int s, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_subadd(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_subadds(ivec3 a, int s, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_muladd(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_muladds(ivec3 a, int s, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_maxadd(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_minadd(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_subsub(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_subsubs(ivec3 a, int s, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_addsub(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_addsubs(ivec3 a, int s, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_mulsub(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_mulsubs(ivec3 a, int s, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_maxsub(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_minsub(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +int +glmc_ivec3_distance2(ivec3 a, ivec3 b); + +CGLM_EXPORT +float +glmc_ivec3_distance(ivec3 a, ivec3 b); + +CGLM_EXPORT +void +glmc_ivec3_fill(ivec3 v, int val); + +CGLM_EXPORT +bool +glmc_ivec3_eq(ivec3 v, int val); + +CGLM_EXPORT +bool +glmc_ivec3_eqv(ivec3 a, ivec3 b); + +CGLM_EXPORT +void +glmc_ivec3_maxv(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_minv(ivec3 a, ivec3 b, ivec3 dest); + +CGLM_EXPORT +void +glmc_ivec3_clamp(ivec3 v, int minVal, int maxVal); + +CGLM_EXPORT +void +glmc_ivec3_abs(ivec3 v, ivec3 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_ivec3_h */ diff --git a/cglm/include/cglm/call/ivec4.h b/cglm/include/cglm/call/ivec4.h new file mode 100644 index 0000000..0e6d721 --- /dev/null +++ b/cglm/include/cglm/call/ivec4.h @@ -0,0 +1,147 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_ivec4_h +#define cglmc_ivec4_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_ivec4(ivec3 v3, int last, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_copy(ivec4 a, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_zero(ivec4 v); + +CGLM_EXPORT +void +glmc_ivec4_one(ivec4 v); + +CGLM_EXPORT +void +glmc_ivec4_add(ivec4 a, ivec4 b, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_adds(ivec4 v, int s, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_sub(ivec4 a, ivec4 b, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_subs(ivec4 v, int s, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_mul(ivec4 a, ivec4 b, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_scale(ivec4 v, int s, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_addadd(ivec4 a, ivec4 b, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_addadds(ivec4 a, int s, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_subadd(ivec4 a, ivec4 b, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_subadds(ivec4 a, int s, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_muladd(ivec4 a, ivec4 b, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_muladds(ivec4 a, int s, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_maxadd(ivec4 a, ivec4 b, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_minadd(ivec4 a, ivec4 b, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_subsub(ivec4 a, ivec4 b, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_subsubs(ivec4 a, int s, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_addsub(ivec4 a, ivec4 b, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_addsubs(ivec4 a, int s, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_mulsub(ivec4 a, ivec4 b, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_mulsubs(ivec4 a, int s, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_maxsub(ivec4 a, ivec4 b, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_minsub(ivec4 a, ivec4 b, ivec4 dest); + +CGLM_EXPORT +int +glmc_ivec4_distance2(ivec4 a, ivec4 b); + +CGLM_EXPORT +float +glmc_ivec4_distance(ivec4 a, ivec4 b); + +CGLM_EXPORT +void +glmc_ivec4_maxv(ivec4 a, ivec4 b, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_minv(ivec4 a, ivec4 b, ivec4 dest); + +CGLM_EXPORT +void +glmc_ivec4_clamp(ivec4 v, int minVal, int maxVal); + +CGLM_EXPORT +void +glmc_ivec4_abs(ivec4 v, ivec4 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_ivec4_h */ diff --git a/cglm/include/cglm/call/mat2.h b/cglm/include/cglm/call/mat2.h new file mode 100644 index 0000000..598b5c8 --- /dev/null +++ b/cglm/include/cglm/call/mat2.h @@ -0,0 +1,83 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_mat2_h +#define cglmc_mat2_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_mat2_copy(mat2 mat, mat2 dest); + +CGLM_EXPORT +void +glmc_mat2_identity(mat2 mat); + +CGLM_EXPORT +void +glmc_mat2_identity_array(mat2 * __restrict mat, size_t count); + +CGLM_EXPORT +void +glmc_mat2_zero(mat2 mat); + +CGLM_EXPORT +void +glmc_mat2_mul(mat2 m1, mat2 m2, mat2 dest); + +CGLM_EXPORT +void +glmc_mat2_transpose_to(mat2 m, mat2 dest); + +CGLM_EXPORT +void +glmc_mat2_transpose(mat2 m); + +CGLM_EXPORT +void +glmc_mat2_mulv(mat2 m, vec2 v, vec2 dest); + +CGLM_EXPORT +float +glmc_mat2_trace(mat2 m); + +CGLM_EXPORT +void +glmc_mat2_scale(mat2 m, float s); + +CGLM_EXPORT +float +glmc_mat2_det(mat2 mat); + +CGLM_EXPORT +void +glmc_mat2_inv(mat2 mat, mat2 dest); + +CGLM_EXPORT +void +glmc_mat2_swap_col(mat2 mat, int col1, int col2); + +CGLM_EXPORT +void +glmc_mat2_swap_row(mat2 mat, int row1, int row2); + +CGLM_EXPORT +float +glmc_mat2_rmc(vec2 r, mat2 m, vec2 c); + +CGLM_EXPORT +void +glmc_mat2_make(const float * __restrict src, mat2 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_mat2_h */ diff --git a/cglm/include/cglm/call/mat2x3.h b/cglm/include/cglm/call/mat2x3.h new file mode 100644 index 0000000..59c9ee3 --- /dev/null +++ b/cglm/include/cglm/call/mat2x3.h @@ -0,0 +1,47 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_mat2x3_h +#define cglmc_mat2x3_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_mat2x3_copy(mat2x3 mat, mat2x3 dest); + +CGLM_EXPORT +void +glmc_mat2x3_zero(mat2x3 mat); + +CGLM_EXPORT +void +glmc_mat2x3_make(const float * __restrict src, mat2x3 dest); + +CGLM_EXPORT +void +glmc_mat2x3_mul(mat2x3 m1, mat3x2 m2, mat3 dest); + +CGLM_EXPORT +void +glmc_mat2x3_mulv(mat2x3 m, vec2 v, vec3 dest); + +CGLM_EXPORT +void +glmc_mat2x3_transpose(mat2x3 m, mat3x2 dest); + +CGLM_EXPORT +void +glmc_mat2x3_scale(mat2x3 m, float s); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_mat2x3_h */ diff --git a/cglm/include/cglm/call/mat2x4.h b/cglm/include/cglm/call/mat2x4.h new file mode 100644 index 0000000..84e805b --- /dev/null +++ b/cglm/include/cglm/call/mat2x4.h @@ -0,0 +1,47 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_mat2x4_h +#define cglmc_mat2x4_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_mat2x4_copy(mat2x4 mat, mat2x4 dest); + +CGLM_EXPORT +void +glmc_mat2x4_zero(mat2x4 mat); + +CGLM_EXPORT +void +glmc_mat2x4_make(const float * __restrict src, mat2x4 dest); + +CGLM_EXPORT +void +glmc_mat2x4_mul(mat2x4 m1, mat4x2 m2, mat4 dest); + +CGLM_EXPORT +void +glmc_mat2x4_mulv(mat2x4 m, vec2 v, vec4 dest); + +CGLM_EXPORT +void +glmc_mat2x4_transpose(mat2x4 m, mat4x2 dest); + +CGLM_EXPORT +void +glmc_mat2x4_scale(mat2x4 m, float s); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_mat2x4_h */ diff --git a/cglm/include/cglm/call/mat3.h b/cglm/include/cglm/call/mat3.h new file mode 100644 index 0000000..3659b6f --- /dev/null +++ b/cglm/include/cglm/call/mat3.h @@ -0,0 +1,90 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_mat3_h +#define cglmc_mat3_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +/* DEPRECATED! use _copy, _ucopy versions */ +#define glmc_mat3_dup(mat, dest) glmc_mat3_copy(mat, dest) + +CGLM_EXPORT +void +glmc_mat3_copy(mat3 mat, mat3 dest); + +CGLM_EXPORT +void +glmc_mat3_identity(mat3 mat); + +CGLM_EXPORT +void +glmc_mat3_zero(mat3 mat); + +CGLM_EXPORT +void +glmc_mat3_identity_array(mat3 * __restrict mat, size_t count); + +CGLM_EXPORT +void +glmc_mat3_mul(mat3 m1, mat3 m2, mat3 dest); + +CGLM_EXPORT +void +glmc_mat3_transpose_to(mat3 m, mat3 dest); + +CGLM_EXPORT +void +glmc_mat3_transpose(mat3 m); + +CGLM_EXPORT +void +glmc_mat3_mulv(mat3 m, vec3 v, vec3 dest); + +CGLM_EXPORT +float +glmc_mat3_trace(mat3 m); + +CGLM_EXPORT +void +glmc_mat3_quat(mat3 m, versor dest); + +CGLM_EXPORT +void +glmc_mat3_scale(mat3 m, float s); + +CGLM_EXPORT +float +glmc_mat3_det(mat3 mat); + +CGLM_EXPORT +void +glmc_mat3_inv(mat3 mat, mat3 dest); + +CGLM_EXPORT +void +glmc_mat3_swap_col(mat3 mat, int col1, int col2); + +CGLM_EXPORT +void +glmc_mat3_swap_row(mat3 mat, int row1, int row2); + +CGLM_EXPORT +float +glmc_mat3_rmc(vec3 r, mat3 m, vec3 c); + +CGLM_EXPORT +void +glmc_mat3_make(const float * __restrict src, mat3 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_mat3_h */ diff --git a/cglm/include/cglm/call/mat3x2.h b/cglm/include/cglm/call/mat3x2.h new file mode 100644 index 0000000..3c0daed --- /dev/null +++ b/cglm/include/cglm/call/mat3x2.h @@ -0,0 +1,47 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_mat3x2_h +#define cglmc_mat3x2_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_mat3x2_copy(mat3x2 mat, mat3x2 dest); + +CGLM_EXPORT +void +glmc_mat3x2_zero(mat3x2 mat); + +CGLM_EXPORT +void +glmc_mat3x2_make(const float * __restrict src, mat3x2 dest); + +CGLM_EXPORT +void +glmc_mat3x2_mul(mat3x2 m1, mat2x3 m2, mat2 dest); + +CGLM_EXPORT +void +glmc_mat3x2_mulv(mat3x2 m, vec3 v, vec2 dest); + +CGLM_EXPORT +void +glmc_mat3x2_transpose(mat3x2 m, mat2x3 dest); + +CGLM_EXPORT +void +glmc_mat3x2_scale(mat3x2 m, float s); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_mat3x2_h */ diff --git a/cglm/include/cglm/call/mat3x4.h b/cglm/include/cglm/call/mat3x4.h new file mode 100644 index 0000000..ff23b4c --- /dev/null +++ b/cglm/include/cglm/call/mat3x4.h @@ -0,0 +1,47 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_mat3x4_h +#define cglmc_mat3x4_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_mat3x4_copy(mat3x4 mat, mat3x4 dest); + +CGLM_EXPORT +void +glmc_mat3x4_zero(mat3x4 mat); + +CGLM_EXPORT +void +glmc_mat3x4_make(const float * __restrict src, mat3x4 dest); + +CGLM_EXPORT +void +glmc_mat3x4_mul(mat3x4 m1, mat4x3 m2, mat4 dest); + +CGLM_EXPORT +void +glmc_mat3x4_mulv(mat3x4 m, vec3 v, vec4 dest); + +CGLM_EXPORT +void +glmc_mat3x4_transpose(mat3x4 m, mat4x3 dest); + +CGLM_EXPORT +void +glmc_mat3x4_scale(mat3x4 m, float s); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_mat3x4_h */ diff --git a/cglm/include/cglm/call/mat4.h b/cglm/include/cglm/call/mat4.h new file mode 100644 index 0000000..5a33f72 --- /dev/null +++ b/cglm/include/cglm/call/mat4.h @@ -0,0 +1,131 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_mat_h +#define cglmc_mat_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +/* DEPRECATED! use _copy, _ucopy versions */ +#define glmc_mat4_udup(mat, dest) glmc_mat4_ucopy(mat, dest) +#define glmc_mat4_dup(mat, dest) glmc_mat4_copy(mat, dest) + +CGLM_EXPORT +void +glmc_mat4_ucopy(mat4 mat, mat4 dest); + +CGLM_EXPORT +void +glmc_mat4_copy(mat4 mat, mat4 dest); + +CGLM_EXPORT +void +glmc_mat4_identity(mat4 mat); + +CGLM_EXPORT +void +glmc_mat4_identity_array(mat4 * __restrict mat, size_t count); + +CGLM_EXPORT +void +glmc_mat4_zero(mat4 mat); + +CGLM_EXPORT +void +glmc_mat4_pick3(mat4 mat, mat3 dest); + +CGLM_EXPORT +void +glmc_mat4_pick3t(mat4 mat, mat3 dest); + +CGLM_EXPORT +void +glmc_mat4_ins3(mat3 mat, mat4 dest); + +CGLM_EXPORT +void +glmc_mat4_mul(mat4 m1, mat4 m2, mat4 dest); + +CGLM_EXPORT +void +glmc_mat4_mulN(mat4 * __restrict matrices[], uint32_t len, mat4 dest); + +CGLM_EXPORT +void +glmc_mat4_mulv(mat4 m, vec4 v, vec4 dest); + +CGLM_EXPORT +void +glmc_mat4_mulv3(mat4 m, vec3 v, float last, vec3 dest); + +CGLM_EXPORT +float +glmc_mat4_trace(mat4 m); + +CGLM_EXPORT +float +glmc_mat4_trace3(mat4 m); + +CGLM_EXPORT +void +glmc_mat4_quat(mat4 m, versor dest); + +CGLM_EXPORT +void +glmc_mat4_transpose_to(mat4 m, mat4 dest); + +CGLM_EXPORT +void +glmc_mat4_transpose(mat4 m); + +CGLM_EXPORT +void +glmc_mat4_scale_p(mat4 m, float s); + +CGLM_EXPORT +void +glmc_mat4_scale(mat4 m, float s); + +CGLM_EXPORT +float +glmc_mat4_det(mat4 mat); + +CGLM_EXPORT +void +glmc_mat4_inv(mat4 mat, mat4 dest); + +CGLM_EXPORT +void +glmc_mat4_inv_precise(mat4 mat, mat4 dest); + +CGLM_EXPORT +void +glmc_mat4_inv_fast(mat4 mat, mat4 dest); + +CGLM_EXPORT +void +glmc_mat4_swap_col(mat4 mat, int col1, int col2); + +CGLM_EXPORT +void +glmc_mat4_swap_row(mat4 mat, int row1, int row2); + +CGLM_EXPORT +float +glmc_mat4_rmc(vec4 r, mat4 m, vec4 c); + +CGLM_EXPORT +void +glmc_mat4_make(const float * __restrict src, mat4 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_mat_h */ diff --git a/cglm/include/cglm/call/mat4x2.h b/cglm/include/cglm/call/mat4x2.h new file mode 100644 index 0000000..2989124 --- /dev/null +++ b/cglm/include/cglm/call/mat4x2.h @@ -0,0 +1,47 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_mat4x2_h +#define cglmc_mat4x2_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_mat4x2_copy(mat4x2 mat, mat4x2 dest); + +CGLM_EXPORT +void +glmc_mat4x2_zero(mat4x2 mat); + +CGLM_EXPORT +void +glmc_mat4x2_make(const float * __restrict src, mat4x2 dest); + +CGLM_EXPORT +void +glmc_mat4x2_mul(mat4x2 m1, mat2x4 m2, mat2 dest); + +CGLM_EXPORT +void +glmc_mat4x2_mulv(mat4x2 m, vec4 v, vec2 dest); + +CGLM_EXPORT +void +glmc_mat4x2_transpose(mat4x2 m, mat2x4 dest); + +CGLM_EXPORT +void +glmc_mat4x2_scale(mat4x2 m, float s); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_mat4x2_h */ diff --git a/cglm/include/cglm/call/mat4x3.h b/cglm/include/cglm/call/mat4x3.h new file mode 100644 index 0000000..bb5c0a8 --- /dev/null +++ b/cglm/include/cglm/call/mat4x3.h @@ -0,0 +1,47 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_mat4x3_h +#define cglmc_mat4x3_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_mat4x3_copy(mat4x3 mat, mat4x3 dest); + +CGLM_EXPORT +void +glmc_mat4x3_zero(mat4x3 mat); + +CGLM_EXPORT +void +glmc_mat4x3_make(const float * __restrict src, mat4x3 dest); + +CGLM_EXPORT +void +glmc_mat4x3_mul(mat4x3 m1, mat3x4 m2, mat3 dest); + +CGLM_EXPORT +void +glmc_mat4x3_mulv(mat4x3 m, vec4 v, vec3 dest); + +CGLM_EXPORT +void +glmc_mat4x3_transpose(mat4x3 m, mat3x4 dest); + +CGLM_EXPORT +void +glmc_mat4x3_scale(mat4x3 m, float s); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_mat4x3_h */ diff --git a/cglm/include/cglm/call/noise.h b/cglm/include/cglm/call/noise.h new file mode 100644 index 0000000..6020c89 --- /dev/null +++ b/cglm/include/cglm/call/noise.h @@ -0,0 +1,31 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_noise_h +#define cglmc_noise_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +float +glmc_perlin_vec4(vec4 point); + +CGLM_EXPORT +float +glmc_perlin_vec3(vec3 point); + +CGLM_EXPORT +float +glmc_perlin_vec2(vec2 point); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_noise_h */ diff --git a/cglm/include/cglm/call/plane.h b/cglm/include/cglm/call/plane.h new file mode 100644 index 0000000..f991121 --- /dev/null +++ b/cglm/include/cglm/call/plane.h @@ -0,0 +1,23 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_plane_h +#define cglmc_plane_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_plane_normalize(vec4 plane); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_plane_h */ diff --git a/cglm/include/cglm/call/project.h b/cglm/include/cglm/call/project.h new file mode 100644 index 0000000..8fa7172 --- /dev/null +++ b/cglm/include/cglm/call/project.h @@ -0,0 +1,39 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_project_h +#define cglmc_project_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_unprojecti(vec3 pos, mat4 invMat, vec4 vp, vec3 dest); + +CGLM_EXPORT +void +glmc_unproject(vec3 pos, mat4 m, vec4 vp, vec3 dest); + +CGLM_EXPORT +void +glmc_project(vec3 pos, mat4 m, vec4 vp, vec3 dest); + +CGLM_EXPORT +float +glmc_project_z(vec3 pos, mat4 m); + +CGLM_EXPORT +void +glmc_pickmatrix(vec2 center, vec2 size, vec4 vp, mat4 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_project_h */ diff --git a/cglm/include/cglm/call/quat.h b/cglm/include/cglm/call/quat.h new file mode 100644 index 0000000..4244d36 --- /dev/null +++ b/cglm/include/cglm/call/quat.h @@ -0,0 +1,175 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_quat_h +#define cglmc_quat_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_quat_identity(versor q); + +CGLM_EXPORT +void +glmc_quat_identity_array(versor * __restrict q, size_t count); + +CGLM_EXPORT +void +glmc_quat_init(versor q, float x, float y, float z, float w); + +CGLM_EXPORT +void +glmc_quat(versor q, float angle, float x, float y, float z); + +CGLM_EXPORT +void +glmc_quatv(versor q, float angle, vec3 axis); + +CGLM_EXPORT +void +glmc_quat_copy(versor q, versor dest); + +CGLM_EXPORT +void +glmc_quat_from_vecs(vec3 a, vec3 b, versor dest); + +CGLM_EXPORT +float +glmc_quat_norm(versor q); + +CGLM_EXPORT +void +glmc_quat_normalize_to(versor q, versor dest); + +CGLM_EXPORT +void +glmc_quat_normalize(versor q); + +CGLM_EXPORT +float +glmc_quat_dot(versor p, versor q); + +CGLM_EXPORT +void +glmc_quat_conjugate(versor q, versor dest); + +CGLM_EXPORT +void +glmc_quat_inv(versor q, versor dest); + +CGLM_EXPORT +void +glmc_quat_add(versor p, versor q, versor dest); + +CGLM_EXPORT +void +glmc_quat_sub(versor p, versor q, versor dest); + +CGLM_EXPORT +float +glmc_quat_real(versor q); + +CGLM_EXPORT +void +glmc_quat_imag(versor q, vec3 dest); + +CGLM_EXPORT +void +glmc_quat_imagn(versor q, vec3 dest); + +CGLM_EXPORT +float +glmc_quat_imaglen(versor q); + +CGLM_EXPORT +float +glmc_quat_angle(versor q); + +CGLM_EXPORT +void +glmc_quat_axis(versor q, vec3 dest); + +CGLM_EXPORT +void +glmc_quat_mul(versor p, versor q, versor dest); + +CGLM_EXPORT +void +glmc_quat_mat4(versor q, mat4 dest); + +CGLM_EXPORT +void +glmc_quat_mat4t(versor q, mat4 dest); + +CGLM_EXPORT +void +glmc_quat_mat3(versor q, mat3 dest); + +CGLM_EXPORT +void +glmc_quat_mat3t(versor q, mat3 dest); + +CGLM_EXPORT +void +glmc_quat_lerp(versor from, versor to, float t, versor dest); + +CGLM_EXPORT +void +glmc_quat_lerpc(versor from, versor to, float t, versor dest); + +CGLM_EXPORT +void +glmc_quat_nlerp(versor q, versor r, float t, versor dest); + +CGLM_EXPORT +void +glmc_quat_slerp(versor q, versor r, float t, versor dest); + +CGLM_EXPORT +void +glmc_quat_slerp_longest(versor q, versor r, float t, versor dest); + +CGLM_EXPORT +void +glmc_quat_look(vec3 eye, versor ori, mat4 dest); + +CGLM_EXPORT +void +glmc_quat_for(vec3 dir, vec3 up, versor dest); + +CGLM_EXPORT +void +glmc_quat_forp(vec3 from, vec3 to, vec3 up, versor dest); + +CGLM_EXPORT +void +glmc_quat_rotatev(versor from, vec3 to, vec3 dest); + +CGLM_EXPORT +void +glmc_quat_rotate(mat4 m, versor q, mat4 dest); + +CGLM_EXPORT +void +glmc_quat_rotate_at(mat4 model, versor q, vec3 pivot); + +CGLM_EXPORT +void +glmc_quat_rotate_atm(mat4 m, versor q, vec3 pivot); + +CGLM_EXPORT +void +glmc_quat_make(const float * __restrict src, versor dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_quat_h */ diff --git a/cglm/include/cglm/call/ray.h b/cglm/include/cglm/call/ray.h new file mode 100644 index 0000000..e529fdf --- /dev/null +++ b/cglm/include/cglm/call/ray.h @@ -0,0 +1,39 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_ray_h +#define cglmc_ray_h +#ifdef __cplusplus +extern "C" { +#endif +#include "../cglm.h" + +CGLM_EXPORT +bool +glmc_ray_triangle(vec3 origin, + vec3 direction, + vec3 v0, + vec3 v1, + vec3 v2, + float *d); + +CGLM_EXPORT +bool +glmc_ray_sphere(vec3 origin, + vec3 dir, + vec4 s, + float * __restrict t1, + float * __restrict t2); + +CGLM_EXPORT +void +glmc_ray_at(vec3 orig, vec3 dir, float t, vec3 point); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_ray_h */ diff --git a/cglm/include/cglm/call/sphere.h b/cglm/include/cglm/call/sphere.h new file mode 100644 index 0000000..9b96546 --- /dev/null +++ b/cglm/include/cglm/call/sphere.h @@ -0,0 +1,39 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_sphere_h +#define cglmc_sphere_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +float +glmc_sphere_radii(vec4 s); + +CGLM_EXPORT +void +glmc_sphere_transform(vec4 s, mat4 m, vec4 dest); + +CGLM_EXPORT +void +glmc_sphere_merge(vec4 s1, vec4 s2, vec4 dest); + +CGLM_EXPORT +bool +glmc_sphere_sphere(vec4 s1, vec4 s2); + +CGLM_EXPORT +bool +glmc_sphere_point(vec4 s, vec3 point); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_sphere_h */ diff --git a/cglm/include/cglm/call/vec2.h b/cglm/include/cglm/call/vec2.h new file mode 100644 index 0000000..0284a8c --- /dev/null +++ b/cglm/include/cglm/call/vec2.h @@ -0,0 +1,251 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_vec2_h +#define cglmc_vec2_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +CGLM_EXPORT +void +glmc_vec2(float * __restrict v, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_fill(vec2 v, float val); + +CGLM_EXPORT +bool +glmc_vec2_eq(vec2 v, float val); + +CGLM_EXPORT +bool +glmc_vec2_eqv(vec2 a, vec2 b); + +CGLM_EXPORT +void +glmc_vec2_copy(vec2 a, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_zero(vec2 v); + +CGLM_EXPORT +void +glmc_vec2_one(vec2 v); + +CGLM_EXPORT +float +glmc_vec2_dot(vec2 a, vec2 b); + +CGLM_EXPORT +float +glmc_vec2_cross(vec2 a, vec2 b); + +CGLM_EXPORT +float +glmc_vec2_norm2(vec2 v); + +CGLM_EXPORT +float +glmc_vec2_norm(vec2 v); + +CGLM_EXPORT +void +glmc_vec2_add(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_adds(vec2 v, float s, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_sub(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_subs(vec2 v, float s, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_mul(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_scale(vec2 v, float s, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_scale_as(vec2 v, float s, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_div(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_divs(vec2 v, float s, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_addadd(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_subadd(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_muladd(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_muladds(vec2 a, float s, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_maxadd(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_minadd(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_subsub(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_addsub(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_mulsub(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_mulsubs(vec2 a, float s, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_maxsub(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_minsub(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_negate_to(vec2 v, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_negate(vec2 v); + +CGLM_EXPORT +void +glmc_vec2_normalize(vec2 v); + +CGLM_EXPORT +void +glmc_vec2_normalize_to(vec2 v, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_rotate(vec2 v, float angle, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_center(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +float +glmc_vec2_distance2(vec2 a, vec2 b); + +CGLM_EXPORT +float +glmc_vec2_distance(vec2 a, vec2 b); + +CGLM_EXPORT +void +glmc_vec2_maxv(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_minv(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_clamp(vec2 v, float minval, float maxval); + +CGLM_EXPORT +void +glmc_vec2_abs(vec2 v, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_fract(vec2 v, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_floor(vec2 v, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_mods(vec2 v, float s, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_swizzle(vec2 v, int mask, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_lerp(vec2 from, vec2 to, float t, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_step(vec2 edge, vec2 x, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_steps(float edge, vec2 x, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_stepr(vec2 edge, float x, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_complex_mul(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_complex_div(vec2 a, vec2 b, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_complex_conjugate(vec2 a, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_make(const float * __restrict src, vec2 dest); + +CGLM_EXPORT +void +glmc_vec2_reflect(vec2 v, vec2 n, vec2 dest); + +CGLM_EXPORT +bool +glmc_vec2_refract(vec2 v, vec2 n, float eta, vec2 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_vec2_h */ diff --git a/cglm/include/cglm/call/vec3.h b/cglm/include/cglm/call/vec3.h new file mode 100644 index 0000000..640fac7 --- /dev/null +++ b/cglm/include/cglm/call/vec3.h @@ -0,0 +1,369 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_vec3_h +#define cglmc_vec3_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +/* DEPRECATED! use _copy, _ucopy versions */ +#define glmc_vec_dup(v, dest) glmc_vec3_copy(v, dest) +#define glmc_vec3_flipsign(v) glmc_vec3_negate(v) +#define glmc_vec3_flipsign_to(v, dest) glmc_vec3_negate_to(v, dest) +#define glmc_vec3_inv(v) glmc_vec3_negate(v) +#define glmc_vec3_inv_to(v, dest) glmc_vec3_negate_to(v, dest) +#define glmc_vec3_step_uni(edge, x, dest) glmc_vec3_steps(edge, x, dest); + +CGLM_EXPORT +void +glmc_vec3(vec4 v4, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_copy(vec3 a, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_zero(vec3 v); + +CGLM_EXPORT +void +glmc_vec3_one(vec3 v); + +CGLM_EXPORT +float +glmc_vec3_dot(vec3 a, vec3 b); + +CGLM_EXPORT +void +glmc_vec3_cross(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_crossn(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +float +glmc_vec3_norm(vec3 v); + +CGLM_EXPORT +float +glmc_vec3_norm2(vec3 v); + +CGLM_EXPORT +float +glmc_vec3_norm_one(vec3 v); + +CGLM_EXPORT +float +glmc_vec3_norm_inf(vec3 v); + +CGLM_EXPORT +void +glmc_vec3_normalize_to(vec3 v, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_normalize(vec3 v); + +CGLM_EXPORT +void +glmc_vec3_add(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_adds(vec3 v, float s, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_sub(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_subs(vec3 v, float s, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_mul(vec3 a, vec3 b, vec3 d); + +CGLM_EXPORT +void +glmc_vec3_scale(vec3 v, float s, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_scale_as(vec3 v, float s, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_div(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_divs(vec3 a, float s, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_addadd(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_subadd(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_muladd(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_muladds(vec3 a, float s, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_maxadd(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_minadd(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_subsub(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_addsub(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_mulsub(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_mulsubs(vec3 a, float s, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_maxsub(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_minsub(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_negate(vec3 v); + +CGLM_EXPORT +void +glmc_vec3_negate_to(vec3 v, vec3 dest); + +CGLM_EXPORT +float +glmc_vec3_angle(vec3 a, vec3 b); + +CGLM_EXPORT +void +glmc_vec3_rotate(vec3 v, float angle, vec3 axis); + +CGLM_EXPORT +void +glmc_vec3_rotate_m4(mat4 m, vec3 v, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_rotate_m3(mat3 m, vec3 v, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_proj(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_center(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +float +glmc_vec3_distance2(vec3 a, vec3 b); + +CGLM_EXPORT +float +glmc_vec3_distance(vec3 a, vec3 b); + +CGLM_EXPORT +void +glmc_vec3_maxv(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_minv(vec3 a, vec3 b, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_clamp(vec3 v, float minVal, float maxVal); + +CGLM_EXPORT +void +glmc_vec3_ortho(vec3 v, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_lerp(vec3 from, vec3 to, float t, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_lerpc(vec3 from, vec3 to, float t, vec3 dest); + +CGLM_INLINE +void +glmc_vec3_mix(vec3 from, vec3 to, float t, vec3 dest) { + glmc_vec3_lerp(from, to, t, dest); +} + +CGLM_INLINE +void +glmc_vec3_mixc(vec3 from, vec3 to, float t, vec3 dest) { + glmc_vec3_lerpc(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_vec3_step(vec3 edge, vec3 x, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_smoothstep_uni(float edge0, float edge1, vec3 x, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_smoothstep(vec3 edge0, vec3 edge1, vec3 x, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_smoothinterp(vec3 from, vec3 to, float t, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_smoothinterpc(vec3 from, vec3 to, float t, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_swizzle(vec3 v, int mask, vec3 dest); + +/* ext */ + +CGLM_EXPORT +void +glmc_vec3_mulv(vec3 a, vec3 b, vec3 d); + +CGLM_EXPORT +void +glmc_vec3_broadcast(float val, vec3 d); + +CGLM_EXPORT +void +glmc_vec3_fill(vec3 v, float val); + +CGLM_EXPORT +bool +glmc_vec3_eq(vec3 v, float val); + +CGLM_EXPORT +bool +glmc_vec3_eq_eps(vec3 v, float val); + +CGLM_EXPORT +bool +glmc_vec3_eq_all(vec3 v); + +CGLM_EXPORT +bool +glmc_vec3_eqv(vec3 a, vec3 b); + +CGLM_EXPORT +bool +glmc_vec3_eqv_eps(vec3 a, vec3 b); + +CGLM_EXPORT +float +glmc_vec3_max(vec3 v); + +CGLM_EXPORT +float +glmc_vec3_min(vec3 v); + +CGLM_EXPORT +bool +glmc_vec3_isnan(vec3 v); + +CGLM_EXPORT +bool +glmc_vec3_isinf(vec3 v); + +CGLM_EXPORT +bool +glmc_vec3_isvalid(vec3 v); + +CGLM_EXPORT +void +glmc_vec3_sign(vec3 v, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_abs(vec3 v, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_fract(vec3 v, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_floor(vec3 v, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_mods(vec3 v, float s, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_steps(float edge, vec3 x, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_stepr(vec3 edge, float x, vec3 dest); + +CGLM_EXPORT +float +glmc_vec3_hadd(vec3 v); + +CGLM_EXPORT +void +glmc_vec3_sqrt(vec3 v, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_make(const float * __restrict src, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_faceforward(vec3 n, vec3 v, vec3 nref, vec3 dest); + +CGLM_EXPORT +void +glmc_vec3_reflect(vec3 v, vec3 n, vec3 dest); + +CGLM_EXPORT +bool +glmc_vec3_refract(vec3 v, vec3 n, float eta, vec3 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_vec3_h */ diff --git a/cglm/include/cglm/call/vec4.h b/cglm/include/cglm/call/vec4.h new file mode 100644 index 0000000..22eee24 --- /dev/null +++ b/cglm/include/cglm/call/vec4.h @@ -0,0 +1,342 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglmc_vec4_h +#define cglmc_vec4_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "../cglm.h" + +/* DEPRECATED! use _copy, _ucopy versions */ +#define glmc_vec4_dup3(v, dest) glmc_vec4_copy3(v, dest) +#define glmc_vec4_dup(v, dest) glmc_vec4_copy(v, dest) +#define glmc_vec4_flipsign(v) glmc_vec4_negate(v) +#define glmc_vec4_flipsign_to(v, dest) glmc_vec4_negate_to(v, dest) +#define glmc_vec4_inv(v) glmc_vec4_negate(v) +#define glmc_vec4_inv_to(v, dest) glmc_vec4_negate_to(v, dest) +#define glmc_vec4_step_uni(edge, x, dest) glmc_vec4_steps(edge, x, dest) + +CGLM_EXPORT +void +glmc_vec4(vec3 v3, float last, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_zero(vec4 v); + +CGLM_EXPORT +void +glmc_vec4_one(vec4 v); + +CGLM_EXPORT +void +glmc_vec4_copy3(vec4 v, vec3 dest); + +CGLM_EXPORT +void +glmc_vec4_copy(vec4 v, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_ucopy(vec4 v, vec4 dest); + +CGLM_EXPORT +float +glmc_vec4_dot(vec4 a, vec4 b); + +CGLM_EXPORT +float +glmc_vec4_norm(vec4 v); + +CGLM_EXPORT +float +glmc_vec4_norm2(vec4 v); + +CGLM_EXPORT +float +glmc_vec4_norm_one(vec4 v); + +CGLM_EXPORT +float +glmc_vec4_norm_inf(vec4 v); + +CGLM_EXPORT +void +glmc_vec4_normalize_to(vec4 v, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_normalize(vec4 v); + +CGLM_EXPORT +void +glmc_vec4_add(vec4 a, vec4 b, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_adds(vec4 v, float s, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_sub(vec4 a, vec4 b, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_subs(vec4 v, float s, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_mul(vec4 a, vec4 b, vec4 d); + +CGLM_EXPORT +void +glmc_vec4_scale(vec4 v, float s, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_scale_as(vec4 v, float s, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_div(vec4 a, vec4 b, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_divs(vec4 v, float s, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_addadd(vec4 a, vec4 b, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_subadd(vec4 a, vec4 b, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_muladd(vec4 a, vec4 b, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_muladds(vec4 a, float s, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_maxadd(vec4 a, vec4 b, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_minadd(vec4 a, vec4 b, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_subsub(vec4 a, vec4 b, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_addsub(vec4 a, vec4 b, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_mulsub(vec4 a, vec4 b, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_mulsubs(vec4 a, float s, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_maxsub(vec4 a, vec4 b, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_minsub(vec4 a, vec4 b, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_negate(vec4 v); + +CGLM_EXPORT +void +glmc_vec4_negate_to(vec4 v, vec4 dest); + +CGLM_EXPORT +float +glmc_vec4_distance(vec4 a, vec4 b); + +CGLM_EXPORT +float +glmc_vec4_distance2(vec4 a, vec4 b); + +CGLM_EXPORT +void +glmc_vec4_maxv(vec4 a, vec4 b, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_minv(vec4 a, vec4 b, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_clamp(vec4 v, float minVal, float maxVal); + +CGLM_EXPORT +void +glmc_vec4_lerp(vec4 from, vec4 to, float t, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_lerpc(vec4 from, vec4 to, float t, vec4 dest); + +CGLM_INLINE +void +glmc_vec4_mix(vec4 from, vec4 to, float t, vec4 dest) { + glmc_vec4_lerp(from, to, t, dest); +} + +CGLM_INLINE +void +glmc_vec4_mixc(vec4 from, vec4 to, float t, vec4 dest) { + glmc_vec4_lerpc(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_vec4_step(vec4 edge, vec4 x, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_smoothstep_uni(float edge0, float edge1, vec4 x, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_smoothstep(vec4 edge0, vec4 edge1, vec4 x, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_smoothinterp(vec4 from, vec4 to, float t, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_smoothinterpc(vec4 from, vec4 to, float t, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_cubic(float s, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_swizzle(vec4 v, int mask, vec4 dest); + +/* ext */ + +CGLM_EXPORT +void +glmc_vec4_mulv(vec4 a, vec4 b, vec4 d); + +CGLM_EXPORT +void +glmc_vec4_broadcast(float val, vec4 d); + +CGLM_EXPORT +void +glmc_vec4_fill(vec4 v, float val); + +CGLM_EXPORT +bool +glmc_vec4_eq(vec4 v, float val); + +CGLM_EXPORT +bool +glmc_vec4_eq_eps(vec4 v, float val); + +CGLM_EXPORT +bool +glmc_vec4_eq_all(vec4 v); + +CGLM_EXPORT +bool +glmc_vec4_eqv(vec4 a, vec4 b); + +CGLM_EXPORT +bool +glmc_vec4_eqv_eps(vec4 a, vec4 b); + +CGLM_EXPORT +float +glmc_vec4_max(vec4 v); + +CGLM_EXPORT +float +glmc_vec4_min(vec4 v); + +CGLM_EXPORT +bool +glmc_vec4_isnan(vec4 v); + +CGLM_EXPORT +bool +glmc_vec4_isinf(vec4 v); + +CGLM_EXPORT +bool +glmc_vec4_isvalid(vec4 v); + +CGLM_EXPORT +void +glmc_vec4_sign(vec4 v, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_abs(vec4 v, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_fract(vec4 v, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_floor(vec4 v, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_mods(vec4 v, float s, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_steps(float edge, vec4 x, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_stepr(vec4 edge, float x, vec4 dest); + +CGLM_EXPORT +float +glmc_vec4_hadd(vec4 v); + +CGLM_EXPORT +void +glmc_vec4_sqrt(vec4 v, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_make(const float * __restrict src, vec4 dest); + +CGLM_EXPORT +void +glmc_vec4_reflect(vec4 v, vec4 n, vec4 dest); + +CGLM_EXPORT +bool +glmc_vec4_refract(vec4 v, vec4 n, float eta, vec4 dest); + +#ifdef __cplusplus +} +#endif +#endif /* cglmc_vec4_h */ diff --git a/cglm/include/cglm/cam.h b/cglm/include/cglm/cam.h new file mode 100644 index 0000000..816cb5e --- /dev/null +++ b/cglm/include/cglm/cam.h @@ -0,0 +1,582 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_frustum(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) + CGLM_INLINE void glm_ortho(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) + CGLM_INLINE void glm_ortho_aabb(vec3 box[2], mat4 dest) + CGLM_INLINE void glm_ortho_aabb_p(vec3 box[2], float padding, mat4 dest) + CGLM_INLINE void glm_ortho_aabb_pz(vec3 box[2], float padding, mat4 dest) + CGLM_INLINE void glm_ortho_default(float aspect, mat4 dest) + CGLM_INLINE void glm_ortho_default_s(float aspect, float size, mat4 dest) + CGLM_INLINE void glm_perspective(float fovy, + float aspect, + float nearZ, + float farZ, + mat4 dest) + CGLM_INLINE void glm_perspective_default(float aspect, mat4 dest) + CGLM_INLINE void glm_perspective_resize(float aspect, mat4 proj) + CGLM_INLINE void glm_lookat(vec3 eye, vec3 center, vec3 up, mat4 dest) + CGLM_INLINE void glm_look(vec3 eye, vec3 dir, vec3 up, mat4 dest) + CGLM_INLINE void glm_look_anyup(vec3 eye, vec3 dir, mat4 dest) + CGLM_INLINE void glm_persp_decomp(mat4 proj, + float *nearZ, float *farZ, + float *top, float *bottom, + float *left, float *right) + CGLM_INLINE void glm_persp_decompv(mat4 proj, float dest[6]) + CGLM_INLINE void glm_persp_decomp_x(mat4 proj, float *left, float *right) + CGLM_INLINE void glm_persp_decomp_y(mat4 proj, float *top, float *bottom) + CGLM_INLINE void glm_persp_decomp_z(mat4 proj, float *nearv, float *farv) + CGLM_INLINE void glm_persp_decomp_far(mat4 proj, float *farZ) + CGLM_INLINE void glm_persp_decomp_near(mat4 proj, float *nearZ) + CGLM_INLINE float glm_persp_fovy(mat4 proj) + CGLM_INLINE float glm_persp_aspect(mat4 proj) + CGLM_INLINE void glm_persp_sizes(mat4 proj, float fovy, vec4 dest) + */ + +#ifndef cglm_cam_h +#define cglm_cam_h + +#include "common.h" +#include "plane.h" + +#include "clipspace/persp.h" + +#ifndef CGLM_CLIPSPACE_INCLUDE_ALL +# if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO +# include "clipspace/ortho_lh_zo.h" +# include "clipspace/persp_lh_zo.h" +# include "clipspace/view_lh_zo.h" +# elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO +# include "clipspace/ortho_lh_no.h" +# include "clipspace/persp_lh_no.h" +# include "clipspace/view_lh_no.h" +# elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO +# include "clipspace/ortho_rh_zo.h" +# include "clipspace/persp_rh_zo.h" +# include "clipspace/view_rh_zo.h" +# elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO +# include "clipspace/ortho_rh_no.h" +# include "clipspace/persp_rh_no.h" +# include "clipspace/view_rh_no.h" +# endif +#else +# include "clipspace/ortho_lh_zo.h" +# include "clipspace/persp_lh_zo.h" +# include "clipspace/ortho_lh_no.h" +# include "clipspace/persp_lh_no.h" +# include "clipspace/ortho_rh_zo.h" +# include "clipspace/persp_rh_zo.h" +# include "clipspace/ortho_rh_no.h" +# include "clipspace/persp_rh_no.h" +# include "clipspace/view_lh_zo.h" +# include "clipspace/view_lh_no.h" +# include "clipspace/view_rh_zo.h" +# include "clipspace/view_rh_no.h" +#endif + +/*! + * @brief set up perspective peprojection matrix + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_frustum(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_frustum_lh_zo(left, right, bottom, top, nearZ, farZ, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_frustum_lh_no(left, right, bottom, top, nearZ, farZ, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_frustum_rh_zo(left, right, bottom, top, nearZ, farZ, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_frustum_rh_no(left, right, bottom, top, nearZ, farZ, dest); +#endif +} + +/*! + * @brief set up orthographic projection matrix + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_ortho_lh_zo(left, right, bottom, top, nearZ, farZ, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_ortho_lh_no(left, right, bottom, top, nearZ, farZ, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_ortho_rh_zo(left, right, bottom, top, nearZ, farZ, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_ortho_rh_no(left, right, bottom, top, nearZ, farZ, dest); +#endif +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_aabb(vec3 box[2], mat4 dest) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_ortho_aabb_lh_zo(box, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_ortho_aabb_lh_no(box, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_ortho_aabb_rh_zo(box, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_ortho_aabb_rh_no(box, dest); +#endif +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_aabb_p(vec3 box[2], float padding, mat4 dest) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_ortho_aabb_p_lh_zo(box, padding, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_ortho_aabb_p_lh_no(box, padding, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_ortho_aabb_p_rh_zo(box, padding, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_ortho_aabb_p_rh_no(box, padding, dest); +#endif +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding for near and far + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_aabb_pz(vec3 box[2], float padding, mat4 dest) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_ortho_aabb_pz_lh_zo(box, padding, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_ortho_aabb_pz_lh_no(box, padding, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_ortho_aabb_pz_rh_zo(box, padding, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_ortho_aabb_pz_rh_no(box, padding, dest); +#endif +} + +/*! + * @brief set up unit orthographic projection matrix + * + * @param[in] aspect aspect ration ( width / height ) + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_default(float aspect, mat4 dest) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_ortho_default_lh_zo(aspect, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_ortho_default_lh_no(aspect, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_ortho_default_rh_zo(aspect, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_ortho_default_rh_no(aspect, dest); +#endif +} + +/*! + * @brief set up orthographic projection matrix with given CUBE size + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] size cube size + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_default_s(float aspect, float size, mat4 dest) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_ortho_default_s_lh_zo(aspect, size, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_ortho_default_s_lh_no(aspect, size, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_ortho_default_s_rh_zo(aspect, size, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_ortho_default_s_rh_no(aspect, size, dest); +#endif +} + +/*! + * @brief set up perspective projection matrix + * + * @param[in] fovy field of view angle + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping planes + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_perspective(float fovy, float aspect, float nearZ, float farZ, mat4 dest) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_perspective_lh_zo(fovy, aspect, nearZ, farZ, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_perspective_lh_no(fovy, aspect, nearZ, farZ, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_perspective_rh_zo(fovy, aspect, nearZ, farZ, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_perspective_rh_no(fovy, aspect, nearZ, farZ, dest); +#endif +} + +/*! + * @brief extend perspective projection matrix's far distance + * + * this function does not guarantee far >= near, be aware of that! + * + * @param[in, out] proj projection matrix to extend + * @param[in] deltaFar distance from existing far (negative to shink) + */ +CGLM_INLINE +void +glm_persp_move_far(mat4 proj, float deltaFar) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_persp_move_far_lh_zo(proj, deltaFar); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_persp_move_far_lh_no(proj, deltaFar); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_persp_move_far_rh_zo(proj, deltaFar); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_persp_move_far_rh_no(proj, deltaFar); +#endif +} + +/*! + * @brief set up perspective projection matrix with default near/far + * and angle values + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_perspective_default(float aspect, mat4 dest) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_perspective_default_lh_zo(aspect, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_perspective_default_lh_no(aspect, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_perspective_default_rh_zo(aspect, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_perspective_default_rh_no(aspect, dest); +#endif +} + +/*! + * @brief resize perspective matrix by aspect ratio ( width / height ) + * this makes very easy to resize proj matrix when window /viewport + * reized + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[in, out] proj perspective projection matrix + */ +CGLM_INLINE +void +glm_perspective_resize(float aspect, mat4 proj) { + if (proj[0][0] == 0.0f) + return; + + proj[0][0] = proj[1][1] / aspect; +} + +/*! + * @brief set up view matrix + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] center center vector + * @param[in] up up vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_lookat(vec3 eye, vec3 center, vec3 up, mat4 dest) { +#if CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_LH_BIT + glm_lookat_lh(eye, center, up, dest); +#elif CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_RH_BIT + glm_lookat_rh(eye, center, up, dest); +#endif +} + +/*! + * @brief set up view matrix + * + * convenient wrapper for lookat: if you only have direction not target self + * then this might be useful. Because you need to get target from direction. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[in] up up vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_look(vec3 eye, vec3 dir, vec3 up, mat4 dest) { +#if CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_LH_BIT + glm_look_lh(eye, dir, up, dest); +#elif CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_RH_BIT + glm_look_rh(eye, dir, up, dest); +#endif +} + +/*! + * @brief set up view matrix + * + * convenient wrapper for look: if you only have direction and if you don't + * care what UP vector is then this might be useful to create view matrix + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_look_anyup(vec3 eye, vec3 dir, mat4 dest) { +#if CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_LH_BIT + glm_look_anyup_lh(eye, dir, dest); +#elif CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_RH_BIT + glm_look_anyup_rh(eye, dir, dest); +#endif +} + +/*! + * @brief decomposes frustum values of perspective projection. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + * @param[out] top top + * @param[out] bottom bottom + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glm_persp_decomp(mat4 proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_persp_decomp_lh_zo(proj, nearZ, farZ, top, bottom, left, right); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_persp_decomp_lh_no(proj, nearZ, farZ, top, bottom, left, right); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_persp_decomp_rh_zo(proj, nearZ, farZ, top, bottom, left, right); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_persp_decomp_rh_no(proj, nearZ, farZ, top, bottom, left, right); +#endif +} + +/*! + * @brief decomposes frustum values of perspective projection. + * this makes easy to get all values at once + * + * @param[in] proj perspective projection matrix + * @param[out] dest array + */ +CGLM_INLINE +void +glm_persp_decompv(mat4 proj, float dest[6]) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_persp_decompv_lh_zo(proj, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_persp_decompv_lh_no(proj, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_persp_decompv_rh_zo(proj, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_persp_decompv_rh_no(proj, dest); +#endif +} + +/*! + * @brief decomposes left and right values of perspective projection. + * x stands for x axis (left / right axis) + * + * @param[in] proj perspective projection matrix + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glm_persp_decomp_x(mat4 proj, + float * __restrict left, + float * __restrict right) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_persp_decomp_x_lh_zo(proj, left, right); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_persp_decomp_x_lh_no(proj, left, right); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_persp_decomp_x_rh_zo(proj, left, right); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_persp_decomp_x_rh_no(proj, left, right); +#endif +} + +/*! + * @brief decomposes top and bottom values of perspective projection. + * y stands for y axis (top / bottom axis) + * + * @param[in] proj perspective projection matrix + * @param[out] top top + * @param[out] bottom bottom + */ +CGLM_INLINE +void +glm_persp_decomp_y(mat4 proj, + float * __restrict top, + float * __restrict bottom) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_persp_decomp_y_lh_zo(proj, top, bottom); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_persp_decomp_y_lh_no(proj, top, bottom); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_persp_decomp_y_rh_zo(proj, top, bottom); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_persp_decomp_y_rh_no(proj, top, bottom); +#endif +} + +/*! + * @brief decomposes near and far values of perspective projection. + * z stands for z axis (near / far axis) + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + */ +CGLM_INLINE +void +glm_persp_decomp_z(mat4 proj, float * __restrict nearZ, float * __restrict farZ) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_persp_decomp_z_lh_zo(proj, nearZ, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_persp_decomp_z_lh_no(proj, nearZ, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_persp_decomp_z_rh_zo(proj, nearZ, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_persp_decomp_z_rh_no(proj, nearZ, farZ); +#endif +} + +/*! + * @brief decomposes far value of perspective projection. + * + * @param[in] proj perspective projection matrix + * @param[out] farZ far + */ +CGLM_INLINE +void +glm_persp_decomp_far(mat4 proj, float * __restrict farZ) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_persp_decomp_far_lh_zo(proj, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_persp_decomp_far_lh_no(proj, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_persp_decomp_far_rh_zo(proj, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_persp_decomp_far_rh_no(proj, farZ); +#endif +} + +/*! + * @brief decomposes near value of perspective projection. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + */ +CGLM_INLINE +void +glm_persp_decomp_near(mat4 proj, float * __restrict nearZ) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_persp_decomp_near_lh_zo(proj, nearZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_persp_decomp_near_lh_no(proj, nearZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_persp_decomp_near_rh_zo(proj, nearZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_persp_decomp_near_rh_no(proj, nearZ); +#endif +} + +/*! + * @brief returns sizes of near and far planes of perspective projection + * + * @param[in] proj perspective projection matrix + * @param[in] fovy fovy (see brief) + * @param[out] dest sizes order: [Wnear, Hnear, Wfar, Hfar] + */ +CGLM_INLINE +void +glm_persp_sizes(mat4 proj, float fovy, vec4 dest) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glm_persp_sizes_lh_zo(proj, fovy, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glm_persp_sizes_lh_no(proj, fovy, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glm_persp_sizes_rh_zo(proj, fovy, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glm_persp_sizes_rh_no(proj, fovy, dest); +#endif +} + +#endif /* cglm_cam_h */ diff --git a/cglm/include/cglm/cglm.h b/cglm/include/cglm/cglm.h new file mode 100644 index 0000000..146ded0 --- /dev/null +++ b/cglm/include/cglm/cglm.h @@ -0,0 +1,47 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_h +#define cglm_h + +#include "common.h" +#include "vec2.h" +#include "vec3.h" +#include "vec4.h" +#include "ivec2.h" +#include "ivec3.h" +#include "ivec4.h" +#include "mat4.h" +#include "mat4x2.h" +#include "mat4x3.h" +#include "mat3.h" +#include "mat3x2.h" +#include "mat3x4.h" +#include "mat2.h" +#include "mat2x3.h" +#include "mat2x4.h" +#include "affine.h" +#include "cam.h" +#include "frustum.h" +#include "quat.h" +#include "euler.h" +#include "plane.h" +#include "noise.h" +#include "aabb2d.h" +#include "box.h" +#include "color.h" +#include "util.h" +#include "io.h" +#include "project.h" +#include "sphere.h" +#include "ease.h" +#include "curve.h" +#include "bezier.h" +#include "ray.h" +#include "affine2d.h" + +#endif /* cglm_h */ diff --git a/cglm/include/cglm/clipspace/ortho_lh_no.h b/cglm/include/cglm/clipspace/ortho_lh_no.h new file mode 100644 index 0000000..76c7a94 --- /dev/null +++ b/cglm/include/cglm/clipspace/ortho_lh_no.h @@ -0,0 +1,183 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_ortho_lh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) + CGLM_INLINE void glm_ortho_aabb_lh_no(vec3 box[2], mat4 dest) + CGLM_INLINE void glm_ortho_aabb_p_lh_no(vec3 box[2], + float padding, + mat4 dest) + CGLM_INLINE void glm_ortho_aabb_pz_lh_no(vec3 box[2], + float padding, + mat4 dest) + CGLM_INLINE void glm_ortho_default_lh_no(float aspect, + mat4 dest) + CGLM_INLINE void glm_ortho_default_s_lh_no(float aspect, + float size, + mat4 dest) + */ + +#ifndef cglm_ortho_lh_no_h +#define cglm_ortho_lh_no_h + +#include "../common.h" +#include "../plane.h" +#include "../mat4.h" + +/*! + * @brief set up orthographic projection matrix + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_lh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + float rl, tb, fn; + + glm_mat4_zero(dest); + + rl = 1.0f / (right - left); + tb = 1.0f / (top - bottom); + fn =-1.0f / (farZ - nearZ); + + dest[0][0] = 2.0f * rl; + dest[1][1] = 2.0f * tb; + dest[2][2] =-2.0f * fn; + dest[3][0] =-(right + left) * rl; + dest[3][1] =-(top + bottom) * tb; + dest[3][2] = (farZ + nearZ) * fn; + dest[3][3] = 1.0f; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_aabb_lh_no(vec3 box[2], mat4 dest) { + glm_ortho_lh_no(box[0][0], box[1][0], + box[0][1], box[1][1], + -box[1][2], -box[0][2], + dest); +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_aabb_p_lh_no(vec3 box[2], float padding, mat4 dest) { + glm_ortho_lh_no(box[0][0] - padding, box[1][0] + padding, + box[0][1] - padding, box[1][1] + padding, + -(box[1][2] + padding), -(box[0][2] - padding), + dest); +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding for near and far + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_aabb_pz_lh_no(vec3 box[2], float padding, mat4 dest) { + glm_ortho_lh_no(box[0][0], box[1][0], + box[0][1], box[1][1], + -(box[1][2] + padding), -(box[0][2] - padding), + dest); +} + +/*! + * @brief set up unit orthographic projection matrix + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] aspect aspect ration ( width / height ) + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_default_lh_no(float aspect, mat4 dest) { + if (aspect >= 1.0f) { + glm_ortho_lh_no(-aspect, aspect, -1.0f, 1.0f, -100.0f, 100.0f, dest); + return; + } + + aspect = 1.0f / aspect; + + glm_ortho_lh_no(-1.0f, 1.0f, -aspect, aspect, -100.0f, 100.0f, dest); +} + +/*! + * @brief set up orthographic projection matrix with given CUBE size + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] size cube size + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_default_s_lh_no(float aspect, float size, mat4 dest) { + if (aspect >= 1.0f) { + glm_ortho_lh_no(-size * aspect, + size * aspect, + -size, + size, + -size - 100.0f, + size + 100.0f, + dest); + return; + } + + glm_ortho_lh_no(-size, + size, + -size / aspect, + size / aspect, + -size - 100.0f, + size + 100.0f, + dest); +} + +#endif /*cglm_ortho_lh_no_h*/ diff --git a/cglm/include/cglm/clipspace/ortho_lh_zo.h b/cglm/include/cglm/clipspace/ortho_lh_zo.h new file mode 100644 index 0000000..e45530d --- /dev/null +++ b/cglm/include/cglm/clipspace/ortho_lh_zo.h @@ -0,0 +1,177 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_ortho_lh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) + CGLM_INLINE void glm_ortho_aabb_lh_zo(vec3 box[2], mat4 dest) + CGLM_INLINE void glm_ortho_aabb_p_lh_zo(vec3 box[2], + float padding, + mat4 dest) + CGLM_INLINE void glm_ortho_aabb_pz_lh_zo(vec3 box[2], + float padding, + mat4 dest) + CGLM_INLINE void glm_ortho_default_lh_zo(float aspect, + mat4 dest) + CGLM_INLINE void glm_ortho_default_s_lh_zo(float aspect, + float size, + mat4 dest) + */ + +#ifndef cglm_ortho_lh_zo_h +#define cglm_ortho_lh_zo_h + +#include "../common.h" +#include "../plane.h" +#include "../mat4.h" + +/*! + * @brief set up orthographic projection matrix with a left-hand coordinate + * system and a clip-space of [0, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_lh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + float rl, tb, fn; + + glm_mat4_zero(dest); + + rl = 1.0f / (right - left); + tb = 1.0f / (top - bottom); + fn =-1.0f / (farZ - nearZ); + + dest[0][0] = 2.0f * rl; + dest[1][1] = 2.0f * tb; + dest[2][2] =-fn; + dest[3][0] =-(right + left) * rl; + dest[3][1] =-(top + bottom) * tb; + dest[3][2] = nearZ * fn; + dest[3][3] = 1.0f; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a left-hand coordinate system and a clip-space of [0, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_aabb_lh_zo(vec3 box[2], mat4 dest) { + glm_ortho_lh_zo(box[0][0], box[1][0], + box[0][1], box[1][1], + -box[1][2], -box[0][2], + dest); +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a left-hand coordinate system and a clip-space of [0, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_aabb_p_lh_zo(vec3 box[2], float padding, mat4 dest) { + glm_ortho_lh_zo(box[0][0] - padding, box[1][0] + padding, + box[0][1] - padding, box[1][1] + padding, + -(box[1][2] + padding), -(box[0][2] - padding), + dest); +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a left-hand coordinate system and a clip-space of [0, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding for near and far + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_aabb_pz_lh_zo(vec3 box[2], float padding, mat4 dest) { + glm_ortho_lh_zo(box[0][0], box[1][0], + box[0][1], box[1][1], + -(box[1][2] + padding), -(box[0][2] - padding), + dest); +} + +/*! + * @brief set up unit orthographic projection matrix + * with a left-hand coordinate system and a clip-space of [0, 1]. + * + * @param[in] aspect aspect ration ( width / height ) + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_default_lh_zo(float aspect, mat4 dest) { + if (aspect >= 1.0f) { + glm_ortho_lh_zo(-aspect, aspect, -1.0f, 1.0f, -100.0f, 100.0f, dest); + return; + } + + aspect = 1.0f / aspect; + + glm_ortho_lh_zo(-1.0f, 1.0f, -aspect, aspect, -100.0f, 100.0f, dest); +} + +/*! + * @brief set up orthographic projection matrix with given CUBE size + * with a left-hand coordinate system and a clip-space of [0, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] size cube size + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_default_s_lh_zo(float aspect, float size, mat4 dest) { + if (aspect >= 1.0f) { + glm_ortho_lh_zo(-size * aspect, + size * aspect, + -size, + size, + -size - 100.0f, + size + 100.0f, + dest); + return; + } + + glm_ortho_lh_zo(-size, + size, + -size / aspect, + size / aspect, + -size - 100.0f, + size + 100.0f, + dest); +} + +#endif /*cglm_ortho_lh_zo_h*/ diff --git a/cglm/include/cglm/clipspace/ortho_rh_no.h b/cglm/include/cglm/clipspace/ortho_rh_no.h new file mode 100644 index 0000000..aa7a906 --- /dev/null +++ b/cglm/include/cglm/clipspace/ortho_rh_no.h @@ -0,0 +1,183 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_ortho_rh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) + CGLM_INLINE void glm_ortho_aabb_rh_no(vec3 box[2], mat4 dest) + CGLM_INLINE void glm_ortho_aabb_p_rh_no(vec3 box[2], + float padding, + mat4 dest) + CGLM_INLINE void glm_ortho_aabb_pz_rh_no(vec3 box[2], + float padding, + mat4 dest) + CGLM_INLINE void glm_ortho_default_rh_no(float aspect, + mat4 dest) + CGLM_INLINE void glm_ortho_default_s_rh_no(float aspect, + float size, + mat4 dest) + */ + +#ifndef cglm_ortho_rh_no_h +#define cglm_ortho_rh_no_h + +#include "../common.h" +#include "../plane.h" +#include "../mat4.h" + +/*! + * @brief set up orthographic projection matrix + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_rh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + float rl, tb, fn; + + glm_mat4_zero(dest); + + rl = 1.0f / (right - left); + tb = 1.0f / (top - bottom); + fn =-1.0f / (farZ - nearZ); + + dest[0][0] = 2.0f * rl; + dest[1][1] = 2.0f * tb; + dest[2][2] = 2.0f * fn; + dest[3][0] =-(right + left) * rl; + dest[3][1] =-(top + bottom) * tb; + dest[3][2] = (farZ + nearZ) * fn; + dest[3][3] = 1.0f; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_aabb_rh_no(vec3 box[2], mat4 dest) { + glm_ortho_rh_no(box[0][0], box[1][0], + box[0][1], box[1][1], + -box[1][2], -box[0][2], + dest); +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_aabb_p_rh_no(vec3 box[2], float padding, mat4 dest) { + glm_ortho_rh_no(box[0][0] - padding, box[1][0] + padding, + box[0][1] - padding, box[1][1] + padding, + -(box[1][2] + padding), -(box[0][2] - padding), + dest); +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding for near and far + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_aabb_pz_rh_no(vec3 box[2], float padding, mat4 dest) { + glm_ortho_rh_no(box[0][0], box[1][0], + box[0][1], box[1][1], + -(box[1][2] + padding), -(box[0][2] - padding), + dest); +} + +/*! + * @brief set up unit orthographic projection matrix + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] aspect aspect ration ( width / height ) + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_default_rh_no(float aspect, mat4 dest) { + if (aspect >= 1.0f) { + glm_ortho_rh_no(-aspect, aspect, -1.0f, 1.0f, -100.0f, 100.0f, dest); + return; + } + + aspect = 1.0f / aspect; + + glm_ortho_rh_no(-1.0f, 1.0f, -aspect, aspect, -100.0f, 100.0f, dest); +} + +/*! + * @brief set up orthographic projection matrix with given CUBE size + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] size cube size + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_default_s_rh_no(float aspect, float size, mat4 dest) { + if (aspect >= 1.0f) { + glm_ortho_rh_no(-size * aspect, + size * aspect, + -size, + size, + -size - 100.0f, + size + 100.0f, + dest); + return; + } + + glm_ortho_rh_no(-size, + size, + -size / aspect, + size / aspect, + -size - 100.0f, + size + 100.0f, + dest); +} + +#endif /*cglm_ortho_rh_no_h*/ diff --git a/cglm/include/cglm/clipspace/ortho_rh_zo.h b/cglm/include/cglm/clipspace/ortho_rh_zo.h new file mode 100644 index 0000000..7a0876c --- /dev/null +++ b/cglm/include/cglm/clipspace/ortho_rh_zo.h @@ -0,0 +1,181 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_ortho_rh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) + CGLM_INLINE void glm_ortho_aabb_rh_zo(vec3 box[2], mat4 dest) + CGLM_INLINE void glm_ortho_aabb_p_rh_zo(vec3 box[2], + float padding, + mat4 dest) + CGLM_INLINE void glm_ortho_aabb_pz_rh_zo(vec3 box[2], + float padding, + mat4 dest) + CGLM_INLINE void glm_ortho_default_rh_zo(float aspect, + mat4 dest) + CGLM_INLINE void glm_ortho_default_s_rh_zo(float aspect, + float size, + mat4 dest) + */ + +#ifndef cglm_ortho_rh_zo_h +#define cglm_ortho_rh_zo_h + +#include "../common.h" +#include "../plane.h" +#include "../mat4.h" + +/*! + * @brief set up orthographic projection matrix with a right-hand coordinate + * system and a clip-space of [0, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_rh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + float rl, tb, fn; + + glm_mat4_zero(dest); + + rl = 1.0f / (right - left); + tb = 1.0f / (top - bottom); + fn =-1.0f / (farZ - nearZ); + + dest[0][0] = 2.0f * rl; + dest[1][1] = 2.0f * tb; + dest[2][2] = fn; + dest[3][0] =-(right + left) * rl; + dest[3][1] =-(top + bottom) * tb; + dest[3][2] = nearZ * fn; + dest[3][3] = 1.0f; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a right-hand coordinate system and a clip-space with depth + * values from zero to one. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_aabb_rh_zo(vec3 box[2], mat4 dest) { + glm_ortho_rh_zo(box[0][0], box[1][0], + box[0][1], box[1][1], + -box[1][2], -box[0][2], + dest); +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a right-hand coordinate system and a clip-space with depth + * values from zero to one. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_aabb_p_rh_zo(vec3 box[2], float padding, mat4 dest) { + glm_ortho_rh_zo(box[0][0] - padding, box[1][0] + padding, + box[0][1] - padding, box[1][1] + padding, + -(box[1][2] + padding), -(box[0][2] - padding), + dest); +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a right-hand coordinate system and a clip-space with depth + * values from zero to one. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding for near and far + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_aabb_pz_rh_zo(vec3 box[2], float padding, mat4 dest) { + glm_ortho_rh_zo(box[0][0], box[1][0], + box[0][1], box[1][1], + -(box[1][2] + padding), -(box[0][2] - padding), + dest); +} + +/*! + * @brief set up unit orthographic projection matrix with a right-hand + * coordinate system and a clip-space of [0, 1]. + * + * @param[in] aspect aspect ration ( width / height ) + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_default_rh_zo(float aspect, mat4 dest) { + if (aspect >= 1.0f) { + glm_ortho_rh_zo(-aspect, aspect, -1.0f, 1.0f, -100.0f, 100.0f, dest); + return; + } + + aspect = 1.0f / aspect; + + glm_ortho_rh_zo(-1.0f, 1.0f, -aspect, aspect, -100.0f, 100.0f, dest); +} + +/*! + * @brief set up orthographic projection matrix with given CUBE size + * with a right-hand coordinate system and a clip-space with depth + * values from zero to one. + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] size cube size + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_ortho_default_s_rh_zo(float aspect, float size, mat4 dest) { + if (aspect >= 1.0f) { + glm_ortho_rh_zo(-size * aspect, + size * aspect, + -size, + size, + -size - 100.0f, + size + 100.0f, + dest); + return; + } + + glm_ortho_rh_zo(-size, + size, + -size / aspect, + size / aspect, + -size - 100.0f, + size + 100.0f, + dest); +} + +#endif /*cglm_ortho_rh_zo_h*/ diff --git a/cglm/include/cglm/clipspace/persp.h b/cglm/include/cglm/clipspace/persp.h new file mode 100644 index 0000000..15aa715 --- /dev/null +++ b/cglm/include/cglm/clipspace/persp.h @@ -0,0 +1,48 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_persp_decomp_far(mat4 proj, float *farZ) + CGLM_INLINE float glm_persp_fovy(mat4 proj) + CGLM_INLINE float glm_persp_aspect(mat4 proj) + CGLM_INLINE void glm_persp_sizes(mat4 proj, float fovy, vec4 dest) + */ + +#ifndef cglm_persp_h +#define cglm_persp_h + +#include "../common.h" +#include "../plane.h" +#include "../mat4.h" + +/*! + * @brief returns field of view angle along the Y-axis (in radians) + * + * if you need to degrees, use glm_deg to convert it or use this: + * fovy_deg = glm_deg(glm_persp_fovy(projMatrix)) + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glm_persp_fovy(mat4 proj) { + return 2.0f * atanf(1.0f / proj[1][1]); +} + +/*! + * @brief returns aspect ratio of perspective projection + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glm_persp_aspect(mat4 proj) { + return proj[1][1] / proj[0][0]; +} + +#endif /* cglm_persp_h */ diff --git a/cglm/include/cglm/clipspace/persp_lh_no.h b/cglm/include/cglm/clipspace/persp_lh_no.h new file mode 100644 index 0000000..d28923a --- /dev/null +++ b/cglm/include/cglm/clipspace/persp_lh_no.h @@ -0,0 +1,395 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_frustum_lh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) + CGLM_INLINE void glm_perspective_lh_no(float fovy, + float aspect, + float nearZ, + float farZ, + mat4 dest) + CGLM_INLINE void glm_perspective_default_lh_no(float aspect, mat4 dest) + CGLM_INLINE void glm_perspective_resize_lh_no(float aspect, mat4 proj) + CGLM_INLINE void glm_persp_move_far_lh_no(mat4 proj, + float deltaFar) + CGLM_INLINE void glm_persp_decomp_lh_no(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ, + float * __restrict top, + float * __restrict bottom, + float * __restrict left, + float * __restrict right) + CGLM_INLINE void glm_persp_decompv_lh_no(mat4 proj, + float dest[6]) + CGLM_INLINE void glm_persp_decomp_x_lh_no(mat4 proj, + float * __restrict left, + float * __restrict right) + CGLM_INLINE void glm_persp_decomp_y_lh_no(mat4 proj, + float * __restrict top, + float * __restrict bottom) + CGLM_INLINE void glm_persp_decomp_z_lh_no(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ) + CGLM_INLINE void glm_persp_decomp_far_lh_no(mat4 proj, float * __restrict farZ) + CGLM_INLINE void glm_persp_decomp_near_lh_no(mat4 proj, float * __restrict nearZ) + CGLM_INLINE void glm_persp_sizes_lh_no(mat4 proj, float fovy, vec4 dest) + */ + +#ifndef cglm_persp_lh_no_h +#define cglm_persp_lh_no_h + +#include "../common.h" +#include "persp.h" + +/*! + * @brief set up perspective peprojection matrix + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_frustum_lh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + float rl, tb, fn, nv; + + glm_mat4_zero(dest); + + rl = 1.0f / (right - left); + tb = 1.0f / (top - bottom); + fn =-1.0f / (farZ - nearZ); + nv = 2.0f * nearZ; + + dest[0][0] = nv * rl; + dest[1][1] = nv * tb; + dest[2][0] = (right + left) * rl; + dest[2][1] = (top + bottom) * tb; + dest[2][2] =-(farZ + nearZ) * fn; + dest[2][3] = 1.0f; + dest[3][2] = farZ * nv * fn; +} + +/*! + * @brief set up perspective projection matrix + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] fovy field of view angle + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping planes + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_perspective_lh_no(float fovy, + float aspect, + float nearZ, + float farZ, + mat4 dest) { + float f, fn; + + glm_mat4_zero(dest); + + f = 1.0f / tanf(fovy * 0.5f); + fn = 1.0f / (nearZ - farZ); + + dest[0][0] = f / aspect; + dest[1][1] = f; + dest[2][2] =-(nearZ + farZ) * fn; + dest[2][3] = 1.0f; + dest[3][2] = 2.0f * nearZ * farZ * fn; + +} + +/*! + * @brief set up perspective projection matrix with default near/far + * and angle values with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_perspective_default_lh_no(float aspect, mat4 dest) { + glm_perspective_lh_no(GLM_PI_4f, aspect, 0.01f, 100.0f, dest); +} + +/*! + * @brief resize perspective matrix by aspect ratio ( width / height ) + * this makes very easy to resize proj matrix when window /viewport + * resized with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[in, out] proj perspective projection matrix + */ +CGLM_INLINE +void +glm_perspective_resize_lh_no(float aspect, mat4 proj) { + if (proj[0][0] == 0.0f) + return; + + proj[0][0] = proj[1][1] / aspect; +} + +/*! + * @brief extend perspective projection matrix's far distance + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * this function does not guarantee far >= near, be aware of that! + * + * @param[in, out] proj projection matrix to extend + * @param[in] deltaFar distance from existing far (negative to shink) + */ +CGLM_INLINE +void +glm_persp_move_far_lh_no(mat4 proj, float deltaFar) { + float fn, farZ, nearZ, p22, p32; + + p22 = -proj[2][2]; + p32 = proj[3][2]; + + nearZ = p32 / (p22 - 1.0f); + farZ = p32 / (p22 + 1.0f) + deltaFar; + fn = 1.0f / (nearZ - farZ); + + proj[2][2] = -(farZ + nearZ) * fn; + proj[3][2] = 2.0f * nearZ * farZ * fn; +} + +/*! + * @brief decomposes frustum values of perspective projection + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + * @param[out] top top + * @param[out] bottom bottom + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glm_persp_decomp_lh_no(mat4 proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right) { + float m00, m11, m20, m21, m22, m32, n, f; + float n_m11, n_m00; + + m00 = proj[0][0]; + m11 = proj[1][1]; + m20 = proj[2][0]; + m21 = proj[2][1]; + m22 =-proj[2][2]; + m32 = proj[3][2]; + + n = m32 / (m22 - 1.0f); + f = m32 / (m22 + 1.0f); + + n_m11 = n / m11; + n_m00 = n / m00; + + *nearZ = n; + *farZ = f; + *bottom = n_m11 * (m21 - 1.0f); + *top = n_m11 * (m21 + 1.0f); + *left = n_m00 * (m20 - 1.0f); + *right = n_m00 * (m20 + 1.0f); +} + +/*! + * @brief decomposes frustum values of perspective projection + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * this makes easy to get all values at once + * + * @param[in] proj perspective projection matrix + * @param[out] dest array + */ +CGLM_INLINE +void +glm_persp_decompv_lh_no(mat4 proj, float dest[6]) { + glm_persp_decomp_lh_no(proj, &dest[0], &dest[1], &dest[2], + &dest[3], &dest[4], &dest[5]); +} + +/*! + * @brief decomposes left and right values of perspective projection + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * x stands for x axis (left / right axis) + * + * @param[in] proj perspective projection matrix + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glm_persp_decomp_x_lh_no(mat4 proj, + float * __restrict left, + float * __restrict right) { + float nearZ, m20, m00, m22; + + m00 = proj[0][0]; + m20 = proj[2][0]; + m22 =-proj[2][2]; + + nearZ = proj[3][2] / (m22 - 1.0f); + *left = nearZ * (m20 - 1.0f) / m00; + *right = nearZ * (m20 + 1.0f) / m00; +} + +/*! + * @brief decomposes top and bottom values of perspective projection + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * y stands for y axis (top / bottom axis) + * + * @param[in] proj perspective projection matrix + * @param[out] top top + * @param[out] bottom bottom + */ +CGLM_INLINE +void +glm_persp_decomp_y_lh_no(mat4 proj, + float * __restrict top, + float * __restrict bottom) { + float nearZ, m21, m11, m22; + + m21 = proj[2][1]; + m11 = proj[1][1]; + m22 =-proj[2][2]; + + nearZ = proj[3][2] / (m22 - 1.0f); + *bottom = nearZ * (m21 - 1.0f) / m11; + *top = nearZ * (m21 + 1.0f) / m11; +} + +/*! + * @brief decomposes near and far values of perspective projection + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * z stands for z axis (near / far axis) + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + */ +CGLM_INLINE +void +glm_persp_decomp_z_lh_no(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ) { + float m32, m22; + + m32 = proj[3][2]; + m22 =-proj[2][2]; + + *nearZ = m32 / (m22 - 1.0f); + *farZ = m32 / (m22 + 1.0f); +} + +/*! + * @brief decomposes far value of perspective projection + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] farZ far + */ +CGLM_INLINE +void +glm_persp_decomp_far_lh_no(mat4 proj, float * __restrict farZ) { + *farZ = proj[3][2] / (-proj[2][2] + 1.0f); +} + +/*! + * @brief decomposes near value of perspective projection + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + */ +CGLM_INLINE +void +glm_persp_decomp_near_lh_no(mat4 proj, float * __restrict nearZ) { + *nearZ = proj[3][2] / (-proj[2][2] - 1.0f); +} + +/*! + * @brief returns sizes of near and far planes of perspective projection + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[in] fovy fovy (see brief) + * @param[out] dest sizes order: [Wnear, Hnear, Wfar, Hfar] + */ +CGLM_INLINE +void +glm_persp_sizes_lh_no(mat4 proj, float fovy, vec4 dest) { + float t, a, nearZ, farZ; + + t = 2.0f * tanf(fovy * 0.5f); + a = glm_persp_aspect(proj); + + glm_persp_decomp_z_lh_no(proj, &nearZ, &farZ); + + dest[1] = t * nearZ; + dest[3] = t * farZ; + dest[0] = a * dest[1]; + dest[2] = a * dest[3]; +} + +/*! + * @brief returns field of view angle along the Y-axis (in radians) + * with a left-hand coordinate system and a clip-space of [-1, 1]. + * + * if you need to degrees, use glm_deg to convert it or use this: + * fovy_deg = glm_deg(glm_persp_fovy(projMatrix)) + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glm_persp_fovy_lh_no(mat4 proj) { + return glm_persp_fovy(proj); +} + +/*! + * @brief returns aspect ratio of perspective projection + * with a left-hand coordinate system and a clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glm_persp_aspect_lh_no(mat4 proj) { + return glm_persp_aspect(proj); +} + +#endif /*cglm_cam_lh_no_h*/ diff --git a/cglm/include/cglm/clipspace/persp_lh_zo.h b/cglm/include/cglm/clipspace/persp_lh_zo.h new file mode 100644 index 0000000..de89643 --- /dev/null +++ b/cglm/include/cglm/clipspace/persp_lh_zo.h @@ -0,0 +1,387 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_frustum_lh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) + CGLM_INLINE void glm_perspective_lh_zo(float fovy, + float aspect, + float nearZ, + float farZ, + mat4 dest) + CGLM_INLINE void glm_perspective_default_lh_zo(float aspect, mat4 dest) + CGLM_INLINE void glm_perspective_resize_lh_zo(float aspect, mat4 proj) + CGLM_INLINE void glm_persp_move_far_lh_zo(mat4 proj, + float deltaFar) + CGLM_INLINE void glm_persp_decomp_lh_zo(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ, + float * __restrict top, + float * __restrict bottom, + float * __restrict left, + float * __restrict right) + CGLM_INLINE void glm_persp_decompv_lh_zo(mat4 proj, + float dest[6]) + CGLM_INLINE void glm_persp_decomp_x_lh_zo(mat4 proj, + float * __restrict left, + float * __restrict right) + CGLM_INLINE void glm_persp_decomp_y_lh_zo(mat4 proj, + float * __restrict top, + float * __restrict bottom) + CGLM_INLINE void glm_persp_decomp_z_lh_zo(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ) + CGLM_INLINE void glm_persp_decomp_far_lh_zo(mat4 proj, float * __restrict farZ) + CGLM_INLINE void glm_persp_decomp_near_lh_zo(mat4 proj, float * __restrict nearZ) + CGLM_INLINE void glm_persp_sizes_lh_zo(mat4 proj, float fovy, vec4 dest) + */ + +#ifndef cglm_persp_lh_zo_h +#define cglm_persp_lh_zo_h + +#include "../common.h" +#include "persp.h" + +/*! + * @brief set up perspective peprojection matrix with a left-hand coordinate + * system and a clip-space of [0, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_frustum_lh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + float rl, tb, fn, nv; + + glm_mat4_zero(dest); + + rl = 1.0f / (right - left); + tb = 1.0f / (top - bottom); + fn =-1.0f / (farZ - nearZ); + nv = 2.0f * nearZ; + + dest[0][0] = nv * rl; + dest[1][1] = nv * tb; + dest[2][0] = (right + left) * rl; + dest[2][1] = (top + bottom) * tb; + dest[2][2] =-farZ * fn; + dest[2][3] = 1.0f; + dest[3][2] = farZ * nearZ * fn; +} + +/*! + * @brief set up perspective projection matrix with a left-hand coordinate + * system and a clip-space of [0, 1]. + * + * @param[in] fovy field of view angle + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping planes + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_perspective_lh_zo(float fovy, + float aspect, + float nearZ, + float farZ, + mat4 dest) { + float f, fn; + + glm_mat4_zero(dest); + + f = 1.0f / tanf(fovy * 0.5f); + fn = 1.0f / (nearZ - farZ); + + dest[0][0] = f / aspect; + dest[1][1] = f; + dest[2][2] =-farZ * fn; + dest[2][3] = 1.0f; + dest[3][2] = nearZ * farZ * fn; +} + +/*! + * @brief extend perspective projection matrix's far distance with a + * left-hand coordinate system and a clip-space with depth values + * from zero to one. + * + * this function does not guarantee far >= near, be aware of that! + * + * @param[in, out] proj projection matrix to extend + * @param[in] deltaFar distance from existing far (negative to shink) + */ +CGLM_INLINE +void +glm_persp_move_far_lh_zo(mat4 proj, float deltaFar) { + float fn, farZ, nearZ, p22, p32; + + p22 = -proj[2][2]; + p32 = proj[3][2]; + + nearZ = p32 / p22; + farZ = p32 / (p22 + 1.0f) + deltaFar; + fn = 1.0f / (nearZ - farZ); + + proj[2][2] = -farZ * fn; + proj[3][2] = nearZ * farZ * fn; +} + +/*! + * @brief set up perspective projection matrix with default near/far + * and angle values with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_perspective_default_lh_zo(float aspect, mat4 dest) { + glm_perspective_lh_zo(GLM_PI_4f, aspect, 0.01f, 100.0f, dest); +} + +/*! + * @brief resize perspective matrix by aspect ratio ( width / height ) + * this makes very easy to resize proj matrix when window /viewport + * reized + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[in, out] proj perspective projection matrix + */ +CGLM_INLINE +void +glm_perspective_resize_lh_zo(float aspect, mat4 proj) { + if (proj[0][0] == 0.0f) + return; + + proj[0][0] = proj[1][1] / aspect; +} + +/*! + * @brief decomposes frustum values of perspective projection + * with angle values with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + * @param[out] top top + * @param[out] bottom bottom + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glm_persp_decomp_lh_zo(mat4 proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right) { + float m00, m11, m20, m21, m22, m32, n, f; + float n_m11, n_m00; + + m00 = proj[0][0]; + m11 = proj[1][1]; + m20 = proj[2][0]; + m21 = proj[2][1]; + m22 =-proj[2][2]; + m32 = proj[3][2]; + + n = m32 / m22; + f = m32 / (m22 + 1.0f); + + n_m11 = n / m11; + n_m00 = n / m00; + + *nearZ = n; + *farZ = f; + *bottom = n_m11 * (m21 - 1.0f); + *top = n_m11 * (m21 + 1.0f); + *left = n_m00 * (m20 - 1.0f); + *right = n_m00 * (m20 + 1.0f); +} + +/*! + * @brief decomposes frustum values of perspective projection + * with angle values with a left-hand coordinate system and a + * clip-space of [0, 1]. + * this makes easy to get all values at once + * + * @param[in] proj perspective projection matrix + * @param[out] dest array + */ +CGLM_INLINE +void +glm_persp_decompv_lh_zo(mat4 proj, float dest[6]) { + glm_persp_decomp_lh_zo(proj, &dest[0], &dest[1], &dest[2], + &dest[3], &dest[4], &dest[5]); +} + +/*! + * @brief decomposes left and right values of perspective projection (ZO). + * x stands for x axis (left / right axis) + * + * @param[in] proj perspective projection matrix + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glm_persp_decomp_x_lh_zo(mat4 proj, + float * __restrict left, + float * __restrict right) { + float nearZ, m20, m00; + + m00 = proj[0][0]; + m20 = proj[2][0]; + + nearZ = proj[3][2] / (proj[3][3]); + *left = nearZ * (m20 - 1.0f) / m00; + *right = nearZ * (m20 + 1.0f) / m00; +} + +/*! + * @brief decomposes top and bottom values of perspective projection + * with angle values with a left-hand coordinate system and a + * clip-space of [0, 1]. + * y stands for y axis (top / bottom axis) + * + * @param[in] proj perspective projection matrix + * @param[out] top top + * @param[out] bottom bottom + */ +CGLM_INLINE +void +glm_persp_decomp_y_lh_zo(mat4 proj, + float * __restrict top, + float * __restrict bottom) { + float nearZ, m21, m11; + + m21 = proj[2][1]; + m11 = proj[1][1]; + + nearZ = proj[3][2] / (proj[3][3]); + *bottom = nearZ * (m21 - 1) / m11; + *top = nearZ * (m21 + 1) / m11; +} + +/*! + * @brief decomposes near and far values of perspective projection + * with angle values with a left-hand coordinate system and a + * clip-space of [0, 1]. + * z stands for z axis (near / far axis) + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + */ +CGLM_INLINE +void +glm_persp_decomp_z_lh_zo(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ) { + float m32, m22; + + m32 = proj[3][2]; + m22 = -proj[2][2]; + + *nearZ = m32 / m22; + *farZ = m32 / (m22 + 1.0f); +} + +/*! + * @brief decomposes far value of perspective projection + * with angle values with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] farZ far + */ +CGLM_INLINE +void +glm_persp_decomp_far_lh_zo(mat4 proj, float * __restrict farZ) { + *farZ = proj[3][2] / (-proj[2][2] + 1.0f); +} + +/*! + * @brief decomposes near value of perspective projection + * with angle values with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + */ +CGLM_INLINE +void +glm_persp_decomp_near_lh_zo(mat4 proj, float * __restrict nearZ) { + *nearZ = proj[3][2] / -proj[2][2]; +} + +/*! + * @brief returns sizes of near and far planes of perspective projection + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[in] fovy fovy (see brief) + * @param[out] dest sizes order: [Wnear, Hnear, Wfar, Hfar] + */ +CGLM_INLINE +void +glm_persp_sizes_lh_zo(mat4 proj, float fovy, vec4 dest) { + float t, a, nearZ, farZ; + + t = 2.0f * tanf(fovy * 0.5f); + a = glm_persp_aspect(proj); + + glm_persp_decomp_z_lh_zo(proj, &nearZ, &farZ); + + dest[1] = t * nearZ; + dest[3] = t * farZ; + dest[0] = a * dest[1]; + dest[2] = a * dest[3]; +} + +/*! + * @brief returns field of view angle along the Y-axis (in radians) + * with a left-hand coordinate system and a clip-space of [0, 1]. + * + * if you need to degrees, use glm_deg to convert it or use this: + * fovy_deg = glm_deg(glm_persp_fovy(projMatrix)) + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glm_persp_fovy_lh_zo(mat4 proj) { + return glm_persp_fovy(proj); +} + +/*! + * @brief returns aspect ratio of perspective projection + * with a left-hand coordinate system and a clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glm_persp_aspect_lh_zo(mat4 proj) { + return glm_persp_aspect(proj); +} + +#endif /*cglm_persp_lh_zo_h*/ diff --git a/cglm/include/cglm/clipspace/persp_rh_no.h b/cglm/include/cglm/clipspace/persp_rh_no.h new file mode 100644 index 0000000..9252332 --- /dev/null +++ b/cglm/include/cglm/clipspace/persp_rh_no.h @@ -0,0 +1,395 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_frustum_rh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) + CGLM_INLINE void glm_perspective_rh_no(float fovy, + float aspect, + float nearZ, + float farZ, + mat4 dest) + CGLM_INLINE void glm_perspective_default_rh_no(float aspect, mat4 dest) + CGLM_INLINE void glm_perspective_resize_rh_no(float aspect, mat4 proj) + CGLM_INLINE void glm_persp_move_far_rh_no(mat4 proj, + float deltaFar) + CGLM_INLINE void glm_persp_decomp_rh_no(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ, + float * __restrict top, + float * __restrict bottom, + float * __restrict left, + float * __restrict right) + CGLM_INLINE void glm_persp_decompv_rh_no(mat4 proj, + float dest[6]) + CGLM_INLINE void glm_persp_decomp_x_rh_no(mat4 proj, + float * __restrict left, + float * __restrict right) + CGLM_INLINE void glm_persp_decomp_y_rh_no(mat4 proj, + float * __restrict top, + float * __restrict bottom) + CGLM_INLINE void glm_persp_decomp_z_rh_no(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ) + CGLM_INLINE void glm_persp_decomp_far_rh_no(mat4 proj, float * __restrict farZ) + CGLM_INLINE void glm_persp_decomp_near_rh_no(mat4 proj, float * __restrict nearZ) + CGLM_INLINE void glm_persp_sizes_rh_no(mat4 proj, float fovy, vec4 dest) + */ + +#ifndef cglm_persp_rh_no_h +#define cglm_persp_rh_no_h + +#include "../common.h" +#include "persp.h" + +/*! + * @brief set up perspective peprojection matrix + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_frustum_rh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + float rl, tb, fn, nv; + + glm_mat4_zero(dest); + + rl = 1.0f / (right - left); + tb = 1.0f / (top - bottom); + fn =-1.0f / (farZ - nearZ); + nv = 2.0f * nearZ; + + dest[0][0] = nv * rl; + dest[1][1] = nv * tb; + dest[2][0] = (right + left) * rl; + dest[2][1] = (top + bottom) * tb; + dest[2][2] = (farZ + nearZ) * fn; + dest[2][3] =-1.0f; + dest[3][2] = farZ * nv * fn; +} + +/*! + * @brief set up perspective projection matrix + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] fovy field of view angle + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping planes + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_perspective_rh_no(float fovy, + float aspect, + float nearZ, + float farZ, + mat4 dest) { + float f, fn; + + glm_mat4_zero(dest); + + f = 1.0f / tanf(fovy * 0.5f); + fn = 1.0f / (nearZ - farZ); + + dest[0][0] = f / aspect; + dest[1][1] = f; + dest[2][2] = (nearZ + farZ) * fn; + dest[2][3] =-1.0f; + dest[3][2] = 2.0f * nearZ * farZ * fn; + +} + +/*! + * @brief set up perspective projection matrix with default near/far + * and angle values with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_perspective_default_rh_no(float aspect, mat4 dest) { + glm_perspective_rh_no(GLM_PI_4f, aspect, 0.01f, 100.0f, dest); +} + +/*! + * @brief resize perspective matrix by aspect ratio ( width / height ) + * this makes very easy to resize proj matrix when window /viewport + * resized with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[in, out] proj perspective projection matrix + */ +CGLM_INLINE +void +glm_perspective_resize_rh_no(float aspect, mat4 proj) { + if (proj[0][0] == 0.0f) + return; + + proj[0][0] = proj[1][1] / aspect; +} + +/*! + * @brief extend perspective projection matrix's far distance + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * this function does not guarantee far >= near, be aware of that! + * + * @param[in, out] proj projection matrix to extend + * @param[in] deltaFar distance from existing far (negative to shink) + */ +CGLM_INLINE +void +glm_persp_move_far_rh_no(mat4 proj, float deltaFar) { + float fn, farZ, nearZ, p22, p32; + + p22 = proj[2][2]; + p32 = proj[3][2]; + + nearZ = p32 / (p22 - 1.0f); + farZ = p32 / (p22 + 1.0f) + deltaFar; + fn = 1.0f / (nearZ - farZ); + + proj[2][2] = (farZ + nearZ) * fn; + proj[3][2] = 2.0f * nearZ * farZ * fn; +} + +/*! + * @brief decomposes frustum values of perspective projection + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + * @param[out] top top + * @param[out] bottom bottom + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glm_persp_decomp_rh_no(mat4 proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right) { + float m00, m11, m20, m21, m22, m32, n, f; + float n_m11, n_m00; + + m00 = proj[0][0]; + m11 = proj[1][1]; + m20 = proj[2][0]; + m21 = proj[2][1]; + m22 = proj[2][2]; + m32 = proj[3][2]; + + n = m32 / (m22 - 1.0f); + f = m32 / (m22 + 1.0f); + + n_m11 = n / m11; + n_m00 = n / m00; + + *nearZ = n; + *farZ = f; + *bottom = n_m11 * (m21 - 1.0f); + *top = n_m11 * (m21 + 1.0f); + *left = n_m00 * (m20 - 1.0f); + *right = n_m00 * (m20 + 1.0f); +} + +/*! + * @brief decomposes frustum values of perspective projection + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * this makes easy to get all values at once + * + * @param[in] proj perspective projection matrix + * @param[out] dest array + */ +CGLM_INLINE +void +glm_persp_decompv_rh_no(mat4 proj, float dest[6]) { + glm_persp_decomp_rh_no(proj, &dest[0], &dest[1], &dest[2], + &dest[3], &dest[4], &dest[5]); +} + +/*! + * @brief decomposes left and right values of perspective projection + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * x stands for x axis (left / right axis) + * + * @param[in] proj perspective projection matrix + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glm_persp_decomp_x_rh_no(mat4 proj, + float * __restrict left, + float * __restrict right) { + float nearZ, m20, m00, m22; + + m00 = proj[0][0]; + m20 = proj[2][0]; + m22 = proj[2][2]; + + nearZ = proj[3][2] / (m22 - 1.0f); + *left = nearZ * (m20 - 1.0f) / m00; + *right = nearZ * (m20 + 1.0f) / m00; +} + +/*! + * @brief decomposes top and bottom values of perspective projection + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * y stands for y axis (top / bottom axis) + * + * @param[in] proj perspective projection matrix + * @param[out] top top + * @param[out] bottom bottom + */ +CGLM_INLINE +void +glm_persp_decomp_y_rh_no(mat4 proj, + float * __restrict top, + float * __restrict bottom) { + float nearZ, m21, m11, m22; + + m21 = proj[2][1]; + m11 = proj[1][1]; + m22 = proj[2][2]; + + nearZ = proj[3][2] / (m22 - 1.0f); + *bottom = nearZ * (m21 - 1.0f) / m11; + *top = nearZ * (m21 + 1.0f) / m11; +} + +/*! + * @brief decomposes near and far values of perspective projection + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * z stands for z axis (near / far axis) + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + */ +CGLM_INLINE +void +glm_persp_decomp_z_rh_no(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ) { + float m32, m22; + + m32 = proj[3][2]; + m22 = proj[2][2]; + + *nearZ = m32 / (m22 - 1.0f); + *farZ = m32 / (m22 + 1.0f); +} + +/*! + * @brief decomposes far value of perspective projection + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] farZ far + */ +CGLM_INLINE +void +glm_persp_decomp_far_rh_no(mat4 proj, float * __restrict farZ) { + *farZ = proj[3][2] / (proj[2][2] + 1.0f); +} + +/*! + * @brief decomposes near value of perspective projection + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + */ +CGLM_INLINE +void +glm_persp_decomp_near_rh_no(mat4 proj, float * __restrict nearZ) { + *nearZ = proj[3][2] / (proj[2][2] - 1.0f); +} + +/*! + * @brief returns sizes of near and far planes of perspective projection + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[in] fovy fovy (see brief) + * @param[out] dest sizes order: [Wnear, Hnear, Wfar, Hfar] + */ +CGLM_INLINE +void +glm_persp_sizes_rh_no(mat4 proj, float fovy, vec4 dest) { + float t, a, nearZ, farZ; + + t = 2.0f * tanf(fovy * 0.5f); + a = glm_persp_aspect(proj); + + glm_persp_decomp_z_rh_no(proj, &nearZ, &farZ); + + dest[1] = t * nearZ; + dest[3] = t * farZ; + dest[0] = a * dest[1]; + dest[2] = a * dest[3]; +} + +/*! + * @brief returns field of view angle along the Y-axis (in radians) + * with a right-hand coordinate system and a clip-space of [-1, 1]. + * + * if you need to degrees, use glm_deg to convert it or use this: + * fovy_deg = glm_deg(glm_persp_fovy(projMatrix)) + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glm_persp_fovy_rh_no(mat4 proj) { + return glm_persp_fovy(proj); +} + +/*! + * @brief returns aspect ratio of perspective projection + * with a right-hand coordinate system and a clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glm_persp_aspect_rh_no(mat4 proj) { + return glm_persp_aspect(proj); +} + +#endif /*cglm_cam_rh_no_h*/ diff --git a/cglm/include/cglm/clipspace/persp_rh_zo.h b/cglm/include/cglm/clipspace/persp_rh_zo.h new file mode 100644 index 0000000..ce632b3 --- /dev/null +++ b/cglm/include/cglm/clipspace/persp_rh_zo.h @@ -0,0 +1,389 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_frustum_rh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) + CGLM_INLINE void glm_perspective_rh_zo(float fovy, + float aspect, + float nearZ, + float farZ, + mat4 dest) + CGLM_INLINE void glm_perspective_default_rh_zo(float aspect, mat4 dest) + CGLM_INLINE void glm_perspective_resize_rh_zo(float aspect, mat4 proj) + CGLM_INLINE void glm_persp_move_far_rh_zo(mat4 proj, + float deltaFar) + CGLM_INLINE void glm_persp_decomp_rh_zo(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ, + float * __restrict top, + float * __restrict bottom, + float * __restrict left, + float * __restrict right) + CGLM_INLINE void glm_persp_decompv_rh_zo(mat4 proj, + float dest[6]) + CGLM_INLINE void glm_persp_decomp_x_rh_zo(mat4 proj, + float * __restrict left, + float * __restrict right) + CGLM_INLINE void glm_persp_decomp_y_rh_zo(mat4 proj, + float * __restrict top, + float * __restrict bottom) + CGLM_INLINE void glm_persp_decomp_z_rh_zo(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ) + CGLM_INLINE void glm_persp_decomp_far_rh_zo(mat4 proj, float * __restrict farZ) + CGLM_INLINE void glm_persp_decomp_near_rh_zo(mat4 proj, float * __restrict nearZ) + CGLM_INLINE void glm_persp_sizes_rh_zo(mat4 proj, float fovy, vec4 dest) + */ + +#ifndef cglm_persp_rh_zo_h +#define cglm_persp_rh_zo_h + +#include "../common.h" +#include "persp.h" + +/*! + * @brief set up perspective peprojection matrix with a right-hand coordinate + * system and a clip-space of [0, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_frustum_rh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + float rl, tb, fn, nv; + + glm_mat4_zero(dest); + + rl = 1.0f / (right - left); + tb = 1.0f / (top - bottom); + fn =-1.0f / (farZ - nearZ); + nv = 2.0f * nearZ; + + dest[0][0] = nv * rl; + dest[1][1] = nv * tb; + dest[2][0] = (right + left) * rl; + dest[2][1] = (top + bottom) * tb; + dest[2][2] = farZ * fn; + dest[2][3] =-1.0f; + dest[3][2] = farZ * nearZ * fn; +} + +/*! + * @brief set up perspective projection matrix with a right-hand coordinate + * system and a clip-space of [0, 1]. + * + * @param[in] fovy field of view angle + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping planes + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_perspective_rh_zo(float fovy, + float aspect, + float nearZ, + float farZ, + mat4 dest) { + float f, fn; + + glm_mat4_zero(dest); + + f = 1.0f / tanf(fovy * 0.5f); + fn = 1.0f / (nearZ - farZ); + + dest[0][0] = f / aspect; + dest[1][1] = f; + dest[2][2] = farZ * fn; + dest[2][3] =-1.0f; + dest[3][2] = nearZ * farZ * fn; +} + +/*! + * @brief set up perspective projection matrix with default near/far + * and angle values with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_perspective_default_rh_zo(float aspect, mat4 dest) { + glm_perspective_rh_zo(GLM_PI_4f, aspect, 0.01f, 100.0f, dest); +} + +/*! + * @brief resize perspective matrix by aspect ratio ( width / height ) + * this makes very easy to resize proj matrix when window /viewport + * resized with a right-hand coordinate system and a clip-space of + * [0, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[in, out] proj perspective projection matrix + */ +CGLM_INLINE +void +glm_perspective_resize_rh_zo(float aspect, mat4 proj) { + if (proj[0][0] == 0.0f) + return; + + proj[0][0] = proj[1][1] / aspect; +} + +/*! + * @brief extend perspective projection matrix's far distance with a + * right-hand coordinate system and a clip-space of [0, 1]. + * + * this function does not guarantee far >= near, be aware of that! + * + * @param[in, out] proj projection matrix to extend + * @param[in] deltaFar distance from existing far (negative to shink) + */ +CGLM_INLINE +void +glm_persp_move_far_rh_zo(mat4 proj, float deltaFar) { + float fn, farZ, nearZ, p22, p32; + + p22 = proj[2][2]; + p32 = proj[3][2]; + + nearZ = p32 / p22; + farZ = p32 / (p22 + 1.0f) + deltaFar; + fn = 1.0f / (nearZ - farZ); + + proj[2][2] = farZ * fn; + proj[3][2] = nearZ * farZ * fn; +} + +/*! + * @brief decomposes frustum values of perspective projection + * with angle values with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + * @param[out] top top + * @param[out] bottom bottom + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glm_persp_decomp_rh_zo(mat4 proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right) { + float m00, m11, m20, m21, m22, m32, n, f; + float n_m11, n_m00; + + m00 = proj[0][0]; + m11 = proj[1][1]; + m20 = proj[2][0]; + m21 = proj[2][1]; + m22 = proj[2][2]; + m32 = proj[3][2]; + + n = m32 / m22; + f = m32 / (m22 + 1.0f); + + n_m11 = n / m11; + n_m00 = n / m00; + + *nearZ = n; + *farZ = f; + *bottom = n_m11 * (m21 - 1.0f); + *top = n_m11 * (m21 + 1.0f); + *left = n_m00 * (m20 - 1.0f); + *right = n_m00 * (m20 + 1.0f); +} + +/*! + * @brief decomposes frustum values of perspective projection + * with angle values with a right-hand coordinate system and a + * clip-space of [0, 1]. + * this makes easy to get all values at once + * + * @param[in] proj perspective projection matrix + * @param[out] dest array + */ +CGLM_INLINE +void +glm_persp_decompv_rh_zo(mat4 proj, float dest[6]) { + glm_persp_decomp_rh_zo(proj, &dest[0], &dest[1], &dest[2], + &dest[3], &dest[4], &dest[5]); +} + +/*! + * @brief decomposes left and right values of perspective projection (ZO). + * x stands for x axis (left / right axis) + * + * @param[in] proj perspective projection matrix + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glm_persp_decomp_x_rh_zo(mat4 proj, + float * __restrict left, + float * __restrict right) { + float nearZ, m20, m00, m22; + + m00 = proj[0][0]; + m20 = proj[2][0]; + m22 = proj[2][2]; + + nearZ = proj[3][2] / m22; + *left = nearZ * (m20 - 1.0f) / m00; + *right = nearZ * (m20 + 1.0f) / m00; +} + +/*! + * @brief decomposes top and bottom values of perspective projection + * with angle values with a right-hand coordinate system and a + * clip-space of [0, 1]. + * y stands for y axis (top / bottom axis) + * + * @param[in] proj perspective projection matrix + * @param[out] top top + * @param[out] bottom bottom + */ +CGLM_INLINE +void +glm_persp_decomp_y_rh_zo(mat4 proj, + float * __restrict top, + float * __restrict bottom) { + float nearZ, m21, m11, m22; + + m21 = proj[2][1]; + m11 = proj[1][1]; + m22 = proj[2][2]; + + nearZ = proj[3][2] / m22; + *bottom = nearZ * (m21 - 1) / m11; + *top = nearZ * (m21 + 1) / m11; +} + +/*! + * @brief decomposes near and far values of perspective projection + * with angle values with a right-hand coordinate system and a + * clip-space of [0, 1]. + * z stands for z axis (near / far axis) + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + */ +CGLM_INLINE +void +glm_persp_decomp_z_rh_zo(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ) { + float m32, m22; + + m32 = proj[3][2]; + m22 = proj[2][2]; + + *nearZ = m32 / m22; + *farZ = m32 / (m22 + 1.0f); +} + +/*! + * @brief decomposes far value of perspective projection + * with angle values with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] farZ far + */ +CGLM_INLINE +void +glm_persp_decomp_far_rh_zo(mat4 proj, float * __restrict farZ) { + *farZ = proj[3][2] / (proj[2][2] + 1.0f); +} + +/*! + * @brief decomposes near value of perspective projection + * with angle values with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + */ +CGLM_INLINE +void +glm_persp_decomp_near_rh_zo(mat4 proj, float * __restrict nearZ) { + *nearZ = proj[3][2] / proj[2][2]; +} + +/*! + * @brief returns sizes of near and far planes of perspective projection + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[in] fovy fovy (see brief) + * @param[out] dest sizes order: [Wnear, Hnear, Wfar, Hfar] + */ +CGLM_INLINE +void +glm_persp_sizes_rh_zo(mat4 proj, float fovy, vec4 dest) { + float t, a, nearZ, farZ; + + t = 2.0f * tanf(fovy * 0.5f); + a = glm_persp_aspect(proj); + + glm_persp_decomp_z_rh_zo(proj, &nearZ, &farZ); + + dest[1] = t * nearZ; + dest[3] = t * farZ; + dest[0] = a * dest[1]; + dest[2] = a * dest[3]; +} + +/*! + * @brief returns field of view angle along the Y-axis (in radians) + * with a right-hand coordinate system and a clip-space of [0, 1]. + * + * if you need to degrees, use glm_deg to convert it or use this: + * fovy_deg = glm_deg(glm_persp_fovy(projMatrix)) + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glm_persp_fovy_rh_zo(mat4 proj) { + return glm_persp_fovy(proj); +} + +/*! + * @brief returns aspect ratio of perspective projection + * with a right-hand coordinate system and a clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glm_persp_aspect_rh_zo(mat4 proj) { + return glm_persp_aspect(proj); +} + +#endif /*cglm_persp_rh_zo_h*/ diff --git a/cglm/include/cglm/clipspace/project_no.h b/cglm/include/cglm/clipspace/project_no.h new file mode 100644 index 0000000..71fbc52 --- /dev/null +++ b/cglm/include/cglm/clipspace/project_no.h @@ -0,0 +1,109 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_project_no_h +#define cglm_project_no_h + +#include "../common.h" +#include "../vec3.h" +#include "../vec4.h" +#include "../mat4.h" + +/*! + * @brief maps the specified viewport coordinates into specified space [1] + * the matrix should contain projection matrix. + * + * if you don't have ( and don't want to have ) an inverse matrix then use + * glm_unproject version. You may use existing inverse of matrix in somewhere + * else, this is why glm_unprojecti exists to save save inversion cost + * + * [1] space: + * 1- if m = invProj: View Space + * 2- if m = invViewProj: World Space + * 3- if m = invMVP: Object Space + * + * You probably want to map the coordinates into object space + * so use invMVP as m + * + * Computing viewProj: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * glm_mat4_inv(viewProj, invMVP); + * + * @param[in] pos point/position in viewport coordinates + * @param[in] invMat matrix (see brief) + * @param[in] vp viewport as [x, y, width, height] + * @param[out] dest unprojected coordinates + */ +CGLM_INLINE +void +glm_unprojecti_no(vec3 pos, mat4 invMat, vec4 vp, vec3 dest) { + vec4 v; + + v[0] = 2.0f * (pos[0] - vp[0]) / vp[2] - 1.0f; + v[1] = 2.0f * (pos[1] - vp[1]) / vp[3] - 1.0f; + v[2] = 2.0f * pos[2] - 1.0f; + v[3] = 1.0f; + + glm_mat4_mulv(invMat, v, v); + glm_vec4_scale(v, 1.0f / v[3], v); + glm_vec3(v, dest); +} + +/*! + * @brief map object coordinates to window coordinates + * + * Computing MVP: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * + * @param[in] pos object coordinates + * @param[in] m MVP matrix + * @param[in] vp viewport as [x, y, width, height] + * @param[out] dest projected coordinates + */ +CGLM_INLINE +void +glm_project_no(vec3 pos, mat4 m, vec4 vp, vec3 dest) { + CGLM_ALIGN(16) vec4 pos4; + + glm_vec4(pos, 1.0f, pos4); + + glm_mat4_mulv(m, pos4, pos4); + glm_vec4_scale(pos4, 1.0f / pos4[3], pos4); /* pos = pos / pos.w */ + glm_vec4_scale(pos4, 0.5f, pos4); + glm_vec4_adds(pos4, 0.5f, pos4); + + dest[0] = pos4[0] * vp[2] + vp[0]; + dest[1] = pos4[1] * vp[3] + vp[1]; + dest[2] = pos4[2]; +} + +/*! + * @brief map object's z coordinate to window coordinates + * + * Computing MVP: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * + * @param[in] v object coordinates + * @param[in] m MVP matrix + * + * @returns projected z coordinate + */ +CGLM_INLINE +float +glm_project_z_no(vec3 v, mat4 m) { + float z, w; + + z = m[0][2] * v[0] + m[1][2] * v[1] + m[2][2] * v[2] + m[3][2]; + w = m[0][3] * v[0] + m[1][3] * v[1] + m[2][3] * v[2] + m[3][3]; + + return 0.5f * (z / w) + 0.5f; +} + +#endif /* cglm_project_no_h */ diff --git a/cglm/include/cglm/clipspace/project_zo.h b/cglm/include/cglm/clipspace/project_zo.h new file mode 100644 index 0000000..dc32078 --- /dev/null +++ b/cglm/include/cglm/clipspace/project_zo.h @@ -0,0 +1,111 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_project_zo_h +#define cglm_project_zo_h + +#include "../common.h" +#include "../vec3.h" +#include "../vec4.h" +#include "../mat4.h" + +/*! + * @brief maps the specified viewport coordinates into specified space [1] + * the matrix should contain projection matrix. + * + * if you don't have ( and don't want to have ) an inverse matrix then use + * glm_unproject version. You may use existing inverse of matrix in somewhere + * else, this is why glm_unprojecti exists to save save inversion cost + * + * [1] space: + * 1- if m = invProj: View Space + * 2- if m = invViewProj: World Space + * 3- if m = invMVP: Object Space + * + * You probably want to map the coordinates into object space + * so use invMVP as m + * + * Computing viewProj: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * glm_mat4_inv(viewProj, invMVP); + * + * @param[in] pos point/position in viewport coordinates + * @param[in] invMat matrix (see brief) + * @param[in] vp viewport as [x, y, width, height] + * @param[out] dest unprojected coordinates + */ +CGLM_INLINE +void +glm_unprojecti_zo(vec3 pos, mat4 invMat, vec4 vp, vec3 dest) { + vec4 v; + + v[0] = 2.0f * (pos[0] - vp[0]) / vp[2] - 1.0f; + v[1] = 2.0f * (pos[1] - vp[1]) / vp[3] - 1.0f; + v[2] = pos[2]; + v[3] = 1.0f; + + glm_mat4_mulv(invMat, v, v); + glm_vec4_scale(v, 1.0f / v[3], v); + glm_vec3(v, dest); +} + +/*! + * @brief map object coordinates to window coordinates + * + * Computing MVP: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * + * @param[in] pos object coordinates + * @param[in] m MVP matrix + * @param[in] vp viewport as [x, y, width, height] + * @param[out] dest projected coordinates + */ +CGLM_INLINE +void +glm_project_zo(vec3 pos, mat4 m, vec4 vp, vec3 dest) { + CGLM_ALIGN(16) vec4 pos4; + + glm_vec4(pos, 1.0f, pos4); + + glm_mat4_mulv(m, pos4, pos4); + glm_vec4_scale(pos4, 1.0f / pos4[3], pos4); /* pos = pos / pos.w */ + + dest[2] = pos4[2]; + + glm_vec4_scale(pos4, 0.5f, pos4); + glm_vec4_adds(pos4, 0.5f, pos4); + + dest[0] = pos4[0] * vp[2] + vp[0]; + dest[1] = pos4[1] * vp[3] + vp[1]; +} + +/*! + * @brief map object's z coordinate to window coordinates + * + * Computing MVP: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * + * @param[in] v object coordinates + * @param[in] m MVP matrix + * + * @returns projected z coordinate + */ +CGLM_INLINE +float +glm_project_z_zo(vec3 v, mat4 m) { + float z, w; + + z = m[0][2] * v[0] + m[1][2] * v[1] + m[2][2] * v[2] + m[3][2]; + w = m[0][3] * v[0] + m[1][3] * v[1] + m[2][3] * v[2] + m[3][3]; + + return z / w; +} + +#endif /* cglm_project_zo_h */ diff --git a/cglm/include/cglm/clipspace/view_lh.h b/cglm/include/cglm/clipspace/view_lh.h new file mode 100644 index 0000000..5667694 --- /dev/null +++ b/cglm/include/cglm/clipspace/view_lh.h @@ -0,0 +1,99 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_lookat_lh(vec3 eye, vec3 center, vec3 up, mat4 dest) + CGLM_INLINE void glm_look_lh(vec3 eye, vec3 dir, vec3 up, mat4 dest) + CGLM_INLINE void glm_look_anyup_lh(vec3 eye, vec3 dir, mat4 dest) + */ + +#ifndef cglm_view_lh_h +#define cglm_view_lh_h + +#include "../common.h" +#include "../plane.h" + +/*! + * @brief set up view matrix (LH) + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] center center vector + * @param[in] up up vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_lookat_lh(vec3 eye, vec3 center, vec3 up, mat4 dest) { + CGLM_ALIGN(8) vec3 f, u, s; + + glm_vec3_sub(center, eye, f); + glm_vec3_normalize(f); + + glm_vec3_crossn(up, f, s); + glm_vec3_cross(f, s, u); + + dest[0][0] = s[0]; + dest[0][1] = u[0]; + dest[0][2] = f[0]; + dest[1][0] = s[1]; + dest[1][1] = u[1]; + dest[1][2] = f[1]; + dest[2][0] = s[2]; + dest[2][1] = u[2]; + dest[2][2] = f[2]; + dest[3][0] =-glm_vec3_dot(s, eye); + dest[3][1] =-glm_vec3_dot(u, eye); + dest[3][2] =-glm_vec3_dot(f, eye); + dest[0][3] = dest[1][3] = dest[2][3] = 0.0f; + dest[3][3] = 1.0f; +} + +/*! + * @brief set up view matrix with left handed coordinate system + * + * convenient wrapper for lookat: if you only have direction not target self + * then this might be useful. Because you need to get target from direction. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[in] up up vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_look_lh(vec3 eye, vec3 dir, vec3 up, mat4 dest) { + CGLM_ALIGN(8) vec3 target; + glm_vec3_add(eye, dir, target); + glm_lookat_lh(eye, target, up, dest); +} + +/*! + * @brief set up view matrix with left handed coordinate system + * + * convenient wrapper for look: if you only have direction and if you don't + * care what UP vector is then this might be useful to create view matrix + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_look_anyup_lh(vec3 eye, vec3 dir, mat4 dest) { + CGLM_ALIGN(8) vec3 up; + glm_vec3_ortho(dir, up); + glm_look_lh(eye, dir, up, dest); +} + +#endif /*cglm_view_lh_h*/ diff --git a/cglm/include/cglm/clipspace/view_lh_no.h b/cglm/include/cglm/clipspace/view_lh_no.h new file mode 100644 index 0000000..454d903 --- /dev/null +++ b/cglm/include/cglm/clipspace/view_lh_no.h @@ -0,0 +1,74 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_lookat_lh_no(vec3 eye, vec3 center, vec3 up, mat4 dest) + CGLM_INLINE void glm_look_lh_no(vec3 eye, vec3 dir, vec3 up, mat4 dest) + CGLM_INLINE void glm_look_anyup_lh_no(vec3 eye, vec3 dir, mat4 dest) + */ + +#ifndef cglm_view_lh_no_h +#define cglm_view_lh_no_h + +#include "../common.h" +#include "view_lh.h" + +/*! + * @brief set up view matrix with left handed coordinate system. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] center center vector + * @param[in] up up vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_lookat_lh_no(vec3 eye, vec3 center, vec3 up, mat4 dest) { + glm_lookat_lh(eye, center, up, dest); +} + +/*! + * @brief set up view matrix with left handed coordinate system. + * + * convenient wrapper for lookat: if you only have direction not target self + * then this might be useful. Because you need to get target from direction. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[in] up up vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_look_lh_no(vec3 eye, vec3 dir, vec3 up, mat4 dest) { + glm_look_lh(eye, dir, up, dest); +} + +/*! + * @brief set up view matrix with left handed coordinate system. + * + * convenient wrapper for look: if you only have direction and if you don't + * care what UP vector is then this might be useful to create view matrix + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_look_anyup_lh_no(vec3 eye, vec3 dir, mat4 dest) { + glm_look_anyup_lh(eye, dir, dest); +} + +#endif /*cglm_view_lh_no_h*/ diff --git a/cglm/include/cglm/clipspace/view_lh_zo.h b/cglm/include/cglm/clipspace/view_lh_zo.h new file mode 100644 index 0000000..6b0c4d1 --- /dev/null +++ b/cglm/include/cglm/clipspace/view_lh_zo.h @@ -0,0 +1,74 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_lookat_lh_zo(vec3 eye, vec3 center, vec3 up, mat4 dest) + CGLM_INLINE void glm_look_lh_zo(vec3 eye, vec3 dir, vec3 up, mat4 dest) + CGLM_INLINE void glm_look_anyup_lh_zo(vec3 eye, vec3 dir, mat4 dest) + */ + +#ifndef cglm_view_lh_zo_h +#define cglm_view_lh_zo_h + +#include "../common.h" +#include "view_lh.h" + +/*! + * @brief set up view matrix with left handed coordinate system. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] center center vector + * @param[in] up up vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_lookat_lh_zo(vec3 eye, vec3 center, vec3 up, mat4 dest) { + glm_lookat_lh(eye, center, up, dest); +} + +/*! + * @brief set up view matrix with left handed coordinate system. + * + * convenient wrapper for lookat: if you only have direction not target self + * then this might be useful. Because you need to get target from direction. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[in] up up vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_look_lh_zo(vec3 eye, vec3 dir, vec3 up, mat4 dest) { + glm_look_lh(eye, dir, up, dest); +} + +/*! + * @brief set up view matrix with left handed coordinate system. + * + * convenient wrapper for look: if you only have direction and if you don't + * care what UP vector is then this might be useful to create view matrix + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_look_anyup_lh_zo(vec3 eye, vec3 dir, mat4 dest) { + glm_look_anyup_lh(eye, dir, dest); +} + +#endif /*cglm_view_lh_zo_h*/ diff --git a/cglm/include/cglm/clipspace/view_rh.h b/cglm/include/cglm/clipspace/view_rh.h new file mode 100644 index 0000000..51ec916 --- /dev/null +++ b/cglm/include/cglm/clipspace/view_rh.h @@ -0,0 +1,99 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_lookat_rh(vec3 eye, vec3 center, vec3 up, mat4 dest) + CGLM_INLINE void glm_look_rh(vec3 eye, vec3 dir, vec3 up, mat4 dest) + CGLM_INLINE void glm_look_anyup_rh(vec3 eye, vec3 dir, mat4 dest) + */ + +#ifndef cglm_view_rh_h +#define cglm_view_rh_h + +#include "../common.h" +#include "../plane.h" + +/*! + * @brief set up view matrix with right handed coordinate system. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] center center vector + * @param[in] up up vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_lookat_rh(vec3 eye, vec3 center, vec3 up, mat4 dest) { + CGLM_ALIGN(8) vec3 f, u, s; + + glm_vec3_sub(center, eye, f); + glm_vec3_normalize(f); + + glm_vec3_crossn(f, up, s); + glm_vec3_cross(s, f, u); + + dest[0][0] = s[0]; + dest[0][1] = u[0]; + dest[0][2] =-f[0]; + dest[1][0] = s[1]; + dest[1][1] = u[1]; + dest[1][2] =-f[1]; + dest[2][0] = s[2]; + dest[2][1] = u[2]; + dest[2][2] =-f[2]; + dest[3][0] =-glm_vec3_dot(s, eye); + dest[3][1] =-glm_vec3_dot(u, eye); + dest[3][2] = glm_vec3_dot(f, eye); + dest[0][3] = dest[1][3] = dest[2][3] = 0.0f; + dest[3][3] = 1.0f; +} + +/*! + * @brief set up view matrix with right handed coordinate system. + * + * convenient wrapper for lookat: if you only have direction not target self + * then this might be useful. Because you need to get target from direction. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[in] up up vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_look_rh(vec3 eye, vec3 dir, vec3 up, mat4 dest) { + CGLM_ALIGN(8) vec3 target; + glm_vec3_add(eye, dir, target); + glm_lookat_rh(eye, target, up, dest); +} + +/*! + * @brief set up view matrix with right handed coordinate system. + * + * convenient wrapper for look: if you only have direction and if you don't + * care what UP vector is then this might be useful to create view matrix + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_look_anyup_rh(vec3 eye, vec3 dir, mat4 dest) { + CGLM_ALIGN(8) vec3 up; + glm_vec3_ortho(dir, up); + glm_look_rh(eye, dir, up, dest); +} + +#endif /*cglm_view_rh_h*/ diff --git a/cglm/include/cglm/clipspace/view_rh_no.h b/cglm/include/cglm/clipspace/view_rh_no.h new file mode 100644 index 0000000..ca36d30 --- /dev/null +++ b/cglm/include/cglm/clipspace/view_rh_no.h @@ -0,0 +1,74 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_lookat_rh_no(vec3 eye, vec3 center, vec3 up, mat4 dest) + CGLM_INLINE void glm_look_rh_no(vec3 eye, vec3 dir, vec3 up, mat4 dest) + CGLM_INLINE void glm_look_anyup_rh_no(vec3 eye, vec3 dir, mat4 dest) + */ + +#ifndef cglm_view_rh_no_h +#define cglm_view_rh_no_h + +#include "../common.h" +#include "view_rh.h" + +/*! + * @brief set up view matrix with right handed coordinate system. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] center center vector + * @param[in] up up vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_lookat_rh_no(vec3 eye, vec3 center, vec3 up, mat4 dest) { + glm_lookat_rh(eye, center, up, dest); +} + +/*! + * @brief set up view matrix with right handed coordinate system. + * + * convenient wrapper for lookat: if you only have direction not target self + * then this might be useful. Because you need to get target from direction. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[in] up up vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_look_rh_no(vec3 eye, vec3 dir, vec3 up, mat4 dest) { + glm_look_rh(eye, dir, up, dest); +} + +/*! + * @brief set up view matrix with right handed coordinate system. + * + * convenient wrapper for look: if you only have direction and if you don't + * care what UP vector is then this might be useful to create view matrix + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_look_anyup_rh_no(vec3 eye, vec3 dir, mat4 dest) { + glm_look_anyup_rh(eye, dir, dest); +} + +#endif /*cglm_view_rh_no_h*/ diff --git a/cglm/include/cglm/clipspace/view_rh_zo.h b/cglm/include/cglm/clipspace/view_rh_zo.h new file mode 100644 index 0000000..1ad5c91 --- /dev/null +++ b/cglm/include/cglm/clipspace/view_rh_zo.h @@ -0,0 +1,74 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_lookat_rh_zo(vec3 eye, vec3 center, vec3 up, mat4 dest) + CGLM_INLINE void glm_look_rh_zo(vec3 eye, vec3 dir, vec3 up, mat4 dest) + CGLM_INLINE void glm_look_anyup_rh_zo(vec3 eye, vec3 dir, mat4 dest) + */ + +#ifndef cglm_view_rh_zo_h +#define cglm_view_rh_zo_h + +#include "../common.h" +#include "view_rh.h" + +/*! + * @brief set up view matrix with right handed coordinate system. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] center center vector + * @param[in] up up vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_lookat_rh_zo(vec3 eye, vec3 center, vec3 up, mat4 dest) { + glm_lookat_rh(eye, center, up, dest); +} + +/*! + * @brief set up view matrix with right handed coordinate system. + * + * convenient wrapper for lookat: if you only have direction not target self + * then this might be useful. Because you need to get target from direction. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[in] up up vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_look_rh_zo(vec3 eye, vec3 dir, vec3 up, mat4 dest) { + glm_look_rh(eye, dir, up, dest); +} + +/*! + * @brief set up view matrix with right handed coordinate system. + * + * convenient wrapper for look: if you only have direction and if you don't + * care what UP vector is then this might be useful to create view matrix + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_look_anyup_rh_zo(vec3 eye, vec3 dir, mat4 dest) { + glm_look_anyup_rh(eye, dir, dest); +} + +#endif /*cglm_view_rh_zo_h*/ diff --git a/cglm/include/cglm/color.h b/cglm/include/cglm/color.h new file mode 100644 index 0000000..69566ad --- /dev/null +++ b/cglm/include/cglm/color.h @@ -0,0 +1,26 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_color_h +#define cglm_color_h + +#include "common.h" +#include "vec3.h" + +/*! + * @brief averages the color channels into one value + * + * @param[in] rgb RGB color + */ +CGLM_INLINE +float +glm_luminance(vec3 rgb) { + vec3 l = {0.212671f, 0.715160f, 0.072169f}; + return glm_dot(rgb, l); +} + +#endif /* cglm_color_h */ diff --git a/cglm/include/cglm/common.h b/cglm/include/cglm/common.h new file mode 100644 index 0000000..11588cf --- /dev/null +++ b/cglm/include/cglm/common.h @@ -0,0 +1,128 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_common_h +#define cglm_common_h + +#ifndef _USE_MATH_DEFINES +# define _USE_MATH_DEFINES /* for windows */ +#endif + +#ifndef _CRT_SECURE_NO_WARNINGS +# define _CRT_SECURE_NO_WARNINGS /* for windows */ +#endif + +#include +#include +#include +#include +#include +#include + +#if defined(_MSC_VER) +# ifdef CGLM_STATIC +# define CGLM_EXPORT +# elif defined(CGLM_EXPORTS) +# define CGLM_EXPORT __declspec(dllexport) +# else +# define CGLM_EXPORT __declspec(dllimport) +# endif +# define CGLM_INLINE __forceinline +#else +# define CGLM_EXPORT __attribute__((visibility("default"))) +# define CGLM_INLINE static inline __attribute((always_inline)) +#endif + +#if defined(__GNUC__) || defined(__clang__) +# define CGLM_UNLIKELY(expr) __builtin_expect(!!(expr), 0) +# define CGLM_LIKELY(expr) __builtin_expect(!!(expr), 1) +#else +# define CGLM_UNLIKELY(expr) (expr) +# define CGLM_LIKELY(expr) (expr) +#endif + +#if defined(_M_FP_FAST) || defined(__FAST_MATH__) +# define CGLM_FAST_MATH +#endif + +#define GLM_SHUFFLE4(z, y, x, w) (((z) << 6) | ((y) << 4) | ((x) << 2) | (w)) +#define GLM_SHUFFLE3(z, y, x) (((z) << 4) | ((y) << 2) | (x)) +#define GLM_SHUFFLE2(y, x) (((y) << 2) | (x)) + +#include "types.h" +#include "simd/intrin.h" + +#ifndef CGLM_USE_DEFAULT_EPSILON +# ifndef GLM_FLT_EPSILON +# define GLM_FLT_EPSILON 1e-5f +# endif +#else +# define GLM_FLT_EPSILON FLT_EPSILON +#endif + +/* + * Clip control: define CGLM_FORCE_DEPTH_ZERO_TO_ONE before including + * CGLM to use a clip space between 0 to 1. + * Coordinate system: define CGLM_FORCE_LEFT_HANDED before including + * CGLM to use the left handed coordinate system by default. + */ + +#define CGLM_CLIP_CONTROL_ZO_BIT (1 << 0) /* ZERO_TO_ONE */ +#define CGLM_CLIP_CONTROL_NO_BIT (1 << 1) /* NEGATIVE_ONE_TO_ONE */ +#define CGLM_CLIP_CONTROL_LH_BIT (1 << 2) /* LEFT_HANDED, For DirectX, Metal, Vulkan */ +#define CGLM_CLIP_CONTROL_RH_BIT (1 << 3) /* RIGHT_HANDED, For OpenGL, default in GLM */ + +#define CGLM_CLIP_CONTROL_LH_ZO (CGLM_CLIP_CONTROL_LH_BIT | CGLM_CLIP_CONTROL_ZO_BIT) +#define CGLM_CLIP_CONTROL_LH_NO (CGLM_CLIP_CONTROL_LH_BIT | CGLM_CLIP_CONTROL_NO_BIT) +#define CGLM_CLIP_CONTROL_RH_ZO (CGLM_CLIP_CONTROL_RH_BIT | CGLM_CLIP_CONTROL_ZO_BIT) +#define CGLM_CLIP_CONTROL_RH_NO (CGLM_CLIP_CONTROL_RH_BIT | CGLM_CLIP_CONTROL_NO_BIT) + +#ifdef CGLM_FORCE_DEPTH_ZERO_TO_ONE +# ifdef CGLM_FORCE_LEFT_HANDED +# define CGLM_CONFIG_CLIP_CONTROL CGLM_CLIP_CONTROL_LH_ZO +# else +# define CGLM_CONFIG_CLIP_CONTROL CGLM_CLIP_CONTROL_RH_ZO +# endif +#else +# ifdef CGLM_FORCE_LEFT_HANDED +# define CGLM_CONFIG_CLIP_CONTROL CGLM_CLIP_CONTROL_LH_NO +# else +# define CGLM_CONFIG_CLIP_CONTROL CGLM_CLIP_CONTROL_RH_NO +# endif +#endif + +/* struct API configurator */ +/* TODO: move struct/common.h? */ +/* WARN: dont use concant helpers outside cglm headers, because they may be changed */ + +#define CGLM_MACRO_CONCAT_HELPER(A, B, C, D, E, ...) A ## B ## C ## D ## E ## __VA_ARGS__ +#define CGLM_MACRO_CONCAT(A, B, C, D, E, ...) CGLM_MACRO_CONCAT_HELPER(A, B, C, D, E,__VA_ARGS__) + +#ifndef CGLM_OMIT_NS_FROM_STRUCT_API +# ifndef CGLM_STRUCT_API_NS +# define CGLM_STRUCT_API_NS glms +# endif +# ifndef CGLM_STRUCT_API_NS_SEPERATOR +# define CGLM_STRUCT_API_NS_SEPERATOR _ +# endif +#else +# define CGLM_STRUCT_API_NS +# define CGLM_STRUCT_API_NS_SEPERATOR +#endif + +#ifndef CGLM_STRUCT_API_NAME_SUFFIX +# define CGLM_STRUCT_API_NAME_SUFFIX +#endif + +#define CGLM_STRUCTAPI(A, ...) CGLM_MACRO_CONCAT(CGLM_STRUCT_API_NS, \ + CGLM_STRUCT_API_NS_SEPERATOR, \ + A, \ + CGLM_STRUCT_API_NAME_SUFFIX, \ + _, \ + __VA_ARGS__) + +#endif /* cglm_common_h */ diff --git a/cglm/include/cglm/curve.h b/cglm/include/cglm/curve.h new file mode 100644 index 0000000..5033be5 --- /dev/null +++ b/cglm/include/cglm/curve.h @@ -0,0 +1,40 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_curve_h +#define cglm_curve_h + +#include "common.h" +#include "vec4.h" +#include "mat4.h" + +/*! + * @brief helper function to calculate S*M*C multiplication for curves + * + * This function does not encourage you to use SMC, + * instead it is a helper if you use SMC. + * + * if you want to specify S as vector then use more generic glm_mat4_rmc() func. + * + * Example usage: + * B(s) = glm_smc(s, GLM_BEZIER_MAT, (vec4){p0, c0, c1, p1}) + * + * @param[in] s parameter between 0 and 1 (this will be [s3, s2, s, 1]) + * @param[in] m basis matrix + * @param[in] c position/control vector + * + * @return B(s) + */ +CGLM_INLINE +float +glm_smc(float s, mat4 m, vec4 c) { + vec4 vs; + glm_vec4_cubic(s, vs); + return glm_mat4_rmc(vs, m, c); +} + +#endif /* cglm_curve_h */ diff --git a/cglm/include/cglm/ease.h b/cglm/include/cglm/ease.h new file mode 100644 index 0000000..e26b48c --- /dev/null +++ b/cglm/include/cglm/ease.h @@ -0,0 +1,317 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_ease_h +#define cglm_ease_h + +#include "common.h" + +CGLM_INLINE +float +glm_ease_linear(float t) { + return t; +} + +CGLM_INLINE +float +glm_ease_sine_in(float t) { + return sinf((t - 1.0f) * GLM_PI_2f) + 1.0f; +} + +CGLM_INLINE +float +glm_ease_sine_out(float t) { + return sinf(t * GLM_PI_2f); +} + +CGLM_INLINE +float +glm_ease_sine_inout(float t) { + return 0.5f * (1.0f - cosf(t * GLM_PIf)); +} + +CGLM_INLINE +float +glm_ease_quad_in(float t) { + return t * t; +} + +CGLM_INLINE +float +glm_ease_quad_out(float t) { + return -(t * (t - 2.0f)); +} + +CGLM_INLINE +float +glm_ease_quad_inout(float t) { + float tt; + + tt = t * t; + if (t < 0.5f) + return 2.0f * tt; + + return (-2.0f * tt) + (4.0f * t) - 1.0f; +} + +CGLM_INLINE +float +glm_ease_cubic_in(float t) { + return t * t * t; +} + +CGLM_INLINE +float +glm_ease_cubic_out(float t) { + float f; + f = t - 1.0f; + return f * f * f + 1.0f; +} + +CGLM_INLINE +float +glm_ease_cubic_inout(float t) { + float f; + + if (t < 0.5f) + return 4.0f * t * t * t; + + f = 2.0f * t - 2.0f; + + return 0.5f * f * f * f + 1.0f; +} + +CGLM_INLINE +float +glm_ease_quart_in(float t) { + float f; + f = t * t; + return f * f; +} + +CGLM_INLINE +float +glm_ease_quart_out(float t) { + float f; + + f = t - 1.0f; + + return f * f * f * (1.0f - t) + 1.0f; +} + +CGLM_INLINE +float +glm_ease_quart_inout(float t) { + float f, g; + + if (t < 0.5f) { + f = t * t; + return 8.0f * f * f; + } + + f = t - 1.0f; + g = f * f; + + return -8.0f * g * g + 1.0f; +} + +CGLM_INLINE +float +glm_ease_quint_in(float t) { + float f; + f = t * t; + return f * f * t; +} + +CGLM_INLINE +float +glm_ease_quint_out(float t) { + float f, g; + + f = t - 1.0f; + g = f * f; + + return g * g * f + 1.0f; +} + +CGLM_INLINE +float +glm_ease_quint_inout(float t) { + float f, g; + + if (t < 0.5f) { + f = t * t; + return 16.0f * f * f * t; + } + + f = 2.0f * t - 2.0f; + g = f * f; + + return 0.5f * g * g * f + 1.0f; +} + +CGLM_INLINE +float +glm_ease_exp_in(float t) { + if (t == 0.0f) + return t; + + return powf(2.0f, 10.0f * (t - 1.0f)); +} + +CGLM_INLINE +float +glm_ease_exp_out(float t) { + if (t == 1.0f) + return t; + + return 1.0f - powf(2.0f, -10.0f * t); +} + +CGLM_INLINE +float +glm_ease_exp_inout(float t) { + if (t == 0.0f || t == 1.0f) + return t; + + if (t < 0.5f) + return 0.5f * powf(2.0f, (20.0f * t) - 10.0f); + + return -0.5f * powf(2.0f, (-20.0f * t) + 10.0f) + 1.0f; +} + +CGLM_INLINE +float +glm_ease_circ_in(float t) { + return 1.0f - sqrtf(1.0f - (t * t)); +} + +CGLM_INLINE +float +glm_ease_circ_out(float t) { + return sqrtf((2.0f - t) * t); +} + +CGLM_INLINE +float +glm_ease_circ_inout(float t) { + if (t < 0.5f) + return 0.5f * (1.0f - sqrtf(1.0f - 4.0f * (t * t))); + + return 0.5f * (sqrtf(-((2.0f * t) - 3.0f) * ((2.0f * t) - 1.0f)) + 1.0f); +} + +CGLM_INLINE +float +glm_ease_back_in(float t) { + float o, z; + + o = 1.70158f; + z = ((o + 1.0f) * t) - o; + + return t * t * z; +} + +CGLM_INLINE +float +glm_ease_back_out(float t) { + float o, z, n; + + o = 1.70158f; + n = t - 1.0f; + z = (o + 1.0f) * n + o; + + return n * n * z + 1.0f; +} + +CGLM_INLINE +float +glm_ease_back_inout(float t) { + float o, z, n, m, s, x; + + o = 1.70158f; + s = o * 1.525f; + x = 0.5f; + n = t / 0.5f; + + if (n < 1.0f) { + z = (s + 1) * n - s; + m = n * n * z; + return x * m; + } + + n -= 2.0f; + z = (s + 1.0f) * n + s; + m = (n * n * z) + 2; + + return x * m; +} + +CGLM_INLINE +float +glm_ease_elast_in(float t) { + return sinf(13.0f * GLM_PI_2f * t) * powf(2.0f, 10.0f * (t - 1.0f)); +} + +CGLM_INLINE +float +glm_ease_elast_out(float t) { + return sinf(-13.0f * GLM_PI_2f * (t + 1.0f)) * powf(2.0f, -10.0f * t) + 1.0f; +} + +CGLM_INLINE +float +glm_ease_elast_inout(float t) { + float a; + + a = 2.0f * t; + + if (t < 0.5f) + return 0.5f * sinf(13.0f * GLM_PI_2f * a) + * powf(2.0f, 10.0f * (a - 1.0f)); + + return 0.5f * (sinf(-13.0f * GLM_PI_2f * a) + * powf(2.0f, -10.0f * (a - 1.0f)) + 2.0f); +} + +CGLM_INLINE +float +glm_ease_bounce_out(float t) { + float tt; + + tt = t * t; + + if (t < (4.0f / 11.0f)) + return (121.0f * tt) / 16.0f; + + if (t < 8.0f / 11.0f) + return ((363.0f / 40.0f) * tt) - ((99.0f / 10.0f) * t) + (17.0f / 5.0f); + + if (t < (9.0f / 10.0f)) + return (4356.0f / 361.0f) * tt + - (35442.0f / 1805.0f) * t + + (16061.0f / 1805.0f); + + return ((54.0f / 5.0f) * tt) - ((513.0f / 25.0f) * t) + (268.0f / 25.0f); +} + +CGLM_INLINE +float +glm_ease_bounce_in(float t) { + return 1.0f - glm_ease_bounce_out(1.0f - t); +} + +CGLM_INLINE +float +glm_ease_bounce_inout(float t) { + if (t < 0.5f) + return 0.5f * (1.0f - glm_ease_bounce_out(t * 2.0f)); + + return 0.5f * glm_ease_bounce_out(t * 2.0f - 1.0f) + 0.5f; +} + +#endif /* cglm_ease_h */ diff --git a/cglm/include/cglm/euler.h b/cglm/include/cglm/euler.h new file mode 100644 index 0000000..8fae039 --- /dev/null +++ b/cglm/include/cglm/euler.h @@ -0,0 +1,601 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + NOTE: + angles must be passed as [X-Angle, Y-Angle, Z-angle] order + For instance you don't pass angles as [Z-Angle, X-Angle, Y-angle] to + glm_euler_zxy function, All RELATED functions accept angles same order + which is [X, Y, Z]. + */ + +/* + Types: + enum glm_euler_seq + + Functions: + CGLM_INLINE glm_euler_seq glm_euler_order(int newOrder[3]); + CGLM_INLINE void glm_euler_angles(mat4 m, vec3 dest); + CGLM_INLINE void glm_euler(vec3 angles, mat4 dest); + CGLM_INLINE void glm_euler_xyz(vec3 angles, mat4 dest); + CGLM_INLINE void glm_euler_zyx(vec3 angles, mat4 dest); + CGLM_INLINE void glm_euler_zxy(vec3 angles, mat4 dest); + CGLM_INLINE void glm_euler_xzy(vec3 angles, mat4 dest); + CGLM_INLINE void glm_euler_yzx(vec3 angles, mat4 dest); + CGLM_INLINE void glm_euler_yxz(vec3 angles, mat4 dest); + CGLM_INLINE void glm_euler_by_order(vec3 angles, + glm_euler_seq ord, + mat4 dest); + CGLM_INLINE void glm_euler_xyz_quat(vec3 angles, versor dest); + CGLM_INLINE void glm_euler_xzy_quat(vec3 angles, versor dest); + CGLM_INLINE void glm_euler_yxz_quat(vec3 angles, versor dest); + CGLM_INLINE void glm_euler_yzx_quat(vec3 angles, versor dest); + CGLM_INLINE void glm_euler_zxy_quat(vec3 angles, versor dest); + CGLM_INLINE void glm_euler_zyx_quat(vec3 angles, versor dest); + */ + +#ifndef cglm_euler_h +#define cglm_euler_h + +#include "common.h" + +#ifdef CGLM_FORCE_LEFT_HANDED +# include "handed/euler_to_quat_lh.h" +#else +# include "handed/euler_to_quat_rh.h" +#endif + + +#ifndef CGLM_CLIPSPACE_INCLUDE_ALL +# if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO +# include "clipspace/ortho_lh_zo.h" +# include "clipspace/persp_lh_zo.h" +# include "clipspace/view_lh_zo.h" +# elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO +# include "clipspace/ortho_lh_no.h" +# include "clipspace/persp_lh_no.h" +# include "clipspace/view_lh_no.h" +# elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO +# include "clipspace/ortho_rh_zo.h" +# include "clipspace/persp_rh_zo.h" +# include "clipspace/view_rh_zo.h" +# elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO +# include "clipspace/ortho_rh_no.h" +# include "clipspace/persp_rh_no.h" +# include "clipspace/view_rh_no.h" +# endif +#else +# include "clipspace/ortho_lh_zo.h" +# include "clipspace/persp_lh_zo.h" +# include "clipspace/ortho_lh_no.h" +# include "clipspace/persp_lh_no.h" +# include "clipspace/ortho_rh_zo.h" +# include "clipspace/persp_rh_zo.h" +# include "clipspace/ortho_rh_no.h" +# include "clipspace/persp_rh_no.h" +# include "clipspace/view_lh_zo.h" +# include "clipspace/view_lh_no.h" +# include "clipspace/view_rh_zo.h" +# include "clipspace/view_rh_no.h" +#endif + + +/*! + * if you have axis order like vec3 orderVec = [0, 1, 2] or [0, 2, 1]... + * vector then you can convert it to this enum by doing this: + * @code + * glm_euler_seq order; + * order = orderVec[0] | orderVec[1] << 2 | orderVec[2] << 4; + * @endcode + * you may need to explicit cast if required + */ +typedef enum glm_euler_seq { + GLM_EULER_XYZ = 0 << 0 | 1 << 2 | 2 << 4, + GLM_EULER_XZY = 0 << 0 | 2 << 2 | 1 << 4, + GLM_EULER_YZX = 1 << 0 | 2 << 2 | 0 << 4, + GLM_EULER_YXZ = 1 << 0 | 0 << 2 | 2 << 4, + GLM_EULER_ZXY = 2 << 0 | 0 << 2 | 1 << 4, + GLM_EULER_ZYX = 2 << 0 | 1 << 2 | 0 << 4 +} glm_euler_seq; + +CGLM_INLINE +glm_euler_seq +glm_euler_order(int ord[3]) { + return (glm_euler_seq)(ord[0] << 0 | ord[1] << 2 | ord[2] << 4); +} + +/*! + * @brief extract euler angles (in radians) using xyz order + * + * @param[in] m affine transform + * @param[out] dest angles vector [x, y, z] + */ +CGLM_INLINE +void +glm_euler_angles(mat4 m, vec3 dest) { + float m00, m01, m10, m11, m20, m21, m22; + float thetaX, thetaY, thetaZ; + + m00 = m[0][0]; m10 = m[1][0]; m20 = m[2][0]; + m01 = m[0][1]; m11 = m[1][1]; m21 = m[2][1]; + m22 = m[2][2]; + + if (m20 < 1.0f) { + if (m20 > -1.0f) { + thetaY = asinf(m20); + thetaX = atan2f(-m21, m22); + thetaZ = atan2f(-m10, m00); + } else { /* m20 == -1 */ + /* Not a unique solution */ + thetaY = -GLM_PI_2f; + thetaX = -atan2f(m01, m11); + thetaZ = 0.0f; + } + } else { /* m20 == +1 */ + thetaY = GLM_PI_2f; + thetaX = atan2f(m01, m11); + thetaZ = 0.0f; + } + + dest[0] = thetaX; + dest[1] = thetaY; + dest[2] = thetaZ; +} + +/*! + * @brief build rotation matrix from euler angles + * + * @param[in] angles angles as vector [Xangle, Yangle, Zangle] + * @param[out] dest rotation matrix + */ +CGLM_INLINE +void +glm_euler_xyz(vec3 angles, mat4 dest) { + float cx, cy, cz, + sx, sy, sz, czsx, cxcz, sysz; + + sx = sinf(angles[0]); cx = cosf(angles[0]); + sy = sinf(angles[1]); cy = cosf(angles[1]); + sz = sinf(angles[2]); cz = cosf(angles[2]); + + czsx = cz * sx; + cxcz = cx * cz; + sysz = sy * sz; + + dest[0][0] = cy * cz; + dest[0][1] = czsx * sy + cx * sz; + dest[0][2] = -cxcz * sy + sx * sz; + dest[1][0] = -cy * sz; + dest[1][1] = cxcz - sx * sysz; + dest[1][2] = czsx + cx * sysz; + dest[2][0] = sy; + dest[2][1] = -cy * sx; + dest[2][2] = cx * cy; + dest[0][3] = 0.0f; + dest[1][3] = 0.0f; + dest[2][3] = 0.0f; + dest[3][0] = 0.0f; + dest[3][1] = 0.0f; + dest[3][2] = 0.0f; + dest[3][3] = 1.0f; +} + +/*! + * @brief build rotation matrix from euler angles + * + * @param[in] angles angles as vector [Xangle, Yangle, Zangle] + * @param[out] dest rotation matrix + */ +CGLM_INLINE +void +glm_euler(vec3 angles, mat4 dest) { + glm_euler_xyz(angles, dest); +} + +/*! + * @brief build rotation matrix from euler angles + * + * @param[in] angles angles as vector [Xangle, Yangle, Zangle] + * @param[out] dest rotation matrix + */ +CGLM_INLINE +void +glm_euler_xzy(vec3 angles, mat4 dest) { + float cx, cy, cz, + sx, sy, sz, sxsy, cysx, cxsy, cxcy; + + sx = sinf(angles[0]); cx = cosf(angles[0]); + sy = sinf(angles[1]); cy = cosf(angles[1]); + sz = sinf(angles[2]); cz = cosf(angles[2]); + + sxsy = sx * sy; + cysx = cy * sx; + cxsy = cx * sy; + cxcy = cx * cy; + + dest[0][0] = cy * cz; + dest[0][1] = sxsy + cxcy * sz; + dest[0][2] = -cxsy + cysx * sz; + dest[1][0] = -sz; + dest[1][1] = cx * cz; + dest[1][2] = cz * sx; + dest[2][0] = cz * sy; + dest[2][1] = -cysx + cxsy * sz; + dest[2][2] = cxcy + sxsy * sz; + dest[0][3] = 0.0f; + dest[1][3] = 0.0f; + dest[2][3] = 0.0f; + dest[3][0] = 0.0f; + dest[3][1] = 0.0f; + dest[3][2] = 0.0f; + dest[3][3] = 1.0f; +} + +/*! + * @brief build rotation matrix from euler angles + * + * @param[in] angles angles as vector [Xangle, Yangle, Zangle] + * @param[out] dest rotation matrix + */ +CGLM_INLINE +void +glm_euler_yxz(vec3 angles, mat4 dest) { + float cx, cy, cz, + sx, sy, sz, cycz, sysz, czsy, cysz; + + sx = sinf(angles[0]); cx = cosf(angles[0]); + sy = sinf(angles[1]); cy = cosf(angles[1]); + sz = sinf(angles[2]); cz = cosf(angles[2]); + + cycz = cy * cz; + sysz = sy * sz; + czsy = cz * sy; + cysz = cy * sz; + + dest[0][0] = cycz + sx * sysz; + dest[0][1] = cx * sz; + dest[0][2] = -czsy + cysz * sx; + dest[1][0] = -cysz + czsy * sx; + dest[1][1] = cx * cz; + dest[1][2] = cycz * sx + sysz; + dest[2][0] = cx * sy; + dest[2][1] = -sx; + dest[2][2] = cx * cy; + dest[0][3] = 0.0f; + dest[1][3] = 0.0f; + dest[2][3] = 0.0f; + dest[3][0] = 0.0f; + dest[3][1] = 0.0f; + dest[3][2] = 0.0f; + dest[3][3] = 1.0f; +} + +/*! + * @brief build rotation matrix from euler angles + * + * @param[in] angles angles as vector [Xangle, Yangle, Zangle] + * @param[out] dest rotation matrix + */ +CGLM_INLINE +void +glm_euler_yzx(vec3 angles, mat4 dest) { + float cx, cy, cz, + sx, sy, sz, sxsy, cxcy, cysx, cxsy; + + sx = sinf(angles[0]); cx = cosf(angles[0]); + sy = sinf(angles[1]); cy = cosf(angles[1]); + sz = sinf(angles[2]); cz = cosf(angles[2]); + + sxsy = sx * sy; + cxcy = cx * cy; + cysx = cy * sx; + cxsy = cx * sy; + + dest[0][0] = cy * cz; + dest[0][1] = sz; + dest[0][2] = -cz * sy; + dest[1][0] = sxsy - cxcy * sz; + dest[1][1] = cx * cz; + dest[1][2] = cysx + cxsy * sz; + dest[2][0] = cxsy + cysx * sz; + dest[2][1] = -cz * sx; + dest[2][2] = cxcy - sxsy * sz; + dest[0][3] = 0.0f; + dest[1][3] = 0.0f; + dest[2][3] = 0.0f; + dest[3][0] = 0.0f; + dest[3][1] = 0.0f; + dest[3][2] = 0.0f; + dest[3][3] = 1.0f; +} + +/*! + * @brief build rotation matrix from euler angles + * + * @param[in] angles angles as vector [Xangle, Yangle, Zangle] + * @param[out] dest rotation matrix + */ +CGLM_INLINE +void +glm_euler_zxy(vec3 angles, mat4 dest) { + float cx, cy, cz, + sx, sy, sz, cycz, sxsy, cysz; + + sx = sinf(angles[0]); cx = cosf(angles[0]); + sy = sinf(angles[1]); cy = cosf(angles[1]); + sz = sinf(angles[2]); cz = cosf(angles[2]); + + cycz = cy * cz; + sxsy = sx * sy; + cysz = cy * sz; + + dest[0][0] = cycz - sxsy * sz; + dest[0][1] = cz * sxsy + cysz; + dest[0][2] = -cx * sy; + dest[1][0] = -cx * sz; + dest[1][1] = cx * cz; + dest[1][2] = sx; + dest[2][0] = cz * sy + cysz * sx; + dest[2][1] = -cycz * sx + sy * sz; + dest[2][2] = cx * cy; + dest[0][3] = 0.0f; + dest[1][3] = 0.0f; + dest[2][3] = 0.0f; + dest[3][0] = 0.0f; + dest[3][1] = 0.0f; + dest[3][2] = 0.0f; + dest[3][3] = 1.0f; +} + +/*! + * @brief build rotation matrix from euler angles + * + * @param[in] angles angles as vector [Xangle, Yangle, Zangle] + * @param[out] dest rotation matrix + */ +CGLM_INLINE +void +glm_euler_zyx(vec3 angles, mat4 dest) { + float cx, cy, cz, + sx, sy, sz, czsx, cxcz, sysz; + + sx = sinf(angles[0]); cx = cosf(angles[0]); + sy = sinf(angles[1]); cy = cosf(angles[1]); + sz = sinf(angles[2]); cz = cosf(angles[2]); + + czsx = cz * sx; + cxcz = cx * cz; + sysz = sy * sz; + + dest[0][0] = cy * cz; + dest[0][1] = cy * sz; + dest[0][2] = -sy; + dest[1][0] = czsx * sy - cx * sz; + dest[1][1] = cxcz + sx * sysz; + dest[1][2] = cy * sx; + dest[2][0] = cxcz * sy + sx * sz; + dest[2][1] = -czsx + cx * sysz; + dest[2][2] = cx * cy; + dest[0][3] = 0.0f; + dest[1][3] = 0.0f; + dest[2][3] = 0.0f; + dest[3][0] = 0.0f; + dest[3][1] = 0.0f; + dest[3][2] = 0.0f; + dest[3][3] = 1.0f; +} + +/*! + * @brief build rotation matrix from euler angles + * + * @param[in] angles angles as vector [Xangle, Yangle, Zangle] + * @param[in] ord euler order + * @param[out] dest rotation matrix + */ +CGLM_INLINE +void +glm_euler_by_order(vec3 angles, glm_euler_seq ord, mat4 dest) { + float cx, cy, cz, + sx, sy, sz; + + float cycz, cysz, cysx, cxcy, + czsy, cxcz, czsx, cxsz, + sysz; + + sx = sinf(angles[0]); cx = cosf(angles[0]); + sy = sinf(angles[1]); cy = cosf(angles[1]); + sz = sinf(angles[2]); cz = cosf(angles[2]); + + cycz = cy * cz; cysz = cy * sz; + cysx = cy * sx; cxcy = cx * cy; + czsy = cz * sy; cxcz = cx * cz; + czsx = cz * sx; cxsz = cx * sz; + sysz = sy * sz; + + switch (ord) { + case GLM_EULER_XZY: + dest[0][0] = cycz; + dest[0][1] = sx * sy + cx * cysz; + dest[0][2] = -cx * sy + cysx * sz; + dest[1][0] = -sz; + dest[1][1] = cxcz; + dest[1][2] = czsx; + dest[2][0] = czsy; + dest[2][1] = -cysx + cx * sysz; + dest[2][2] = cxcy + sx * sysz; + break; + case GLM_EULER_XYZ: + dest[0][0] = cycz; + dest[0][1] = czsx * sy + cxsz; + dest[0][2] = -cx * czsy + sx * sz; + dest[1][0] = -cysz; + dest[1][1] = cxcz - sx * sysz; + dest[1][2] = czsx + cx * sysz; + dest[2][0] = sy; + dest[2][1] = -cysx; + dest[2][2] = cxcy; + break; + case GLM_EULER_YXZ: + dest[0][0] = cycz + sx * sysz; + dest[0][1] = cxsz; + dest[0][2] = -czsy + cysx * sz; + dest[1][0] = czsx * sy - cysz; + dest[1][1] = cxcz; + dest[1][2] = cycz * sx + sysz; + dest[2][0] = cx * sy; + dest[2][1] = -sx; + dest[2][2] = cxcy; + break; + case GLM_EULER_YZX: + dest[0][0] = cycz; + dest[0][1] = sz; + dest[0][2] = -czsy; + dest[1][0] = sx * sy - cx * cysz; + dest[1][1] = cxcz; + dest[1][2] = cysx + cx * sysz; + dest[2][0] = cx * sy + cysx * sz; + dest[2][1] = -czsx; + dest[2][2] = cxcy - sx * sysz; + break; + case GLM_EULER_ZXY: + dest[0][0] = cycz - sx * sysz; + dest[0][1] = czsx * sy + cysz; + dest[0][2] = -cx * sy; + dest[1][0] = -cxsz; + dest[1][1] = cxcz; + dest[1][2] = sx; + dest[2][0] = czsy + cysx * sz; + dest[2][1] = -cycz * sx + sysz; + dest[2][2] = cxcy; + break; + case GLM_EULER_ZYX: + dest[0][0] = cycz; + dest[0][1] = cysz; + dest[0][2] = -sy; + dest[1][0] = czsx * sy - cxsz; + dest[1][1] = cxcz + sx * sysz; + dest[1][2] = cysx; + dest[2][0] = cx * czsy + sx * sz; + dest[2][1] = -czsx + cx * sysz; + dest[2][2] = cxcy; + break; + } + + dest[0][3] = 0.0f; + dest[1][3] = 0.0f; + dest[2][3] = 0.0f; + dest[3][0] = 0.0f; + dest[3][1] = 0.0f; + dest[3][2] = 0.0f; + dest[3][3] = 1.0f; +} + + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in x y z order (roll pitch yaw) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_xyz_quat(vec3 angles, versor dest) { +#ifdef CGLM_FORCE_LEFT_HANDED + glm_euler_xyz_quat_lh(angles, dest); +#else + glm_euler_xyz_quat_rh(angles, dest); +#endif +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in x z y order (roll yaw pitch) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_xzy_quat(vec3 angles, versor dest) { +#ifdef CGLM_FORCE_LEFT_HANDED + glm_euler_xzy_quat_lh(angles, dest); +#else + glm_euler_xzy_quat_rh(angles, dest); +#endif +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in y x z order (pitch roll yaw) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_yxz_quat(vec3 angles, versor dest) { +#ifdef CGLM_FORCE_LEFT_HANDED + glm_euler_yxz_quat_lh(angles, dest); +#else + glm_euler_yxz_quat_rh(angles, dest); +#endif +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in y z x order (pitch yaw roll) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_yzx_quat(vec3 angles, versor dest) { +#ifdef CGLM_FORCE_LEFT_HANDED + glm_euler_yzx_quat_lh(angles, dest); +#else + glm_euler_yzx_quat_rh(angles, dest); +#endif +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in z x y order (yaw roll pitch) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_zxy_quat(vec3 angles, versor dest) { +#ifdef CGLM_FORCE_LEFT_HANDED + glm_euler_zxy_quat_lh(angles, dest); +#else + glm_euler_zxy_quat_rh(angles, dest); +#endif +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in z y x order (yaw pitch roll) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_zyx_quat(vec3 angles, versor dest) { +#ifdef CGLM_FORCE_LEFT_HANDED + glm_euler_zyx_quat_lh(angles, dest); +#else + glm_euler_zyx_quat_rh(angles, dest); +#endif +} + + +#endif /* cglm_euler_h */ diff --git a/cglm/include/cglm/frustum.h b/cglm/include/cglm/frustum.h new file mode 100644 index 0000000..5aa3c17 --- /dev/null +++ b/cglm/include/cglm/frustum.h @@ -0,0 +1,255 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_frustum_h +#define cglm_frustum_h + +#include "common.h" +#include "plane.h" +#include "vec3.h" +#include "vec4.h" +#include "mat4.h" + +#define GLM_LBN 0 /* left bottom near */ +#define GLM_LTN 1 /* left top near */ +#define GLM_RTN 2 /* right top near */ +#define GLM_RBN 3 /* right bottom near */ + +#define GLM_LBF 4 /* left bottom far */ +#define GLM_LTF 5 /* left top far */ +#define GLM_RTF 6 /* right top far */ +#define GLM_RBF 7 /* right bottom far */ + +#define GLM_LEFT 0 +#define GLM_RIGHT 1 +#define GLM_BOTTOM 2 +#define GLM_TOP 3 +#define GLM_NEAR 4 +#define GLM_FAR 5 + +/* you can override clip space coords + but you have to provide all with same name + e.g.: define GLM_CSCOORD_LBN {0.0f, 0.0f, 1.0f, 1.0f} */ +#ifndef GLM_CUSTOM_CLIPSPACE + +/* near */ +#define GLM_CSCOORD_LBN {-1.0f, -1.0f, -1.0f, 1.0f} +#define GLM_CSCOORD_LTN {-1.0f, 1.0f, -1.0f, 1.0f} +#define GLM_CSCOORD_RTN { 1.0f, 1.0f, -1.0f, 1.0f} +#define GLM_CSCOORD_RBN { 1.0f, -1.0f, -1.0f, 1.0f} + +/* far */ +#define GLM_CSCOORD_LBF {-1.0f, -1.0f, 1.0f, 1.0f} +#define GLM_CSCOORD_LTF {-1.0f, 1.0f, 1.0f, 1.0f} +#define GLM_CSCOORD_RTF { 1.0f, 1.0f, 1.0f, 1.0f} +#define GLM_CSCOORD_RBF { 1.0f, -1.0f, 1.0f, 1.0f} + +#endif + +/*! + * @brief extracts view frustum planes + * + * planes' space: + * 1- if m = proj: View Space + * 2- if m = viewProj: World Space + * 3- if m = MVP: Object Space + * + * You probably want to extract planes in world space so use viewProj as m + * Computing viewProj: + * glm_mat4_mul(proj, view, viewProj); + * + * Exracted planes order: [left, right, bottom, top, near, far] + * + * @param[in] m matrix (see brief) + * @param[out] dest extracted view frustum planes (see brief) + */ +CGLM_INLINE +void +glm_frustum_planes(mat4 m, vec4 dest[6]) { + mat4 t; + + glm_mat4_transpose_to(m, t); + + glm_vec4_add(t[3], t[0], dest[0]); /* left */ + glm_vec4_sub(t[3], t[0], dest[1]); /* right */ + glm_vec4_add(t[3], t[1], dest[2]); /* bottom */ + glm_vec4_sub(t[3], t[1], dest[3]); /* top */ + glm_vec4_add(t[3], t[2], dest[4]); /* near */ + glm_vec4_sub(t[3], t[2], dest[5]); /* far */ + + glm_plane_normalize(dest[0]); + glm_plane_normalize(dest[1]); + glm_plane_normalize(dest[2]); + glm_plane_normalize(dest[3]); + glm_plane_normalize(dest[4]); + glm_plane_normalize(dest[5]); +} + +/*! + * @brief extracts view frustum corners using clip-space coordinates + * + * corners' space: + * 1- if m = invViewProj: World Space + * 2- if m = invMVP: Object Space + * + * You probably want to extract corners in world space so use invViewProj + * Computing invViewProj: + * glm_mat4_mul(proj, view, viewProj); + * ... + * glm_mat4_inv(viewProj, invViewProj); + * + * if you have a near coord at i index, you can get it's far coord by i + 4 + * + * Find center coordinates: + * for (j = 0; j < 4; j++) { + * glm_vec3_center(corners[i], corners[i + 4], centerCorners[i]); + * } + * + * @param[in] invMat matrix (see brief) + * @param[out] dest exracted view frustum corners (see brief) + */ +CGLM_INLINE +void +glm_frustum_corners(mat4 invMat, vec4 dest[8]) { + vec4 c[8]; + + /* indexOf(nearCoord) = indexOf(farCoord) + 4 */ + vec4 csCoords[8] = { + GLM_CSCOORD_LBN, + GLM_CSCOORD_LTN, + GLM_CSCOORD_RTN, + GLM_CSCOORD_RBN, + + GLM_CSCOORD_LBF, + GLM_CSCOORD_LTF, + GLM_CSCOORD_RTF, + GLM_CSCOORD_RBF + }; + + glm_mat4_mulv(invMat, csCoords[0], c[0]); + glm_mat4_mulv(invMat, csCoords[1], c[1]); + glm_mat4_mulv(invMat, csCoords[2], c[2]); + glm_mat4_mulv(invMat, csCoords[3], c[3]); + glm_mat4_mulv(invMat, csCoords[4], c[4]); + glm_mat4_mulv(invMat, csCoords[5], c[5]); + glm_mat4_mulv(invMat, csCoords[6], c[6]); + glm_mat4_mulv(invMat, csCoords[7], c[7]); + + glm_vec4_scale(c[0], 1.0f / c[0][3], dest[0]); + glm_vec4_scale(c[1], 1.0f / c[1][3], dest[1]); + glm_vec4_scale(c[2], 1.0f / c[2][3], dest[2]); + glm_vec4_scale(c[3], 1.0f / c[3][3], dest[3]); + glm_vec4_scale(c[4], 1.0f / c[4][3], dest[4]); + glm_vec4_scale(c[5], 1.0f / c[5][3], dest[5]); + glm_vec4_scale(c[6], 1.0f / c[6][3], dest[6]); + glm_vec4_scale(c[7], 1.0f / c[7][3], dest[7]); +} + +/*! + * @brief finds center of view frustum + * + * @param[in] corners view frustum corners + * @param[out] dest view frustum center + */ +CGLM_INLINE +void +glm_frustum_center(vec4 corners[8], vec4 dest) { + vec4 center; + + glm_vec4_copy(corners[0], center); + + glm_vec4_add(corners[1], center, center); + glm_vec4_add(corners[2], center, center); + glm_vec4_add(corners[3], center, center); + glm_vec4_add(corners[4], center, center); + glm_vec4_add(corners[5], center, center); + glm_vec4_add(corners[6], center, center); + glm_vec4_add(corners[7], center, center); + + glm_vec4_scale(center, 0.125f, dest); +} + +/*! + * @brief finds bounding box of frustum relative to given matrix e.g. view mat + * + * @param[in] corners view frustum corners + * @param[in] m matrix to convert existing conners + * @param[out] box bounding box as array [min, max] + */ +CGLM_INLINE +void +glm_frustum_box(vec4 corners[8], mat4 m, vec3 box[2]) { + vec4 v; + vec3 min, max; + int i; + + glm_vec3_broadcast(FLT_MAX, min); + glm_vec3_broadcast(-FLT_MAX, max); + + for (i = 0; i < 8; i++) { + glm_mat4_mulv(m, corners[i], v); + + min[0] = glm_min(min[0], v[0]); + min[1] = glm_min(min[1], v[1]); + min[2] = glm_min(min[2], v[2]); + + max[0] = glm_max(max[0], v[0]); + max[1] = glm_max(max[1], v[1]); + max[2] = glm_max(max[2], v[2]); + } + + glm_vec3_copy(min, box[0]); + glm_vec3_copy(max, box[1]); +} + +/*! + * @brief finds planes corners which is between near and far planes (parallel) + * + * this will be helpful if you want to split a frustum e.g. CSM/PSSM. This will + * find planes' corners but you will need to one more plane. + * Actually you have it, it is near, far or created previously with this func ;) + * + * @param[in] corners view frustum corners + * @param[in] splitDist split distance + * @param[in] farDist far distance (zFar) + * @param[out] planeCorners plane corners [LB, LT, RT, RB] + */ +CGLM_INLINE +void +glm_frustum_corners_at(vec4 corners[8], + float splitDist, + float farDist, + vec4 planeCorners[4]) { + vec4 corner; + float dist, sc; + + /* because distance and scale is same for all */ + dist = glm_vec3_distance(corners[GLM_RTF], corners[GLM_RTN]); + sc = dist * (splitDist / farDist); + + /* left bottom */ + glm_vec4_sub(corners[GLM_LBF], corners[GLM_LBN], corner); + glm_vec4_scale_as(corner, sc, corner); + glm_vec4_add(corners[GLM_LBN], corner, planeCorners[0]); + + /* left top */ + glm_vec4_sub(corners[GLM_LTF], corners[GLM_LTN], corner); + glm_vec4_scale_as(corner, sc, corner); + glm_vec4_add(corners[GLM_LTN], corner, planeCorners[1]); + + /* right top */ + glm_vec4_sub(corners[GLM_RTF], corners[GLM_RTN], corner); + glm_vec4_scale_as(corner, sc, corner); + glm_vec4_add(corners[GLM_RTN], corner, planeCorners[2]); + + /* right bottom */ + glm_vec4_sub(corners[GLM_RBF], corners[GLM_RBN], corner); + glm_vec4_scale_as(corner, sc, corner); + glm_vec4_add(corners[GLM_RBN], corner, planeCorners[3]); +} + +#endif /* cglm_frustum_h */ diff --git a/cglm/include/cglm/handed/euler_to_quat_lh.h b/cglm/include/cglm/handed/euler_to_quat_lh.h new file mode 100644 index 0000000..1bb350b --- /dev/null +++ b/cglm/include/cglm/handed/euler_to_quat_lh.h @@ -0,0 +1,167 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_euler_xyz_quat_lh(vec3 angles, versor dest); + CGLM_INLINE void glm_euler_xzy_quat_lh(vec3 angles, versor dest); + CGLM_INLINE void glm_euler_yxz_quat_lh(vec3 angles, versor dest); + CGLM_INLINE void glm_euler_yzx_quat_lh(vec3 angles, versor dest); + CGLM_INLINE void glm_euler_zxy_quat_lh(vec3 angles, versor dest); + CGLM_INLINE void glm_euler_zyx_quat_lh(vec3 angles, versor dest); + */ + +/* + Things to note: + The only difference between euler to quat rh vs lh is that the zsin part is negative + */ + +#ifndef cglm_euler_to_quat_lh_h +#define cglm_euler_to_quat_lh_h + +#include "../common.h" + + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in x y z order in left hand (roll pitch yaw) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_xyz_quat_lh(vec3 angles, versor dest) { + float xc, yc, zc, + xs, ys, zs; + + xs = sinf(angles[0] * 0.5f); xc = cosf(angles[0] * 0.5f); + ys = sinf(angles[1] * 0.5f); yc = cosf(angles[1] * 0.5f); + zs = -sinf(angles[2] * 0.5f); zc = cosf(angles[2] * 0.5f); + + dest[0] = xc * ys * zs + xs * yc * zc; + dest[1] = xc * ys * zc - xs * yc * zs; + dest[2] = xc * yc * zs + xs * ys * zc; + dest[3] = xc * yc * zc - xs * ys * zs; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in x z y order in left hand (roll yaw pitch) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_xzy_quat_lh(vec3 angles, versor dest) { + float xc, yc, zc, + xs, ys, zs; + + xs = sinf(angles[0] * 0.5f); xc = cosf(angles[0] * 0.5f); + ys = sinf(angles[1] * 0.5f); yc = cosf(angles[1] * 0.5f); + zs = -sinf(angles[2] * 0.5f); zc = cosf(angles[2] * 0.5f); + + dest[0] = -xc * zs * ys + xs * zc * yc; + dest[1] = xc * zc * ys - xs * zs * yc; + dest[2] = xc * zs * yc + xs * zc * ys; + dest[3] = xc * zc * yc + xs * zs * ys; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in y x z order in left hand (pitch roll yaw) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_yxz_quat_lh(vec3 angles, versor dest) { + float xc, yc, zc, + xs, ys, zs; + + xs = sinf(angles[0] * 0.5f); xc = cosf(angles[0] * 0.5f); + ys = sinf(angles[1] * 0.5f); yc = cosf(angles[1] * 0.5f); + zs = -sinf(angles[2] * 0.5f); zc = cosf(angles[2] * 0.5f); + + dest[0] = yc * xs * zc + ys * xc * zs; + dest[1] = -yc * xs * zs + ys * xc * zc; + dest[2] = yc * xc * zs - ys * xs * zc; + dest[3] = yc * xc * zc + ys * xs * zs; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in y z x order in left hand (pitch yaw roll) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_yzx_quat_lh(vec3 angles, versor dest) { + float xc, yc, zc, + xs, ys, zs; + + xs = sinf(angles[0] * 0.5f); xc = cosf(angles[0] * 0.5f); + ys = sinf(angles[1] * 0.5f); yc = cosf(angles[1] * 0.5f); + zs = -sinf(angles[2] * 0.5f); zc = cosf(angles[2] * 0.5f); + + dest[0] = yc * zc * xs + ys * zs * xc; + dest[1] = yc * zs * xs + ys * zc * xc; + dest[2] = yc * zs * xc - ys * zc * xs; + dest[3] = yc * zc * xc - ys * zs * xs; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in z x y order in left hand (yaw roll pitch) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_zxy_quat_lh(vec3 angles, versor dest) { + float xc, yc, zc, + xs, ys, zs; + + xs = sinf(angles[0] * 0.5f); xc = cosf(angles[0] * 0.5f); + ys = sinf(angles[1] * 0.5f); yc = cosf(angles[1] * 0.5f); + zs = -sinf(angles[2] * 0.5f); zc = cosf(angles[2] * 0.5f); + + dest[0] = zc * xs * yc - zs * xc * ys; + dest[1] = zc * xc * ys + zs * xs * yc; + dest[2] = zc * xs * ys + zs * xc * yc; + dest[3] = zc * xc * yc - zs * xs * ys; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in z y x order in left hand (yaw pitch roll) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_zyx_quat_lh(vec3 angles, versor dest) { + float xc, yc, zc, + xs, ys, zs; + + xs = sinf(angles[0] * 0.5f); xc = cosf(angles[0] * 0.5f); + ys = sinf(angles[1] * 0.5f); yc = cosf(angles[1] * 0.5f); + zs = -sinf(angles[2] * 0.5f); zc = cosf(angles[2] * 0.5f); + + dest[0] = zc * yc * xs - zs * ys * xc; + dest[1] = zc * ys * xc + zs * yc * xs; + dest[2] = -zc * ys * xs + zs * yc * xc; + dest[3] = zc * yc * xc + zs * ys * xs; +} + +#endif /*cglm_euler_to_quat_lh_h*/ diff --git a/cglm/include/cglm/handed/euler_to_quat_rh.h b/cglm/include/cglm/handed/euler_to_quat_rh.h new file mode 100644 index 0000000..aeb6f81 --- /dev/null +++ b/cglm/include/cglm/handed/euler_to_quat_rh.h @@ -0,0 +1,170 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_euler_xyz_quat_rh(vec3 angles, versor dest); + CGLM_INLINE void glm_euler_xzy_quat_rh(vec3 angles, versor dest); + CGLM_INLINE void glm_euler_yxz_quat_rh(vec3 angles, versor dest); + CGLM_INLINE void glm_euler_yzx_quat_rh(vec3 angles, versor dest); + CGLM_INLINE void glm_euler_zxy_quat_rh(vec3 angles, versor dest); + CGLM_INLINE void glm_euler_zyx_quat_rh(vec3 angles, versor dest); + */ + +/* + Things to note: + The only difference between euler to quat rh vs lh is that the zsin part is negative + */ + +#ifndef cglm_euler_to_quat_rh_h +#define cglm_euler_to_quat_rh_h + +#include "../common.h" + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in x y z order in right hand (roll pitch yaw) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_xyz_quat_rh(vec3 angles, versor dest) { + float xc, yc, zc, + xs, ys, zs; + + xs = sinf(angles[0] * 0.5f); xc = cosf(angles[0] * 0.5f); + ys = sinf(angles[1] * 0.5f); yc = cosf(angles[1] * 0.5f); + zs = sinf(angles[2] * 0.5f); zc = cosf(angles[2] * 0.5f); + + dest[0] = xc * ys * zs + xs * yc * zc; + dest[1] = xc * ys * zc - xs * yc * zs; + dest[2] = xc * yc * zs + xs * ys * zc; + dest[3] = xc * yc * zc - xs * ys * zs; + +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in x z y order in right hand (roll yaw pitch) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_xzy_quat_rh(vec3 angles, versor dest) { + float xc, yc, zc, + xs, ys, zs; + + xs = sinf(angles[0] * 0.5f); xc = cosf(angles[0] * 0.5f); + ys = sinf(angles[1] * 0.5f); yc = cosf(angles[1] * 0.5f); + zs = sinf(angles[2] * 0.5f); zc = cosf(angles[2] * 0.5f); + + dest[0] = -xc * zs * ys + xs * zc * yc; + dest[1] = xc * zc * ys - xs * zs * yc; + dest[2] = xc * zs * yc + xs * zc * ys; + dest[3] = xc * zc * yc + xs * zs * ys; + +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in y x z order in right hand (pitch roll yaw) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_yxz_quat_rh(vec3 angles, versor dest) { + float xc, yc, zc, + xs, ys, zs; + + xs = sinf(angles[0] * 0.5f); xc = cosf(angles[0] * 0.5f); + ys = sinf(angles[1] * 0.5f); yc = cosf(angles[1] * 0.5f); + zs = sinf(angles[2] * 0.5f); zc = cosf(angles[2] * 0.5f); + + dest[0] = yc * xs * zc + ys * xc * zs; + dest[1] = -yc * xs * zs + ys * xc * zc; + dest[2] = yc * xc * zs - ys * xs * zc; + dest[3] = yc * xc * zc + ys * xs * zs; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in y z x order in right hand (pitch yaw roll) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_yzx_quat_rh(vec3 angles, versor dest) { + float xc, yc, zc, + xs, ys, zs; + + xs = sinf(angles[0] * 0.5f); xc = cosf(angles[0] * 0.5f); + ys = sinf(angles[1] * 0.5f); yc = cosf(angles[1] * 0.5f); + zs = sinf(angles[2] * 0.5f); zc = cosf(angles[2] * 0.5f); + + dest[0] = yc * zc * xs + ys * zs * xc; + dest[1] = yc * zs * xs + ys * zc * xc; + dest[2] = yc * zs * xc - ys * zc * xs; + dest[3] = yc * zc * xc - ys * zs * xs; + +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in z x y order in right hand (yaw roll pitch) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_zxy_quat_rh(vec3 angles, versor dest) { + float xc, yc, zc, + xs, ys, zs; + + xs = sinf(angles[0] * 0.5f); xc = cosf(angles[0] * 0.5f); + ys = sinf(angles[1] * 0.5f); yc = cosf(angles[1] * 0.5f); + zs = sinf(angles[2] * 0.5f); zc = cosf(angles[2] * 0.5f); + + dest[0] = zc * xs * yc - zs * xc * ys; + dest[1] = zc * xc * ys + zs * xs * yc; + dest[2] = zc * xs * ys + zs * xc * yc; + dest[3] = zc * xc * yc - zs * xs * ys; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in z y x order in right hand (yaw pitch roll) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_euler_zyx_quat_rh(vec3 angles, versor dest) { + float xc, yc, zc, + xs, ys, zs; + + xs = sinf(angles[0] * 0.5f); xc = cosf(angles[0] * 0.5f); + ys = sinf(angles[1] * 0.5f); yc = cosf(angles[1] * 0.5f); + zs = sinf(angles[2] * 0.5f); zc = cosf(angles[2] * 0.5f); + + dest[0] = zc * yc * xs - zs * ys * xc; + dest[1] = zc * ys * xc + zs * yc * xs; + dest[2] = -zc * ys * xs + zs * yc * xc; + dest[3] = zc * yc * xc + zs * ys * xs; +} + + +#endif /*cglm_euler_to_quat_rh_h*/ diff --git a/cglm/include/cglm/io.h b/cglm/include/cglm/io.h new file mode 100644 index 0000000..baa80f1 --- /dev/null +++ b/cglm/include/cglm/io.h @@ -0,0 +1,440 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_mat4_print(mat4 matrix, FILE *ostream); + CGLM_INLINE void glm_mat3_print(mat3 matrix, FILE *ostream); + CGLM_INLINE void glm_vec4_print(vec4 vec, FILE *ostream); + CGLM_INLINE void glm_ivec4_print(ivec4 vec, FILE *ostream); + CGLM_INLINE void glm_vec3_print(vec3 vec, FILE *ostream); + CGLM_INLINE void glm_ivec3_print(ivec3 vec, FILE *ostream); + CGLM_INLINE void glm_vec2_print(vec2 vec, FILE *ostream); + CGLM_INLINE void glm_ivec2_print(ivec2 vec, FILE *ostream); + CGLM_INLINE void glm_versor_print(versor vec, FILE *ostream); + CGLM_INLINE void glm_arch_print(FILE *ostream); + */ + +/* + cglm tried to enable print functions in debug mode and disable them in + release/production mode to eliminate printing costs. + + if you need to force enable then define CGLM_DEFINE_PRINTS macro not DEBUG one + + Print functions are enabled if: + + - DEBUG or _DEBUG macro is defined (mostly defined automatically in debugging) + - CGLM_DEFINE_PRINTS macro is defined including release/production + which makes enabled printing always + - glmc_ calls for io are always prints + + */ + +/* DEPRECATED: CGLM_NO_PRINTS_NOOP (use CGLM_DEFINE_PRINTS) */ + +#ifndef cglm_io_h +#define cglm_io_h +#if !defined(NDEBUG) \ + || defined(CGLM_DEFINE_PRINTS) || defined(CGLM_LIB_SRC) \ + || defined(CGLM_NO_PRINTS_NOOP) + +#include "common.h" +#include "util.h" + +#include +#include + +#ifndef CGLM_PRINT_PRECISION +# define CGLM_PRINT_PRECISION 5 +#endif + +#ifndef CGLM_PRINT_MAX_TO_SHORT +# define CGLM_PRINT_MAX_TO_SHORT 1e5f +#endif + +#ifndef GLM_TESTS_NO_COLORFUL_OUTPUT +# ifndef CGLM_PRINT_COLOR +# define CGLM_PRINT_COLOR "\033[36m" +# endif +# ifndef CGLM_PRINT_COLOR_RESET +# define CGLM_PRINT_COLOR_RESET "\033[0m" +# endif +#else +# ifndef CGLM_PRINT_COLOR +# define CGLM_PRINT_COLOR +# endif +# ifndef CGLM_PRINT_COLOR_RESET +# define CGLM_PRINT_COLOR_RESET +# endif +#endif + +/*! + * @brief prints current SIMD path in general + * + * @param[in] ostream stream to print e.g. stdout, stderr, FILE ... + */ +CGLM_INLINE +void +glm_arch_print(FILE* __restrict ostream) { + fprintf(ostream, CGLM_PRINT_COLOR "arch: " +#if defined(CGLM_SIMD_WASM) + "wasm SIMD128" +#elif defined(CGLM_SIMD_x86) + "x86 SSE* " +# ifdef __AVX__ + " AVX" +# endif +#elif defined(CGLM_SIMD_ARM) + "arm" +# ifndef __ARM_NEON_FP + " NEON_FP" +# endif +# ifdef CGLM_ARM64 + " ARM64" +# endif +#else + "uncommon" +#endif + CGLM_PRINT_COLOR_RESET); +} + +/*! + * @brief prints current SIMD path in general + * + * @param[in] ostream stream to print e.g. stdout, stderr, FILE ... + */ +CGLM_INLINE +void +glm_arch_print_name(FILE* __restrict ostream) { + fprintf(ostream, CGLM_PRINT_COLOR "\ncglm "); + glm_arch_print(ostream); + fprintf(ostream, "\n\n" CGLM_PRINT_COLOR_RESET); +} + +CGLM_INLINE +void +glm_mat4_print(mat4 matrix, + FILE * __restrict ostream) { + char buff[16]; + int i, j, cw[4], cwi; + +#define m 4 +#define n 4 + + fprintf(ostream, "Matrix (float%dx%d): " CGLM_PRINT_COLOR "\n" , m, n); + + cw[0] = cw[1] = cw[2] = cw[3] = 0; + + for (i = 0; i < m; i++) { + for (j = 0; j < n; j++) { + if (matrix[i][j] < CGLM_PRINT_MAX_TO_SHORT) + cwi = snprintf(buff, sizeof(buff), "% .*f", CGLM_PRINT_PRECISION, (double)matrix[i][j]); + else + cwi = snprintf(buff, sizeof(buff), "% g", (double)matrix[i][j]); + cw[i] = GLM_MAX(cw[i], cwi); + } + } + + for (i = 0; i < m; i++) { + fprintf(ostream, " |"); + + for (j = 0; j < n; j++) + if (matrix[i][j] < CGLM_PRINT_MAX_TO_SHORT) + fprintf(ostream, " % *.*f", cw[j], CGLM_PRINT_PRECISION, (double)matrix[j][i]); + else + fprintf(ostream, " % *g", cw[j], (double)matrix[j][i]); + + fprintf(ostream, " |\n"); + } + + fprintf(ostream, CGLM_PRINT_COLOR_RESET "\n"); + +#undef m +#undef n +} + + +CGLM_INLINE +void +glm_mat3_print(mat3 matrix, + FILE * __restrict ostream) { + char buff[16]; + int i, j, cw[4], cwi; + +#define m 3 +#define n 3 + + fprintf(ostream, "Matrix (float%dx%d): " CGLM_PRINT_COLOR "\n", m, n); + + cw[0] = cw[1] = cw[2] = 0; + + for (i = 0; i < m; i++) { + for (j = 0; j < n; j++) { + if (matrix[i][j] < CGLM_PRINT_MAX_TO_SHORT) + cwi = snprintf(buff, sizeof(buff), "% .*f", CGLM_PRINT_PRECISION, (double)matrix[i][j]); + else + cwi = snprintf(buff, sizeof(buff), "% g", (double)matrix[i][j]); + cw[i] = GLM_MAX(cw[i], cwi); + } + } + + for (i = 0; i < m; i++) { + fprintf(ostream, " |"); + + for (j = 0; j < n; j++) + if (matrix[i][j] < CGLM_PRINT_MAX_TO_SHORT) + fprintf(ostream, " % *.*f", cw[j], CGLM_PRINT_PRECISION, (double)matrix[j][i]); + else + fprintf(ostream, " % *g", cw[j], (double)matrix[j][i]); + + fprintf(ostream, " |\n"); + } + + fprintf(ostream, CGLM_PRINT_COLOR_RESET "\n"); + +#undef m +#undef n +} + +CGLM_INLINE +void +glm_mat2_print(mat2 matrix, + FILE * __restrict ostream) { + char buff[16]; + int i, j, cw[4], cwi; + +#define m 2 +#define n 2 + + fprintf(ostream, "Matrix (float%dx%d): " CGLM_PRINT_COLOR "\n", m, n); + + cw[0] = cw[1] = 0; + + for (i = 0; i < m; i++) { + for (j = 0; j < n; j++) { + if (matrix[i][j] < CGLM_PRINT_MAX_TO_SHORT) + cwi = snprintf(buff, sizeof(buff), "% .*f", CGLM_PRINT_PRECISION, (double)matrix[i][j]); + else + cwi = snprintf(buff, sizeof(buff), "% g", (double)matrix[i][j]); + cw[i] = GLM_MAX(cw[i], cwi); + } + } + + for (i = 0; i < m; i++) { + fprintf(ostream, " |"); + + for (j = 0; j < n; j++) + if (matrix[i][j] < CGLM_PRINT_MAX_TO_SHORT) + fprintf(ostream, " % *.*f", cw[j], CGLM_PRINT_PRECISION, (double)matrix[j][i]); + else + fprintf(ostream, " % *g", cw[j], (double)matrix[j][i]); + + fprintf(ostream, " |\n"); + } + + fprintf(ostream, CGLM_PRINT_COLOR_RESET "\n"); + +#undef m +#undef n +} + +CGLM_INLINE +void +glm_vec4_print(vec4 vec, + FILE * __restrict ostream) { + int i; + +#define m 4 + + fprintf(ostream, "Vector (float%d): " CGLM_PRINT_COLOR "\n (", m); + + for (i = 0; i < m; i++) { + if (vec[i] < CGLM_PRINT_MAX_TO_SHORT) + fprintf(ostream, " % .*f", CGLM_PRINT_PRECISION, (double)vec[i]); + else + fprintf(ostream, " % g", (double)vec[i]); + } + + fprintf(ostream, " )" CGLM_PRINT_COLOR_RESET "\n\n"); + +#undef m +} + +CGLM_INLINE +void +glm_ivec4_print(ivec4 vec, + FILE * __restrict ostream) { + int i; + +#define m 4 + + fprintf(ostream, "Vector (int%d): " CGLM_PRINT_COLOR "\n (", m); + + for (i = 0; i < m; i++) + fprintf(ostream, " % d", vec[i]); + + fprintf(ostream, " )" CGLM_PRINT_COLOR_RESET "\n\n"); + +#undef m +} + +CGLM_INLINE +void +glm_vec3_print(vec3 vec, + FILE * __restrict ostream) { + int i; + +#define m 3 + + fprintf(ostream, "Vector (float%d): " CGLM_PRINT_COLOR "\n (", m); + + for (i = 0; i < m; i++) { + if (vec[i] < CGLM_PRINT_MAX_TO_SHORT) + fprintf(ostream, " % .*f", CGLM_PRINT_PRECISION, (double)vec[i]); + else + fprintf(ostream, " % g", (double)vec[i]); + } + + fprintf(ostream, " )" CGLM_PRINT_COLOR_RESET "\n\n"); + +#undef m +} + +CGLM_INLINE +void +glm_ivec3_print(ivec3 vec, + FILE * __restrict ostream) { + int i; + +#define m 3 + + fprintf(ostream, "Vector (int%d): " CGLM_PRINT_COLOR "\n (", m); + + for (i = 0; i < m; i++) + fprintf(ostream, " % d", vec[i]); + + fprintf(ostream, " )" CGLM_PRINT_COLOR_RESET "\n\n"); + +#undef m +} + +CGLM_INLINE +void +glm_vec2_print(vec2 vec, + FILE * __restrict ostream) { + int i; + +#define m 2 + + fprintf(ostream, "Vector (float%d): " CGLM_PRINT_COLOR "\n (", m); + + for (i = 0; i < m; i++) { + if (vec[i] < CGLM_PRINT_MAX_TO_SHORT) + fprintf(ostream, " % .*f", CGLM_PRINT_PRECISION, (double)vec[i]); + else + fprintf(ostream, " % g", (double)vec[i]); + } + + fprintf(ostream, " )" CGLM_PRINT_COLOR_RESET "\n\n"); + +#undef m +} + +CGLM_INLINE +void +glm_ivec2_print(ivec2 vec, + FILE * __restrict ostream) { + int i; + +#define m 2 + + fprintf(ostream, "Vector (int%d): " CGLM_PRINT_COLOR "\n (", m); + + for (i = 0; i < m; i++) + fprintf(ostream, " % d", vec[i]); + + fprintf(ostream, " )" CGLM_PRINT_COLOR_RESET "\n\n"); + +#undef m +} + +CGLM_INLINE +void +glm_versor_print(versor vec, + FILE * __restrict ostream) { + int i; + +#define m 4 + + fprintf(ostream, "Quaternion (float%d): " CGLM_PRINT_COLOR "\n (", m); + + for (i = 0; i < m; i++) { + if (vec[i] < CGLM_PRINT_MAX_TO_SHORT) + fprintf(ostream, " % .*f", CGLM_PRINT_PRECISION, (double)vec[i]); + else + fprintf(ostream, " % g", (double)vec[i]); + } + + + fprintf(ostream, " )" CGLM_PRINT_COLOR_RESET "\n\n"); + +#undef m +} + +CGLM_INLINE +void +glm_aabb_print(vec3 bbox[2], + const char * __restrict tag, + FILE * __restrict ostream) { + int i, j; + +#define m 3 + + fprintf(ostream, "AABB (%s): " CGLM_PRINT_COLOR "\n", tag ? tag: "float"); + + for (i = 0; i < 2; i++) { + fprintf(ostream, " ("); + + for (j = 0; j < m; j++) { + if (bbox[i][j] < CGLM_PRINT_MAX_TO_SHORT) + fprintf(ostream, " % .*f", CGLM_PRINT_PRECISION, (double)bbox[i][j]); + else + fprintf(ostream, " % g", (double)bbox[i][j]); + } + + fprintf(ostream, " )\n"); + } + + fprintf(ostream, CGLM_PRINT_COLOR_RESET "\n"); + +#undef m +} + +#else + +#include "common.h" + +#include +#include + +/* NOOP: Remove print from DEBUG */ +#define glm_mat4_print(v, s) (void)v; (void)s; +#define glm_mat3_print(v, s) (void)v; (void)s; +#define glm_mat2_print(v, s) (void)v; (void)s; +#define glm_vec4_print(v, s) (void)v; (void)s; +#define glm_ivec4_print(v, s) (void)v; (void)s; +#define glm_vec3_print(v, s) (void)v; (void)s; +#define glm_ivec3_print(v, s) (void)v; (void)s; +#define glm_vec2_print(v, s) (void)v; (void)s; +#define glm_ivec2_print(v, s) (void)v; (void)s; +#define glm_versor_print(v, s) (void)v; (void)s; +#define glm_aabb_print(v, t, s) (void)v; (void)t; (void)s; +#define glm_arch_print(s) (void)s; +#define glm_arch_print_name(s) (void)s; + +#endif +#endif /* cglm_io_h */ diff --git a/cglm/include/cglm/ivec2.h b/cglm/include/cglm/ivec2.h new file mode 100644 index 0000000..8d5ad88 --- /dev/null +++ b/cglm/include/cglm/ivec2.h @@ -0,0 +1,659 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_IVEC2_ONE_INIT + GLM_IVEC2_ZERO_INIT + GLM_IVEC2_ONE + GLM_IVEC2_ZERO + + Functions: + CGLM_INLINE void glm_ivec2(int * __restrict v, ivec2 dest) + CGLM_INLINE void glm_ivec2_copy(ivec2 a, ivec2 dest) + CGLM_INLINE void glm_ivec2_zero(ivec2 v) + CGLM_INLINE void glm_ivec2_one(ivec2 v) + CGLM_INLINE int glm_ivec2_dot(ivec2 a, ivec2 b) + CGLM_INLINE int glm_ivec2_cross(ivec2 a, ivec2 b) + CGLM_INLINE void glm_ivec2_add(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_adds(ivec2 v, int s, ivec2 dest) + CGLM_INLINE void glm_ivec2_sub(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_subs(ivec2 v, int s, ivec2 dest) + CGLM_INLINE void glm_ivec2_mul(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_scale(ivec2 v, int s, ivec2 dest) + CGLM_INLINE void glm_ivec2_div(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_divs(ivec2 v, int s, ivec2 dest) + CGLM_INLINE void glm_ivec2_mod(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_addadd(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_addadds(ivec2 a, int s, ivec2 dest) + CGLM_INLINE void glm_ivec2_subadd(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_subadds(ivec2 a, int s, ivec2 dest) + CGLM_INLINE void glm_ivec2_muladd(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_muladds(ivec2 a, int s, ivec2 dest) + CGLM_INLINE void glm_ivec2_maxadd(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_minadd(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_subsub(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_subsubs(ivec2 a, int s, ivec2 dest) + CGLM_INLINE void glm_ivec2_addsub(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_addsubs(ivec2 a, int s, ivec2 dest) + CGLM_INLINE void glm_ivec2_mulsub(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_mulsubs(ivec2 a, int s, ivec2 dest) + CGLM_INLINE void glm_ivec2_maxsub(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_minsub(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE int glm_ivec2_distance2(ivec2 a, ivec2 b) + CGLM_INLINE float glm_ivec2_distance(ivec2 a, ivec2 b) + CGLM_INLINE void glm_ivec2_fill(ivec2 v, int val); + CGLM_INLINE bool glm_ivec2_eq(ivec2 v, int val); + CGLM_INLINE bool glm_ivec2_eqv(ivec2 a, ivec2 b); + CGLM_INLINE void glm_ivec2_maxv(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_minv(ivec2 a, ivec2 b, ivec2 dest) + CGLM_INLINE void glm_ivec2_clamp(ivec2 v, int minVal, int maxVal) + CGLM_INLINE void glm_ivec2_abs(ivec2 v, ivec2 dest) + */ + +#ifndef cglm_ivec2_h +#define cglm_ivec2_h + +#include "common.h" +#include "util.h" + +#define GLM_IVEC2_ONE_INIT {1, 1} +#define GLM_IVEC2_ZERO_INIT {0, 0} + +#define GLM_IVEC2_ONE ((ivec2)GLM_IVEC2_ONE_INIT) +#define GLM_IVEC2_ZERO ((ivec2)GLM_IVEC2_ZERO_INIT) + +/*! + * @brief init ivec2 using vec3 or vec4 + * + * @param[in] v vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec2(int * __restrict v, ivec2 dest) { + dest[0] = v[0]; + dest[1] = v[1]; +} + +/*! + * @brief copy all members of [a] to [dest] + * + * @param[in] a source vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec2_copy(ivec2 a, ivec2 dest) { + dest[0] = a[0]; + dest[1] = a[1]; +} + +/*! + * @brief set all members of [v] to zero + * + * @param[out] v vector + */ +CGLM_INLINE +void +glm_ivec2_zero(ivec2 v) { + v[0] = v[1] = 0; +} + +/*! + * @brief set all members of [v] to one + * + * @param[out] v vector + */ +CGLM_INLINE +void +glm_ivec2_one(ivec2 v) { + v[0] = v[1] = 1; +} + +/*! + * @brief ivec2 dot product + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return dot product + */ +CGLM_INLINE +int +glm_ivec2_dot(ivec2 a, ivec2 b) { + return a[0] * b[0] + a[1] * b[1]; +} + +/*! + * @brief ivec2 cross product + * + * REF: http://allenchou.net/2013/07/cross-product-of-2d-vectors/ + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return Z component of cross product + */ +CGLM_INLINE +int +glm_ivec2_cross(ivec2 a, ivec2 b) { + return a[0] * b[1] - a[1] * b[0]; +} + +/*! + * @brief add vector [a] to vector [b] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec2_add(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] = a[0] + b[0]; + dest[1] = a[1] + b[1]; +} + +/*! + * @brief add scalar s to vector [v] and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec2_adds(ivec2 v, int s, ivec2 dest) { + dest[0] = v[0] + s; + dest[1] = v[1] + s; +} + +/*! + * @brief subtract vector [b] from vector [a] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec2_sub(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] = a[0] - b[0]; + dest[1] = a[1] - b[1]; +} + +/*! + * @brief subtract scalar s from vector [v] and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec2_subs(ivec2 v, int s, ivec2 dest) { + dest[0] = v[0] - s; + dest[1] = v[1] - s; +} + +/*! + * @brief multiply vector [a] with vector [b] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec2_mul(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] = a[0] * b[0]; + dest[1] = a[1] * b[1]; +} + +/*! + * @brief multiply vector [a] with scalar s and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec2_scale(ivec2 v, int s, ivec2 dest) { + dest[0] = v[0] * s; + dest[1] = v[1] * s; +} + +/*! + * @brief div vector with another component-wise division: d = a / b + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest result = (a[0]/b[0], a[1]/b[1]) + */ +CGLM_INLINE +void +glm_ivec2_div(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] = a[0] / b[0]; + dest[1] = a[1] / b[1]; +} + +/*! + * @brief div vector with scalar: d = v / s + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest result = (a[0]/s, a[1]/s) + */ +CGLM_INLINE +void +glm_ivec2_divs(ivec2 v, int s, ivec2 dest) { + dest[0] = v[0] / s; + dest[1] = v[1] / s; +} + +/*! + * @brief mod vector with another component-wise modulo: d = a % b + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest result = (a[0]%b[0], a[1]%b[1]) + */ +CGLM_INLINE +void +glm_ivec2_mod(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] = a[0] % b[0]; + dest[1] = a[1] % b[1]; +} + +/*! + * @brief add vector [a] with vector [b] and add result to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest += (a + b) + */ +CGLM_INLINE +void +glm_ivec2_addadd(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] += a[0] + b[0]; + dest[1] += a[1] + b[1]; +} + +/*! + * @brief add scalar [s] onto vector [a] and add result to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest += (a + s) + */ +CGLM_INLINE +void +glm_ivec2_addadds(ivec2 a, int s, ivec2 dest) { + dest[0] += a[0] + s; + dest[1] += a[1] + s; +} + +/*! + * @brief subtract vector [a] from vector [b] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest += (a - b) + */ +CGLM_INLINE +void +glm_ivec2_subadd(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] += a[0] - b[0]; + dest[1] += a[1] - b[1]; +} + +/*! + * @brief subtract scalar [s] from vector [a] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first + * @param[in] s scalar + * @param[out] dest dest += (a - s) + */ +CGLM_INLINE +void +glm_ivec2_subadds(ivec2 a, int s, ivec2 dest) { + dest[0] += a[0] - s; + dest[1] += a[1] - s; +} + +/*! + * @brief multiply vector [a] with vector [b] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest += (a * b) + */ +CGLM_INLINE +void +glm_ivec2_muladd(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] += a[0] * b[0]; + dest[1] += a[1] * b[1]; +} + +/*! + * @brief multiply vector [a] with scalar [s] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest += (a * s) + */ +CGLM_INLINE +void +glm_ivec2_muladds(ivec2 a, int s, ivec2 dest) { + dest[0] += a[0] * s; + dest[1] += a[1] * s; +} + +/*! + * @brief add maximum of vector [a] and vector [b] to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest += max(a, b) + */ +CGLM_INLINE +void +glm_ivec2_maxadd(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] += glm_imax(a[0], b[0]); + dest[1] += glm_imax(a[1], b[1]); +} + +/*! + * @brief add minimum of vector [a] and vector [b] to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest += min(a, b) + */ +CGLM_INLINE +void +glm_ivec2_minadd(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] += glm_imin(a[0], b[0]); + dest[1] += glm_imin(a[1], b[1]); +} + +/*! + * @brief subtract vector [a] from vector [b] and subtract result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest -= (a - b) + */ +CGLM_INLINE +void +glm_ivec2_subsub(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] -= a[0] - b[0]; + dest[1] -= a[1] - b[1]; +} + +/*! + * @brief subtract scalar [s] from vector [a] and subtract result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest -= (a - s) + */ +CGLM_INLINE +void +glm_ivec2_subsubs(ivec2 a, int s, ivec2 dest) { + dest[0] -= a[0] - s; + dest[1] -= a[1] - s; +} + +/*! + * @brief add vector [a] to vector [b] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] b scalar + * @param[out] dest dest -= (a + b) + */ +CGLM_INLINE +void +glm_ivec2_addsub(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] -= a[0] + b[0]; + dest[1] -= a[1] + b[1]; +} + +/*! + * @brief add scalar [s] to vector [a] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest -= (a + b) + */ +CGLM_INLINE +void +glm_ivec2_addsubs(ivec2 a, int s, ivec2 dest) { + dest[0] -= a[0] + s; + dest[1] -= a[1] + s; +} + +/*! + * @brief multiply vector [a] and vector [b] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] b scalar + * @param[out] dest dest -= (a * b) + */ +CGLM_INLINE +void +glm_ivec2_mulsub(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] -= a[0] * b[0]; + dest[1] -= a[1] * b[1]; +} + +/*! + * @brief multiply vector [a] with scalar [s] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest -= (a * s) + */ +CGLM_INLINE +void +glm_ivec2_mulsubs(ivec2 a, int s, ivec2 dest) { + dest[0] -= a[0] * s; + dest[1] -= a[1] * s; +} + +/*! + * @brief subtract maximum of vector [a] and vector [b] from vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest -= max(a, b) + */ +CGLM_INLINE +void +glm_ivec2_maxsub(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] -= glm_imax(a[0], b[0]); + dest[1] -= glm_imax(a[1], b[1]); +} + +/*! + * @brief subtract minimum of vector [a] and vector [b] from vector [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest -= min(a, b) + */ +CGLM_INLINE +void +glm_ivec2_minsub(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] -= glm_imin(a[0], b[0]); + dest[1] -= glm_imin(a[1], b[1]); +} + +/*! + * @brief squared distance between two vectors + * + * @param[in] a first vector + * @param[in] b second vector + * @return returns squared distance (distance * distance) + */ +CGLM_INLINE +int +glm_ivec2_distance2(ivec2 a, ivec2 b) { + int xd, yd; + xd = a[0] - b[0]; + yd = a[1] - b[1]; + return xd * xd + yd * yd; +} + +/*! + * @brief distance between two vectors + * + * @param[in] a first vector + * @param[in] b second vector + * @return returns distance + */ +CGLM_INLINE +float +glm_ivec2_distance(ivec2 a, ivec2 b) { + return sqrtf((float)glm_ivec2_distance2(a, b)); +} + + +/*! + * @brief fill a vector with specified value + * + * @param[out] v dest + * @param[in] val value + */ +CGLM_INLINE +void +glm_ivec2_fill(ivec2 v, int val) { + v[0] = v[1] = val; +} + +/*! + * @brief check if vector is equal to value + * + * @param[in] v vector + * @param[in] val value + */ +CGLM_INLINE +bool +glm_ivec2_eq(ivec2 v, int val) { + return v[0] == val && v[0] == v[1]; +} + +/*! + * @brief check if vector is equal to another + * + * @param[in] a vector + * @param[in] b vector + */ +CGLM_INLINE +bool +glm_ivec2_eqv(ivec2 a, ivec2 b) { + return a[0] == b[0] + && a[1] == b[1]; +} + +/*! + * @brief set each member of dest to greater of vector a and b + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec2_maxv(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] = a[0] > b[0] ? a[0] : b[0]; + dest[1] = a[1] > b[1] ? a[1] : b[1]; +} + +/*! + * @brief set each member of dest to lesser of vector a and b + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec2_minv(ivec2 a, ivec2 b, ivec2 dest) { + dest[0] = a[0] < b[0] ? a[0] : b[0]; + dest[1] = a[1] < b[1] ? a[1] : b[1]; +} + +/*! + * @brief clamp each member of [v] between minVal and maxVal (inclusive) + * + * @param[in, out] v vector + * @param[in] minVal minimum value + * @param[in] maxVal maximum value + */ +CGLM_INLINE +void +glm_ivec2_clamp(ivec2 v, int minVal, int maxVal) { + if (v[0] < minVal) + v[0] = minVal; + else if(v[0] > maxVal) + v[0] = maxVal; + + if (v[1] < minVal) + v[1] = minVal; + else if(v[1] > maxVal) + v[1] = maxVal; +} + +/*! + * @brief absolute value of v + * + * @param[in] v vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec2_abs(ivec2 v, ivec2 dest) { + dest[0] = abs(v[0]); + dest[1] = abs(v[1]); +} + +#endif /* cglm_ivec2_h */ diff --git a/cglm/include/cglm/ivec3.h b/cglm/include/cglm/ivec3.h new file mode 100644 index 0000000..67eaa22 --- /dev/null +++ b/cglm/include/cglm/ivec3.h @@ -0,0 +1,713 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_IVEC3_ONE_INIT + GLM_IVEC3_ZERO_INIT + GLM_IVEC3_ONE + GLM_IVEC3_ZERO + + Functions: + CGLM_INLINE void glm_ivec3(ivec4 v4, ivec3 dest) + CGLM_INLINE void glm_ivec3_copy(ivec3 a, ivec3 dest) + CGLM_INLINE void glm_ivec3_zero(ivec3 v) + CGLM_INLINE void glm_ivec3_one(ivec3 v) + CGLM_INLINE int glm_ivec3_dot(ivec3 a, ivec3 b) + CGLM_INLINE int glm_ivec3_norm2(ivec3 v) + CGLM_INLINE int glm_ivec3_norm(ivec3 v) + CGLM_INLINE void glm_ivec3_add(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_adds(ivec3 v, int s, ivec3 dest) + CGLM_INLINE void glm_ivec3_sub(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_subs(ivec3 v, int s, ivec3 dest) + CGLM_INLINE void glm_ivec3_mul(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_scale(ivec3 v, int s, ivec3 dest) + CGLM_INLINE void glm_ivec3_div(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_divs(ivec3 v, int s, ivec3 dest) + CGLM_INLINE void glm_ivec3_mod(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_addadd(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_addadds(ivec3 a, int s, ivec3 dest) + CGLM_INLINE void glm_ivec3_subadd(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_subadds(ivec3 a, int s, ivec3 dest) + CGLM_INLINE void glm_ivec3_muladd(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_muladds(ivec3 a, int s, ivec3 dest) + CGLM_INLINE void glm_ivec3_maxadd(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_minadd(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_subsub(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_subsubs(ivec3 a, int s, ivec3 dest) + CGLM_INLINE void glm_ivec3_addsub(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_addsubs(ivec3 a, int s, ivec3 dest) + CGLM_INLINE void glm_ivec3_mulsub(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_mulsubs(ivec3 a, int s, ivec3 dest) + CGLM_INLINE void glm_ivec3_maxsub(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_minsub(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE int glm_ivec3_distance2(ivec3 a, ivec3 b) + CGLM_INLINE float glm_ivec3_distance(ivec3 a, ivec3 b) + CGLM_INLINE void glm_ivec3_fill(ivec3 v, int val); + CGLM_INLINE bool glm_ivec3_eq(ivec3 v, int val); + CGLM_INLINE bool glm_ivec3_eqv(ivec3 a, ivec3 b); + CGLM_INLINE void glm_ivec3_maxv(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_minv(ivec3 a, ivec3 b, ivec3 dest) + CGLM_INLINE void glm_ivec3_clamp(ivec3 v, int minVal, int maxVal) + CGLM_INLINE void glm_ivec3_abs(ivec3 v, ivec3 dest) + */ + +#ifndef cglm_ivec3_h +#define cglm_ivec3_h + +#include "common.h" +#include "util.h" + +#define GLM_IVEC3_ONE_INIT {1, 1, 1} +#define GLM_IVEC3_ZERO_INIT {0, 0, 0} + +#define GLM_IVEC3_ONE ((ivec3)GLM_IVEC3_ONE_INIT) +#define GLM_IVEC3_ZERO ((ivec3)GLM_IVEC3_ZERO_INIT) + +/*! + * @brief init ivec3 using ivec4 + * + * @param[in] v4 vector4 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec3(ivec4 v4, ivec3 dest) { + dest[0] = v4[0]; + dest[1] = v4[1]; + dest[2] = v4[2]; +} + +/*! + * @brief copy all members of [a] to [dest] + * + * @param[in] a source vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec3_copy(ivec3 a, ivec3 dest) { + dest[0] = a[0]; + dest[1] = a[1]; + dest[2] = a[2]; +} + +/*! + * @brief set all members of [v] to zero + * + * @param[out] v vector + */ +CGLM_INLINE +void +glm_ivec3_zero(ivec3 v) { + v[0] = v[1] = v[2] = 0; +} + +/*! + * @brief set all members of [v] to one + * + * @param[out] v vector + */ +CGLM_INLINE +void +glm_ivec3_one(ivec3 v) { + v[0] = v[1] = v[2] = 1; +} + +/*! + * @brief ivec3 dot product + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return dot product + */ +CGLM_INLINE +int +glm_ivec3_dot(ivec3 a, ivec3 b) { + return a[0] * b[0] + a[1] * b[1] + a[2] * b[2]; +} + +/*! + * @brief norm * norm (magnitude) of vec + * + * we can use this func instead of calling norm * norm, because it would call + * sqrtf function twice but with this func we can avoid func call, maybe this is + * not good name for this func + * + * @param[in] v vector + * + * @return norm * norm + */ +CGLM_INLINE +int +glm_ivec3_norm2(ivec3 v) { + return glm_ivec3_dot(v, v); +} + +/*! + * @brief euclidean norm (magnitude), also called L2 norm + * this will give magnitude of vector in euclidean space + * + * @param[in] v vector + * + * @return norm + */ +CGLM_INLINE +int +glm_ivec3_norm(ivec3 v) { + return (int)sqrtf((float)glm_ivec3_norm2(v)); +} + +/*! + * @brief add vector [a] to vector [b] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec3_add(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] = a[0] + b[0]; + dest[1] = a[1] + b[1]; + dest[2] = a[2] + b[2]; +} + +/*! + * @brief add scalar s to vector [v] and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec3_adds(ivec3 v, int s, ivec3 dest) { + dest[0] = v[0] + s; + dest[1] = v[1] + s; + dest[2] = v[2] + s; +} + +/*! + * @brief subtract vector [b] from vector [a] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec3_sub(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] = a[0] - b[0]; + dest[1] = a[1] - b[1]; + dest[2] = a[2] - b[2]; +} + +/*! + * @brief subtract scalar s from vector [v] and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec3_subs(ivec3 v, int s, ivec3 dest) { + dest[0] = v[0] - s; + dest[1] = v[1] - s; + dest[2] = v[2] - s; +} + +/*! + * @brief multiply vector [a] with vector [b] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec3_mul(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] = a[0] * b[0]; + dest[1] = a[1] * b[1]; + dest[2] = a[2] * b[2]; +} + +/*! + * @brief multiply vector [a] with scalar s and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec3_scale(ivec3 v, int s, ivec3 dest) { + dest[0] = v[0] * s; + dest[1] = v[1] * s; + dest[2] = v[2] * s; +} + +/*! + * @brief div vector with another component-wise division: d = a / b + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest result = (a[0]/b[0], a[1]/b[1], a[2]/b[2]) + */ +CGLM_INLINE +void +glm_ivec3_div(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] = a[0] / b[0]; + dest[1] = a[1] / b[1]; + dest[2] = a[2] / b[2]; +} + +/*! + * @brief div vector with scalar: d = v / s + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest result = (a[0]/s, a[1]/s, a[2]/s) + */ +CGLM_INLINE +void +glm_ivec3_divs(ivec3 v, int s, ivec3 dest) { + dest[0] = v[0] / s; + dest[1] = v[1] / s; + dest[2] = v[2] / s; +} + +/*! + * @brief Element-wise modulo operation on ivec3 vectors: dest = a % b + * + * Performs element-wise modulo on each component of vectors `a` and `b`. + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest result = (a[0]%b[0], a[1]%b[1], a[2]%b[2]) + */ +CGLM_INLINE +void +glm_ivec3_mod(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] = a[0] % b[0]; + dest[1] = a[1] % b[1]; + dest[2] = a[2] % b[2]; +} + +/*! + * @brief add vector [a] with vector [b] and add result to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest += (a + b) + */ +CGLM_INLINE +void +glm_ivec3_addadd(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] += a[0] + b[0]; + dest[1] += a[1] + b[1]; + dest[2] += a[2] + b[2]; +} + +/*! + * @brief add scalar [s] onto vector [a] and add result to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest += (a + s) + */ +CGLM_INLINE +void +glm_ivec3_addadds(ivec3 a, int s, ivec3 dest) { + dest[0] += a[0] + s; + dest[1] += a[1] + s; + dest[2] += a[2] + s; +} + +/*! + * @brief subtract vector [a] from vector [b] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest += (a - b) + */ +CGLM_INLINE +void +glm_ivec3_subadd(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] += a[0] - b[0]; + dest[1] += a[1] - b[1]; + dest[2] += a[2] - b[2]; +} + +/*! + * @brief subtract scalar [s] from vector [a] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first + * @param[in] s scalar + * @param[out] dest dest += (a - s) + */ +CGLM_INLINE +void +glm_ivec3_subadds(ivec3 a, int s, ivec3 dest) { + dest[0] += a[0] - s; + dest[1] += a[1] - s; + dest[2] += a[2] - s; +} + +/*! + * @brief multiply vector [a] with vector [b] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest += (a * b) + */ +CGLM_INLINE +void +glm_ivec3_muladd(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] += a[0] * b[0]; + dest[1] += a[1] * b[1]; + dest[2] += a[2] * b[2]; +} + +/*! + * @brief multiply vector [a] with scalar [s] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest += (a * s) + */ +CGLM_INLINE +void +glm_ivec3_muladds(ivec3 a, int s, ivec3 dest) { + dest[0] += a[0] * s; + dest[1] += a[1] * s; + dest[2] += a[2] * s; +} + +/*! + * @brief add maximum of vector [a] and vector [b] to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest += max(a, b) + */ +CGLM_INLINE +void +glm_ivec3_maxadd(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] += glm_imax(a[0], b[0]); + dest[1] += glm_imax(a[1], b[1]); + dest[2] += glm_imax(a[2], b[2]); +} + +/*! + * @brief add minimum of vector [a] and vector [b] to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest += min(a, b) + */ +CGLM_INLINE +void +glm_ivec3_minadd(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] += glm_imin(a[0], b[0]); + dest[1] += glm_imin(a[1], b[1]); + dest[2] += glm_imin(a[2], b[2]); +} + +/*! + * @brief subtract vector [a] from vector [b] and subtract result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest -= (a - b) + */ +CGLM_INLINE +void +glm_ivec3_subsub(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] -= a[0] - b[0]; + dest[1] -= a[1] - b[1]; + dest[2] -= a[2] - b[2]; +} + +/*! + * @brief subtract scalar [s] from vector [a] and subtract result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest -= (a - s) + */ +CGLM_INLINE +void +glm_ivec3_subsubs(ivec3 a, int s, ivec3 dest) { + dest[0] -= a[0] - s; + dest[1] -= a[1] - s; + dest[2] -= a[2] - s; +} + +/*! + * @brief add vector [a] to vector [b] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] b scalar + * @param[out] dest dest -= (a + b) + */ +CGLM_INLINE +void +glm_ivec3_addsub(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] -= a[0] + b[0]; + dest[1] -= a[1] + b[1]; + dest[2] -= a[2] + b[2]; +} + +/*! + * @brief add scalar [s] to vector [a] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest -= (a + b) + */ +CGLM_INLINE +void +glm_ivec3_addsubs(ivec3 a, int s, ivec3 dest) { + dest[0] -= a[0] + s; + dest[1] -= a[1] + s; + dest[2] -= a[2] + s; +} + +/*! + * @brief multiply vector [a] and vector [b] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] b scalar + * @param[out] dest dest -= (a * b) + */ +CGLM_INLINE +void +glm_ivec3_mulsub(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] -= a[0] * b[0]; + dest[1] -= a[1] * b[1]; + dest[2] -= a[2] * b[2]; +} + +/*! + * @brief multiply vector [a] with scalar [s] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest -= (a * s) + */ +CGLM_INLINE +void +glm_ivec3_mulsubs(ivec3 a, int s, ivec3 dest) { + dest[0] -= a[0] * s; + dest[1] -= a[1] * s; + dest[2] -= a[2] * s; +} + +/*! + * @brief subtract maximum of vector [a] and vector [b] from vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest -= max(a, b) + */ +CGLM_INLINE +void +glm_ivec3_maxsub(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] -= glm_imax(a[0], b[0]); + dest[1] -= glm_imax(a[1], b[1]); + dest[2] -= glm_imax(a[2], b[2]); +} + +/*! + * @brief subtract minimum of vector [a] and vector [b] from vector [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest -= min(a, b) + */ +CGLM_INLINE +void +glm_ivec3_minsub(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] -= glm_imin(a[0], b[0]); + dest[1] -= glm_imin(a[1], b[1]); + dest[2] -= glm_imin(a[2], b[2]); +} + +/*! + * @brief squared distance between two vectors + * + * @param[in] a first vector + * @param[in] b second vector + * @return returns squared distance (distance * distance) + */ +CGLM_INLINE +int +glm_ivec3_distance2(ivec3 a, ivec3 b) { + int xd, yd, zd; + xd = a[0] - b[0]; + yd = a[1] - b[1]; + zd = a[2] - b[2]; + return xd * xd + yd * yd + zd * zd; +} + +/*! + * @brief distance between two vectors + * + * @param[in] a first vector + * @param[in] b second vector + * @return returns distance + */ +CGLM_INLINE +float +glm_ivec3_distance(ivec3 a, ivec3 b) { + return sqrtf((float)glm_ivec3_distance2(a, b)); +} + +/*! + * @brief fill a vector with specified value + * + * @param[out] v dest + * @param[in] val value + */ +CGLM_INLINE +void +glm_ivec3_fill(ivec3 v, int val) { + v[0] = v[1] = v[2] = val; +} + +/*! + * @brief check if vector is equal to value + * + * @param[in] v vector + * @param[in] val value + */ +CGLM_INLINE +bool +glm_ivec3_eq(ivec3 v, int val) { + return v[0] == val && v[0] == v[1] && v[0] == v[2]; +} + +/*! + * @brief check if vector is equal to another + * + * @param[in] a vector + * @param[in] b vector + */ +CGLM_INLINE +bool +glm_ivec3_eqv(ivec3 a, ivec3 b) { + return a[0] == b[0] + && a[1] == b[1] + && a[2] == b[2]; +} + +/*! + * @brief set each member of dest to greater of vector a and b + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec3_maxv(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] = a[0] > b[0] ? a[0] : b[0]; + dest[1] = a[1] > b[1] ? a[1] : b[1]; + dest[2] = a[2] > b[2] ? a[2] : b[2]; +} + +/*! + * @brief set each member of dest to lesser of vector a and b + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec3_minv(ivec3 a, ivec3 b, ivec3 dest) { + dest[0] = a[0] < b[0] ? a[0] : b[0]; + dest[1] = a[1] < b[1] ? a[1] : b[1]; + dest[2] = a[2] < b[2] ? a[2] : b[2]; +} + +/*! + * @brief clamp each member of [v] between minVal and maxVal (inclusive) + * + * @param[in, out] v vector + * @param[in] minVal minimum value + * @param[in] maxVal maximum value + */ +CGLM_INLINE +void +glm_ivec3_clamp(ivec3 v, int minVal, int maxVal) { + if (v[0] < minVal) + v[0] = minVal; + else if(v[0] > maxVal) + v[0] = maxVal; + + if (v[1] < minVal) + v[1] = minVal; + else if(v[1] > maxVal) + v[1] = maxVal; + + if (v[2] < minVal) + v[2] = minVal; + else if(v[2] > maxVal) + v[2] = maxVal; +} + +/*! + * @brief absolute value of v + * + * @param[in] v vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec3_abs(ivec3 v, ivec3 dest) { + dest[0] = abs(v[0]); + dest[1] = abs(v[1]); + dest[2] = abs(v[2]); +} + +#endif /* cglm_ivec3_h */ diff --git a/cglm/include/cglm/ivec4.h b/cglm/include/cglm/ivec4.h new file mode 100644 index 0000000..6357599 --- /dev/null +++ b/cglm/include/cglm/ivec4.h @@ -0,0 +1,608 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_IVEC4_ONE_INIT + GLM_IVEC4_ZERO_INIT + GLM_IVEC4_ONE + GLM_IVEC4_ZERO + + Functions: + CGLM_INLINE void glm_ivec4(ivec3 v3, int last, ivec4 dest) + CGLM_INLINE void glm_ivec4_copy(ivec4 a, ivec4 dest) + CGLM_INLINE void glm_ivec4_zero(ivec4 v) + CGLM_INLINE void glm_ivec4_one(ivec4 v) + CGLM_INLINE void glm_ivec4_add(ivec4 a, ivec4 b, ivec4 dest) + CGLM_INLINE void glm_ivec4_adds(ivec4 v, int s, ivec4 dest) + CGLM_INLINE void glm_ivec4_sub(ivec4 a, ivec4 b, ivec4 dest) + CGLM_INLINE void glm_ivec4_subs(ivec4 v, int s, ivec4 dest) + CGLM_INLINE void glm_ivec4_mul(ivec4 a, ivec4 b, ivec4 dest) + CGLM_INLINE void glm_ivec4_scale(ivec4 v, int s, ivec4 dest) + CGLM_INLINE void glm_ivec4_addadd(ivec4 a, ivec4 b, ivec4 dest) + CGLM_INLINE void glm_ivec4_addadds(ivec4 a, int s, ivec4 dest) + CGLM_INLINE void glm_ivec4_subadd(ivec4 a, ivec4 b, ivec4 dest) + CGLM_INLINE void glm_ivec4_subadds(ivec4 a, int s, ivec4 dest) + CGLM_INLINE void glm_ivec4_muladd(ivec4 a, ivec4 b, ivec4 dest) + CGLM_INLINE void glm_ivec4_muladds(ivec4 a, int s, ivec4 dest) + CGLM_INLINE void glm_ivec4_maxadd(ivec4 a, ivec4 b, ivec4 dest) + CGLM_INLINE void glm_ivec4_minadd(ivec4 a, ivec4 b, ivec4 dest) + CGLM_INLINE void glm_ivec4_subsub(ivec4 a, ivec4 b, ivec4 dest) + CGLM_INLINE void glm_ivec4_subsubs(ivec4 a, int s, ivec4 dest) + CGLM_INLINE void glm_ivec4_addsub(ivec4 a, ivec4 b, ivec4 dest) + CGLM_INLINE void glm_ivec4_addsubs(ivec4 a, int s, ivec4 dest) + CGLM_INLINE void glm_ivec4_mulsub(ivec4 a, ivec4 b, ivec4 dest) + CGLM_INLINE void glm_ivec4_mulsubs(ivec4 a, int s, ivec4 dest) + CGLM_INLINE void glm_ivec4_maxsub(ivec4 a, ivec4 b, ivec4 dest) + CGLM_INLINE void glm_ivec4_minsub(ivec4 a, ivec4 b, ivec4 dest) + CGLM_INLINE int glm_ivec4_distance2(ivec4 a, ivec4 b) + CGLM_INLINE float glm_ivec4_distance(ivec4 a, ivec4 b) + CGLM_INLINE void glm_ivec4_maxv(ivec4 a, ivec4 b, ivec4 dest) + CGLM_INLINE void glm_ivec4_minv(ivec4 a, ivec4 b, ivec4 dest) + CGLM_INLINE void glm_ivec4_clamp(ivec4 v, int minVal, int maxVal) + CGLM_INLINE void glm_ivec4_abs(ivec4 v, ivec4 dest) + */ + +#ifndef cglm_ivec4_h +#define cglm_ivec4_h + +#include "common.h" +#include "util.h" + +#define GLM_IVEC4_ONE_INIT {1, 1, 1, 1} +#define GLM_IVEC4_ZERO_INIT {0, 0, 0, 0} + +#define GLM_IVEC4_ONE ((ivec4)GLM_IVEC4_ONE_INIT) +#define GLM_IVEC4_ZERO ((ivec4)GLM_IVEC4_ZERO_INIT) + +/*! + * @brief init ivec4 using ivec3 + * + * @param[in] v3 vector3 + * @param[in] last last item + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec4(ivec3 v3, int last, ivec4 dest) { + dest[0] = v3[0]; + dest[1] = v3[1]; + dest[2] = v3[2]; + dest[3] = last; +} + +/*! + * @brief copy all members of [a] to [dest] + * + * @param[in] a source vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec4_copy(ivec4 a, ivec4 dest) { + dest[0] = a[0]; + dest[1] = a[1]; + dest[2] = a[2]; + dest[3] = a[3]; +} + +/*! + * @brief set all members of [v] to zero + * + * @param[out] v vector + */ +CGLM_INLINE +void +glm_ivec4_zero(ivec4 v) { + v[0] = v[1] = v[2] = v[3] = 0; +} + +/*! + * @brief set all members of [v] to one + * + * @param[out] v vector + */ +CGLM_INLINE +void +glm_ivec4_one(ivec4 v) { + v[0] = v[1] = v[2] = v[3] = 1; +} + +/*! + * @brief add vector [a] to vector [b] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec4_add(ivec4 a, ivec4 b, ivec4 dest) { + dest[0] = a[0] + b[0]; + dest[1] = a[1] + b[1]; + dest[2] = a[2] + b[2]; + dest[3] = a[3] + b[3]; +} + +/*! + * @brief add scalar s to vector [v] and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec4_adds(ivec4 v, int s, ivec4 dest) { + dest[0] = v[0] + s; + dest[1] = v[1] + s; + dest[2] = v[2] + s; + dest[3] = v[3] + s; +} + +/*! + * @brief subtract vector [b] from vector [a] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec4_sub(ivec4 a, ivec4 b, ivec4 dest) { + dest[0] = a[0] - b[0]; + dest[1] = a[1] - b[1]; + dest[2] = a[2] - b[2]; + dest[3] = a[3] - b[3]; +} + +/*! + * @brief subtract scalar s from vector [v] and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec4_subs(ivec4 v, int s, ivec4 dest) { + dest[0] = v[0] - s; + dest[1] = v[1] - s; + dest[2] = v[2] - s; + dest[3] = v[3] - s; +} + +/*! + * @brief multiply vector [a] with vector [b] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec4_mul(ivec4 a, ivec4 b, ivec4 dest) { + dest[0] = a[0] * b[0]; + dest[1] = a[1] * b[1]; + dest[2] = a[2] * b[2]; + dest[3] = a[3] * b[3]; +} + +/*! + * @brief multiply vector [a] with scalar s and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec4_scale(ivec4 v, int s, ivec4 dest) { + dest[0] = v[0] * s; + dest[1] = v[1] * s; + dest[2] = v[2] * s; + dest[3] = v[3] * s; +} + +/*! + * @brief add vector [a] with vector [b] and add result to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest += (a + b) + */ +CGLM_INLINE +void +glm_ivec4_addadd(ivec4 a, ivec4 b, ivec4 dest) { + dest[0] += a[0] + b[0]; + dest[1] += a[1] + b[1]; + dest[2] += a[2] + b[2]; + dest[3] += a[3] + b[3]; +} + +/*! + * @brief add scalar [s] onto vector [a] and add result to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest += (a + s) + */ +CGLM_INLINE +void +glm_ivec4_addadds(ivec4 a, int s, ivec4 dest) { + dest[0] += a[0] + s; + dest[1] += a[1] + s; + dest[2] += a[2] + s; + dest[3] += a[3] + s; +} + +/*! + * @brief subtract vector [a] from vector [b] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest += (a - b) + */ +CGLM_INLINE +void +glm_ivec4_subadd(ivec4 a, ivec4 b, ivec4 dest) { + dest[0] += a[0] - b[0]; + dest[1] += a[1] - b[1]; + dest[2] += a[2] - b[2]; + dest[3] += a[3] - b[3]; +} + +/*! + * @brief subtract scalar [s] from vector [a] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first + * @param[in] s scalar + * @param[out] dest dest += (a - s) + */ +CGLM_INLINE +void +glm_ivec4_subadds(ivec4 a, int s, ivec4 dest) { + dest[0] += a[0] - s; + dest[1] += a[1] - s; + dest[2] += a[2] - s; + dest[3] += a[3] - s; +} + +/*! + * @brief multiply vector [a] with vector [b] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest += (a * b) + */ +CGLM_INLINE +void +glm_ivec4_muladd(ivec4 a, ivec4 b, ivec4 dest) { + dest[0] += a[0] * b[0]; + dest[1] += a[1] * b[1]; + dest[2] += a[2] * b[2]; + dest[3] += a[3] * b[3]; +} + +/*! + * @brief multiply vector [a] with scalar [s] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest += (a * s) + */ +CGLM_INLINE +void +glm_ivec4_muladds(ivec4 a, int s, ivec4 dest) { + dest[0] += a[0] * s; + dest[1] += a[1] * s; + dest[2] += a[2] * s; + dest[3] += a[3] * s; +} + +/*! + * @brief add maximum of vector [a] and vector [b] to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest += max(a, b) + */ +CGLM_INLINE +void +glm_ivec4_maxadd(ivec4 a, ivec4 b, ivec4 dest) { + dest[0] += glm_imax(a[0], b[0]); + dest[1] += glm_imax(a[1], b[1]); + dest[2] += glm_imax(a[2], b[2]); + dest[3] += glm_imax(a[3], b[3]); +} + +/*! + * @brief add minimum of vector [a] and vector [b] to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest += min(a, b) + */ +CGLM_INLINE +void +glm_ivec4_minadd(ivec4 a, ivec4 b, ivec4 dest) { + dest[0] += glm_imin(a[0], b[0]); + dest[1] += glm_imin(a[1], b[1]); + dest[2] += glm_imin(a[2], b[2]); + dest[3] += glm_imin(a[3], b[3]); +} + +/*! + * @brief subtract vector [a] from vector [b] and subtract result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest -= (a - b) + */ +CGLM_INLINE +void +glm_ivec4_subsub(ivec4 a, ivec4 b, ivec4 dest) { + dest[0] -= a[0] - b[0]; + dest[1] -= a[1] - b[1]; + dest[2] -= a[2] - b[2]; + dest[3] -= a[3] - b[3]; +} + +/*! + * @brief subtract scalar [s] from vector [a] and subtract result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest -= (a - s) + */ +CGLM_INLINE +void +glm_ivec4_subsubs(ivec4 a, int s, ivec4 dest) { + dest[0] -= a[0] - s; + dest[1] -= a[1] - s; + dest[2] -= a[2] - s; + dest[3] -= a[3] - s; +} + +/*! + * @brief add vector [a] to vector [b] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] b scalar + * @param[out] dest dest -= (a + b) + */ +CGLM_INLINE +void +glm_ivec4_addsub(ivec4 a, ivec4 b, ivec4 dest) { + dest[0] -= a[0] + b[0]; + dest[1] -= a[1] + b[1]; + dest[2] -= a[2] + b[2]; + dest[3] -= a[3] + b[3]; +} + +/*! + * @brief add scalar [s] to vector [a] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest -= (a + b) + */ +CGLM_INLINE +void +glm_ivec4_addsubs(ivec4 a, int s, ivec4 dest) { + dest[0] -= a[0] + s; + dest[1] -= a[1] + s; + dest[2] -= a[2] + s; + dest[3] -= a[3] + s; +} + +/*! + * @brief multiply vector [a] and vector [b] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] b scalar + * @param[out] dest dest -= (a * b) + */ +CGLM_INLINE +void +glm_ivec4_mulsub(ivec4 a, ivec4 b, ivec4 dest) { + dest[0] -= a[0] * b[0]; + dest[1] -= a[1] * b[1]; + dest[2] -= a[2] * b[2]; + dest[3] -= a[3] * b[3]; +} + +/*! + * @brief multiply vector [a] with scalar [s] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest -= (a * s) + */ +CGLM_INLINE +void +glm_ivec4_mulsubs(ivec4 a, int s, ivec4 dest) { + dest[0] -= a[0] * s; + dest[1] -= a[1] * s; + dest[2] -= a[2] * s; + dest[3] -= a[3] * s; +} + +/*! + * @brief subtract maximum of vector [a] and vector [b] from vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest -= max(a, b) + */ +CGLM_INLINE +void +glm_ivec4_maxsub(ivec4 a, ivec4 b, ivec4 dest) { + dest[0] -= glm_imax(a[0], b[0]); + dest[1] -= glm_imax(a[1], b[1]); + dest[2] -= glm_imax(a[2], b[2]); + dest[3] -= glm_imax(a[3], b[3]); +} + +/*! + * @brief subtract minimum of vector [a] and vector [b] from vector [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest dest -= min(a, b) + */ +CGLM_INLINE +void +glm_ivec4_minsub(ivec4 a, ivec4 b, ivec4 dest) { + dest[0] -= glm_imin(a[0], b[0]); + dest[1] -= glm_imin(a[1], b[1]); + dest[2] -= glm_imin(a[2], b[2]); + dest[3] -= glm_imin(a[3], b[3]); +} + +/*! + * @brief squared distance between two vectors + * + * @param[in] a first vector + * @param[in] b second vector + * @return returns squared distance (distance * distance) + */ +CGLM_INLINE +int +glm_ivec4_distance2(ivec4 a, ivec4 b) { + int xd, yd, zd, wd; + xd = a[0] - b[0]; + yd = a[1] - b[1]; + zd = a[2] - b[2]; + wd = a[3] - b[3]; + return xd * xd + yd * yd + zd * zd + wd * wd; +} + +/*! + * @brief distance between two vectors + * + * @param[in] a first vector + * @param[in] b second vector + * @return returns distance + */ +CGLM_INLINE +float +glm_ivec4_distance(ivec4 a, ivec4 b) { + return sqrtf((float)glm_ivec4_distance2(a, b)); +} + +/*! + * @brief set each member of dest to greater of vector a and b + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec4_maxv(ivec4 a, ivec4 b, ivec4 dest) { + dest[0] = a[0] > b[0] ? a[0] : b[0]; + dest[1] = a[1] > b[1] ? a[1] : b[1]; + dest[2] = a[2] > b[2] ? a[2] : b[2]; + dest[3] = a[3] > b[3] ? a[3] : b[3]; +} + +/*! + * @brief set each member of dest to lesser of vector a and b + * + * @param[in] a first vector + * @param[in] b second vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec4_minv(ivec4 a, ivec4 b, ivec4 dest) { + dest[0] = a[0] < b[0] ? a[0] : b[0]; + dest[1] = a[1] < b[1] ? a[1] : b[1]; + dest[2] = a[2] < b[2] ? a[2] : b[2]; + dest[3] = a[3] < b[3] ? a[3] : b[3]; +} + +/*! + * @brief clamp each member of [v] between minVal and maxVal (inclusive) + * + * @param[in, out] v vector + * @param[in] minVal minimum value + * @param[in] maxVal maximum value + */ +CGLM_INLINE +void +glm_ivec4_clamp(ivec4 v, int minVal, int maxVal) { + if (v[0] < minVal) + v[0] = minVal; + else if(v[0] > maxVal) + v[0] = maxVal; + + if (v[1] < minVal) + v[1] = minVal; + else if(v[1] > maxVal) + v[1] = maxVal; + + if (v[2] < minVal) + v[2] = minVal; + else if(v[2] > maxVal) + v[2] = maxVal; + + if (v[3] < minVal) + v[3] = minVal; + else if(v[3] > maxVal) + v[3] = maxVal; +} + +/*! + * @brief absolute value of v + * + * @param[in] v vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_ivec4_abs(ivec4 v, ivec4 dest) { + dest[0] = abs(v[0]); + dest[1] = abs(v[1]); + dest[2] = abs(v[2]); + dest[3] = abs(v[3]); +} + +#endif /* cglm_ivec4_h */ diff --git a/cglm/include/cglm/mat2.h b/cglm/include/cglm/mat2.h new file mode 100644 index 0000000..1da0cd4 --- /dev/null +++ b/cglm/include/cglm/mat2.h @@ -0,0 +1,364 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_MAT2_IDENTITY_INIT + GLM_MAT2_ZERO_INIT + GLM_MAT2_IDENTITY + GLM_MAT2_ZERO + + Functions: + CGLM_INLINE void glm_mat2_copy(mat2 mat, mat2 dest) + CGLM_INLINE void glm_mat2_identity(mat2 mat) + CGLM_INLINE void glm_mat2_identity_array(mat2 * restrict mat, size_t count) + CGLM_INLINE void glm_mat2_zero(mat2 mat) + CGLM_INLINE void glm_mat2_mul(mat2 m1, mat2 m2, mat2 dest) + CGLM_INLINE void glm_mat2_transpose_to(mat2 m, mat2 dest) + CGLM_INLINE void glm_mat2_transpose(mat2 m) + CGLM_INLINE void glm_mat2_mulv(mat2 m, vec2 v, vec2 dest) + CGLM_INLINE float glm_mat2_trace(mat2 m) + CGLM_INLINE void glm_mat2_scale(mat2 m, float s) + CGLM_INLINE float glm_mat2_det(mat2 mat) + CGLM_INLINE void glm_mat2_inv(mat2 mat, mat2 dest) + CGLM_INLINE void glm_mat2_swap_col(mat2 mat, int col1, int col2) + CGLM_INLINE void glm_mat2_swap_row(mat2 mat, int row1, int row2) + CGLM_INLINE float glm_mat2_rmc(vec2 r, mat2 m, vec2 c) + CGLM_INLINE void glm_mat2_make(float * restrict src, mat2 dest) + */ + +#ifndef cglm_mat2_h +#define cglm_mat2_h + +#include "common.h" +#include "vec2.h" + +#ifdef CGLM_SSE_FP +# include "simd/sse2/mat2.h" +#endif + +#ifdef CGLM_NEON_FP +# include "simd/neon/mat2.h" +#endif + +#ifdef CGLM_SIMD_WASM +# include "simd/wasm/mat2.h" +#endif + +#define GLM_MAT2_IDENTITY_INIT {{1.0f, 0.0f}, {0.0f, 1.0f}} +#define GLM_MAT2_ZERO_INIT {{0.0f, 0.0f}, {0.0f, 0.0f}} + +/* for C only */ +#define GLM_MAT2_IDENTITY ((mat2)GLM_MAT2_IDENTITY_INIT) +#define GLM_MAT2_ZERO ((mat2)GLM_MAT2_ZERO_INIT) + +/*! + * @brief copy all members of [mat] to [dest] + * + * @param[in] mat source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_mat2_copy(mat2 mat, mat2 dest) { + glm_vec4_ucopy(mat[0], dest[0]); +} + +/*! + * @brief make given matrix identity. It is identical with below, + * but it is more easy to do that with this func especially for members + * e.g. glm_mat2_identity(aStruct->aMatrix); + * + * @code + * glm_mat2_copy(GLM_MAT2_IDENTITY, mat); // C only + * + * // or + * mat2 mat = GLM_MAT2_IDENTITY_INIT; + * @endcode + * + * @param[in, out] mat destination + */ +CGLM_INLINE +void +glm_mat2_identity(mat2 mat) { + CGLM_ALIGN_MAT mat2 t = GLM_MAT2_IDENTITY_INIT; + glm_mat2_copy(t, mat); +} + +/*! + * @brief make given matrix array's each element identity matrix + * + * @param[in, out] mat matrix array (must be aligned (16) + * if alignment is not disabled) + * + * @param[in] count count of matrices + */ +CGLM_INLINE +void +glm_mat2_identity_array(mat2 * __restrict mat, size_t count) { + CGLM_ALIGN_MAT mat2 t = GLM_MAT2_IDENTITY_INIT; + size_t i; + + for (i = 0; i < count; i++) { + glm_mat2_copy(t, mat[i]); + } +} + +/*! + * @brief make given matrix zero. + * + * @param[in, out] mat matrix + */ +CGLM_INLINE +void +glm_mat2_zero(mat2 mat) { + CGLM_ALIGN_MAT mat2 t = GLM_MAT2_ZERO_INIT; + glm_mat2_copy(t, mat); +} + +/*! + * @brief multiply m1 and m2 to dest + * + * m1, m2 and dest matrices can be same matrix, it is possible to write this: + * + * @code + * mat2 m = GLM_MAT2_IDENTITY_INIT; + * glm_mat2_mul(m, m, m); + * @endcode + * + * @param[in] m1 left matrix + * @param[in] m2 right matrix + * @param[out] dest destination matrix + */ +CGLM_INLINE +void +glm_mat2_mul(mat2 m1, mat2 m2, mat2 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glm_mat2_mul_wasm(m1, m2, dest); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glm_mat2_mul_sse2(m1, m2, dest); +#elif defined(CGLM_NEON_FP) + glm_mat2_mul_neon(m1, m2, dest); +#else + float a00 = m1[0][0], a01 = m1[0][1], + a10 = m1[1][0], a11 = m1[1][1], + b00 = m2[0][0], b01 = m2[0][1], + b10 = m2[1][0], b11 = m2[1][1]; + + dest[0][0] = a00 * b00 + a10 * b01; + dest[0][1] = a01 * b00 + a11 * b01; + dest[1][0] = a00 * b10 + a10 * b11; + dest[1][1] = a01 * b10 + a11 * b11; +#endif +} + +/*! + * @brief transpose mat2 and store in dest + * + * source matrix will not be transposed unless dest is m + * + * @param[in] m matrix + * @param[out] dest result + */ +CGLM_INLINE +void +glm_mat2_transpose_to(mat2 m, mat2 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glm_mat2_transp_wasm(m, dest); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glm_mat2_transp_sse2(m, dest); +#else + dest[0][0] = m[0][0]; + dest[0][1] = m[1][0]; + dest[1][0] = m[0][1]; + dest[1][1] = m[1][1]; +#endif +} + +/*! + * @brief transpose mat2 and store result in same matrix + * + * @param[in, out] m source and dest + */ +CGLM_INLINE +void +glm_mat2_transpose(mat2 m) { + float tmp; + tmp = m[0][1]; + m[0][1] = m[1][0]; + m[1][0] = tmp; +} + +/*! + * @brief multiply mat2 with vec2 (column vector) and store in dest vector + * + * @param[in] m mat2 (left) + * @param[in] v vec2 (right, column vector) + * @param[out] dest vec2 (result, column vector) + */ +CGLM_INLINE +void +glm_mat2_mulv(mat2 m, vec2 v, vec2 dest) { + dest[0] = m[0][0] * v[0] + m[1][0] * v[1]; + dest[1] = m[0][1] * v[0] + m[1][1] * v[1]; +} + +/*! + * @brief trace of matrix + * + * sum of the elements on the main diagonal from upper left to the lower right + * + * @param[in] m matrix + */ +CGLM_INLINE +float +glm_mat2_trace(mat2 m) { + return m[0][0] + m[1][1]; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +void +glm_mat2_scale(mat2 m, float s) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(m[0], wasm_f32x4_mul(wasm_v128_load(m[0]), + wasm_f32x4_splat(s))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(m[0], _mm_mul_ps(_mm_loadu_ps(m[0]), glmm_set1(s))); +#elif defined(CGLM_NEON_FP) + vst1q_f32(m[0], vmulq_f32(vld1q_f32(m[0]), vdupq_n_f32(s))); +#else + m[0][0] = m[0][0] * s; + m[0][1] = m[0][1] * s; + m[1][0] = m[1][0] * s; + m[1][1] = m[1][1] * s; +#endif +} + +/*! + * @brief mat2 determinant + * + * @param[in] mat matrix + * + * @return determinant + */ +CGLM_INLINE +float +glm_mat2_det(mat2 mat) { + return mat[0][0] * mat[1][1] - mat[1][0] * mat[0][1]; +} + +/*! + * @brief inverse mat2 and store in dest + * + * @param[in] mat matrix + * @param[out] dest inverse matrix + */ +CGLM_INLINE +void +glm_mat2_inv(mat2 mat, mat2 dest) { + float det; + float a = mat[0][0], b = mat[0][1], + c = mat[1][0], d = mat[1][1]; + + det = 1.0f / (a * d - b * c); + + dest[0][0] = d * det; + dest[0][1] = -b * det; + dest[1][0] = -c * det; + dest[1][1] = a * det; +} + +/*! + * @brief swap two matrix columns + * + * @param[in,out] mat matrix + * @param[in] col1 col1 + * @param[in] col2 col2 + */ +CGLM_INLINE +void +glm_mat2_swap_col(mat2 mat, int col1, int col2) { + float a, b; + + a = mat[col1][0]; + b = mat[col1][1]; + + mat[col1][0] = mat[col2][0]; + mat[col1][1] = mat[col2][1]; + + mat[col2][0] = a; + mat[col2][1] = b; +} + +/*! + * @brief swap two matrix rows + * + * @param[in,out] mat matrix + * @param[in] row1 row1 + * @param[in] row2 row2 + */ +CGLM_INLINE +void +glm_mat2_swap_row(mat2 mat, int row1, int row2) { + float a, b; + + a = mat[0][row1]; + b = mat[1][row1]; + + mat[0][row1] = mat[0][row2]; + mat[1][row1] = mat[1][row2]; + + mat[0][row2] = a; + mat[1][row2] = b; +} + +/*! + * @brief helper for R (row vector) * M (matrix) * C (column vector) + * + * rmc stands for Row * Matrix * Column + * + * the result is scalar because R * M = Matrix1x2 (row vector), + * then Matrix1x2 * Vec2 (column vector) = Matrix1x1 (Scalar) + * + * @param[in] r row vector or matrix1x2 + * @param[in] m matrix2x2 + * @param[in] c column vector or matrix2x1 + * + * @return scalar value e.g. Matrix1x1 + */ +CGLM_INLINE +float +glm_mat2_rmc(vec2 r, mat2 m, vec2 c) { + vec2 tmp; + glm_mat2_mulv(m, c, tmp); + return glm_vec2_dot(r, tmp); +} + +/*! + * @brief Create mat2 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @param[out] dest matrix + */ +CGLM_INLINE +void +glm_mat2_make(const float * __restrict src, mat2 dest) { + dest[0][0] = src[0]; + dest[0][1] = src[1]; + dest[1][0] = src[2]; + dest[1][1] = src[3]; +} + +#endif /* cglm_mat2_h */ diff --git a/cglm/include/cglm/mat2x3.h b/cglm/include/cglm/mat2x3.h new file mode 100644 index 0000000..811baaf --- /dev/null +++ b/cglm/include/cglm/mat2x3.h @@ -0,0 +1,161 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_MAT2X3_ZERO_INIT + GLM_MAT2X3_ZERO + + Functions: + CGLM_INLINE void glm_mat2x3_copy(mat2x3 mat, mat2x3 dest); + CGLM_INLINE void glm_mat2x3_zero(mat2x3 mat); + CGLM_INLINE void glm_mat2x3_make(const float * __restrict src, mat2x3 dest); + CGLM_INLINE void glm_mat2x3_mul(mat2x3 m1, mat3x2 m2, mat3 dest); + CGLM_INLINE void glm_mat2x3_mulv(mat2x3 m, vec2 v, vec3 dest); + CGLM_INLINE void glm_mat2x3_transpose(mat2x3 m, mat3x2 dest); + CGLM_INLINE void glm_mat2x3_scale(mat2x3 m, float s); + */ + +#ifndef cglm_mat2x3_h +#define cglm_mat2x3_h + +#include "common.h" + +#define GLM_MAT2X3_ZERO_INIT {{0.0f, 0.0f, 0.0f}, {0.0f, 0.0f, 0.0f}} + +/* for C only */ +#define GLM_MAT2X3_ZERO GLM_MAT2X3_ZERO_INIT + +/*! + * @brief copy all members of [mat] to [dest] + * + * @param[in] mat source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_mat2x3_copy(mat2x3 mat, mat2x3 dest) { + dest[0][0] = mat[0][0]; + dest[0][1] = mat[0][1]; + dest[0][2] = mat[0][2]; + + dest[1][0] = mat[1][0]; + dest[1][1] = mat[1][1]; + dest[1][2] = mat[1][2]; +} + +/*! + * @brief make given matrix zero. + * + * @param[in, out] mat matrix + */ +CGLM_INLINE +void +glm_mat2x3_zero(mat2x3 mat) { + CGLM_ALIGN_MAT mat2x3 t = GLM_MAT2X3_ZERO_INIT; + glm_mat2x3_copy(t, mat); +} + +/*! + * @brief Create mat2x3 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @param[out] dest matrix + */ +CGLM_INLINE +void +glm_mat2x3_make(const float * __restrict src, mat2x3 dest) { + dest[0][0] = src[0]; + dest[0][1] = src[1]; + dest[0][2] = src[2]; + + dest[1][0] = src[3]; + dest[1][1] = src[4]; + dest[1][2] = src[5]; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * @code + * glm_mat2x3_mul(mat2x3, mat3x2, mat3); + * @endcode + * + * @param[in] m1 left matrix (mat2x3) + * @param[in] m2 right matrix (mat3x2) + * @param[out] dest destination matrix (mat2) + */ +CGLM_INLINE +void +glm_mat2x3_mul(mat2x3 m1, mat3x2 m2, mat3 dest) { + float a00 = m1[0][0], a01 = m1[0][1], a02 = m1[0][2], + a10 = m1[1][0], a11 = m1[1][1], a12 = m1[1][2], + + b00 = m2[0][0], b01 = m2[0][1], + b10 = m2[1][0], b11 = m2[1][1], + b20 = m2[2][0], b21 = m2[2][1]; + + dest[0][0] = a00 * b00 + a10 * b01; + dest[0][1] = a01 * b00 + a11 * b01; + dest[0][2] = a02 * b00 + a12 * b01; + + dest[1][0] = a00 * b10 + a10 * b11; + dest[1][1] = a01 * b10 + a11 * b11; + dest[1][2] = a02 * b10 + a12 * b11; + + dest[2][0] = a00 * b20 + a10 * b21; + dest[2][1] = a01 * b20 + a11 * b21; + dest[2][2] = a02 * b20 + a12 * b21; +} + +/*! + * @brief multiply matrix with column vector and store in dest vector + * + * @param[in] m matrix (left) + * @param[in] v vector (right, column vector) + * @param[out] dest result vector + */ +CGLM_INLINE +void +glm_mat2x3_mulv(mat2x3 m, vec2 v, vec3 dest) { + float v0 = v[0], v1 = v[1]; + + dest[0] = m[0][0] * v0 + m[1][0] * v1; + dest[1] = m[0][1] * v0 + m[1][1] * v1; + dest[2] = m[0][2] * v0 + m[1][2] * v1; +} + +/*! + * @brief transpose matrix and store in dest + * + * @param[in] m matrix + * @param[out] dest result + */ +CGLM_INLINE +void +glm_mat2x3_transpose(mat2x3 m, mat3x2 dest) { + dest[0][0] = m[0][0]; dest[0][1] = m[1][0]; + dest[1][0] = m[0][1]; dest[1][1] = m[1][1]; + dest[2][0] = m[0][2]; dest[2][1] = m[1][2]; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +void +glm_mat2x3_scale(mat2x3 m, float s) { + m[0][0] *= s; m[0][1] *= s; m[0][2] *= s; + m[1][0] *= s; m[1][1] *= s; m[1][2] *= s; +} + +#endif diff --git a/cglm/include/cglm/mat2x4.h b/cglm/include/cglm/mat2x4.h new file mode 100644 index 0000000..e5c1281 --- /dev/null +++ b/cglm/include/cglm/mat2x4.h @@ -0,0 +1,170 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_MAT2X4_ZERO_INIT + GLM_MAT2X4_ZERO + + Functions: + CGLM_INLINE void glm_mat2x4_copy(mat2x4 mat, mat2x4 dest); + CGLM_INLINE void glm_mat2x4_zero(mat2x4 mat); + CGLM_INLINE void glm_mat2x4_make(const float * __restrict src, mat2x4 dest); + CGLM_INLINE void glm_mat2x4_mul(mat2x4 m1, mat4x2 m2, mat4 dest); + CGLM_INLINE void glm_mat2x4_mulv(mat2x4 m, vec2 v, vec4 dest); + CGLM_INLINE void glm_mat2x4_transpose(mat2x4 m, mat4x2 dest); + CGLM_INLINE void glm_mat2x4_scale(mat2x4 m, float s); + */ + +#ifndef cglm_mat2x4_h +#define cglm_mat2x4_h + +#include "common.h" +#include "vec4.h" + +#define GLM_MAT2X4_ZERO_INIT {{0.0f, 0.0f, 0.0f, 0.0f}, {0.0f, 0.0f, 0.0f, 0.0f}} + +/* for C only */ +#define GLM_MAT2X4_ZERO GLM_MAT2X4_ZERO_INIT + +/*! + * @brief copy all members of [mat] to [dest] + * + * @param[in] mat source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_mat2x4_copy(mat2x4 mat, mat2x4 dest) { + glm_vec4_ucopy(mat[0], dest[0]); + glm_vec4_ucopy(mat[1], dest[1]); +} + +/*! + * @brief make given matrix zero. + * + * @param[in, out] mat matrix + */ +CGLM_INLINE +void +glm_mat2x4_zero(mat2x4 mat) { + CGLM_ALIGN_MAT mat2x4 t = GLM_MAT2X4_ZERO_INIT; + glm_mat2x4_copy(t, mat); +} + +/*! + * @brief Create mat2x4 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @param[out] dest matrix + */ +CGLM_INLINE +void +glm_mat2x4_make(const float * __restrict src, mat2x4 dest) { + dest[0][0] = src[0]; + dest[0][1] = src[1]; + dest[0][2] = src[2]; + dest[0][3] = src[3]; + + dest[1][0] = src[4]; + dest[1][1] = src[5]; + dest[1][2] = src[6]; + dest[1][3] = src[7]; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * @code + * glm_mat2x4_mul(mat2x4, mat4x2, mat4); + * @endcode + * + * @param[in] m1 left matrix (mat2x4) + * @param[in] m2 right matrix (mat4x2) + * @param[out] dest destination matrix (mat4) + */ +CGLM_INLINE +void +glm_mat2x4_mul(mat2x4 m1, mat4x2 m2, mat4 dest) { + float a00 = m1[0][0], a01 = m1[0][1], a02 = m1[0][2], a03 = m1[0][3], + a10 = m1[1][0], a11 = m1[1][1], a12 = m1[1][2], a13 = m1[1][3], + + b00 = m2[0][0], b01 = m2[0][1], + b10 = m2[1][0], b11 = m2[1][1], + b20 = m2[2][0], b21 = m2[2][1], + b30 = m2[3][0], b31 = m2[3][1]; + + dest[0][0] = a00 * b00 + a10 * b01; + dest[0][1] = a01 * b00 + a11 * b01; + dest[0][2] = a02 * b00 + a12 * b01; + dest[0][3] = a03 * b00 + a13 * b01; + + dest[1][0] = a00 * b10 + a10 * b11; + dest[1][1] = a01 * b10 + a11 * b11; + dest[1][2] = a02 * b10 + a12 * b11; + dest[1][3] = a03 * b10 + a13 * b11; + + dest[2][0] = a00 * b20 + a10 * b21; + dest[2][1] = a01 * b20 + a11 * b21; + dest[2][2] = a02 * b20 + a12 * b21; + dest[2][3] = a03 * b20 + a13 * b21; + + dest[3][0] = a00 * b30 + a10 * b31; + dest[3][1] = a01 * b30 + a11 * b31; + dest[3][2] = a02 * b30 + a12 * b31; + dest[3][3] = a03 * b30 + a13 * b31; +} + +/*! + * @brief multiply matrix with column vector and store in dest column vector + * + * @param[in] m matrix (left) + * @param[in] v vector (right, column vector) + * @param[out] dest result vector + */ +CGLM_INLINE +void +glm_mat2x4_mulv(mat2x4 m, vec2 v, vec4 dest) { + float v0 = v[0], v1 = v[1]; + + dest[0] = m[0][0] * v0 + m[1][0] * v1; + dest[1] = m[0][1] * v0 + m[1][1] * v1; + dest[2] = m[0][2] * v0 + m[1][2] * v1; + dest[3] = m[0][3] * v0 + m[1][3] * v1; +} + +/*! + * @brief transpose matrix and store in dest + * + * @param[in] m matrix + * @param[out] dest result + */ +CGLM_INLINE +void +glm_mat2x4_transpose(mat2x4 m, mat4x2 dest) { + dest[0][0] = m[0][0]; dest[0][1] = m[1][0]; + dest[1][0] = m[0][1]; dest[1][1] = m[1][1]; + dest[2][0] = m[0][2]; dest[2][1] = m[1][2]; + dest[3][0] = m[0][3]; dest[3][1] = m[1][3]; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +void +glm_mat2x4_scale(mat2x4 m, float s) { + m[0][0] *= s; m[0][1] *= s; m[0][2] *= s; m[0][3] *= s; + m[1][0] *= s; m[1][1] *= s; m[1][2] *= s; m[1][3] *= s; +} + +#endif diff --git a/cglm/include/cglm/mat3.h b/cglm/include/cglm/mat3.h new file mode 100644 index 0000000..9d5f0d4 --- /dev/null +++ b/cglm/include/cglm/mat3.h @@ -0,0 +1,451 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_MAT3_IDENTITY_INIT + GLM_MAT3_ZERO_INIT + GLM_MAT3_IDENTITY + GLM_MAT3_ZERO + glm_mat3_dup(mat, dest) + + Functions: + CGLM_INLINE void glm_mat3_copy(mat3 mat, mat3 dest); + CGLM_INLINE void glm_mat3_identity(mat3 mat); + CGLM_INLINE void glm_mat3_identity_array(mat3 * restrict mat, size_t count); + CGLM_INLINE void glm_mat3_zero(mat3 mat); + CGLM_INLINE void glm_mat3_mul(mat3 m1, mat3 m2, mat3 dest); + CGLM_INLINE void glm_mat3_transpose_to(mat3 m, mat3 dest); + CGLM_INLINE void glm_mat3_transpose(mat3 m); + CGLM_INLINE void glm_mat3_mulv(mat3 m, vec3 v, vec3 dest); + CGLM_INLINE float glm_mat3_trace(mat3 m); + CGLM_INLINE void glm_mat3_quat(mat3 m, versor dest); + CGLM_INLINE void glm_mat3_scale(mat3 m, float s); + CGLM_INLINE float glm_mat3_det(mat3 mat); + CGLM_INLINE void glm_mat3_inv(mat3 mat, mat3 dest); + CGLM_INLINE void glm_mat3_swap_col(mat3 mat, int col1, int col2); + CGLM_INLINE void glm_mat3_swap_row(mat3 mat, int row1, int row2); + CGLM_INLINE float glm_mat3_rmc(vec3 r, mat3 m, vec3 c); + CGLM_INLINE void glm_mat3_make(float * restrict src, mat3 dest); + */ + +#ifndef cglm_mat3_h +#define cglm_mat3_h + +#include "common.h" +#include "vec3.h" + +#ifdef CGLM_SSE_FP +# include "simd/sse2/mat3.h" +#endif + +#ifdef CGLM_SIMD_WASM +# include "simd/wasm/mat3.h" +#endif + +#define GLM_MAT3_IDENTITY_INIT {{1.0f, 0.0f, 0.0f}, \ + {0.0f, 1.0f, 0.0f}, \ + {0.0f, 0.0f, 1.0f}} +#define GLM_MAT3_ZERO_INIT {{0.0f, 0.0f, 0.0f}, \ + {0.0f, 0.0f, 0.0f}, \ + {0.0f, 0.0f, 0.0f}} + + +/* for C only */ +#define GLM_MAT3_IDENTITY ((mat3)GLM_MAT3_IDENTITY_INIT) +#define GLM_MAT3_ZERO ((mat3)GLM_MAT3_ZERO_INIT) + +/* DEPRECATED! use _copy, _ucopy versions */ +#define glm_mat3_dup(mat, dest) glm_mat3_copy(mat, dest) + +/*! + * @brief copy all members of [mat] to [dest] + * + * @param[in] mat source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_mat3_copy(mat3 mat, mat3 dest) { + dest[0][0] = mat[0][0]; + dest[0][1] = mat[0][1]; + dest[0][2] = mat[0][2]; + + dest[1][0] = mat[1][0]; + dest[1][1] = mat[1][1]; + dest[1][2] = mat[1][2]; + + dest[2][0] = mat[2][0]; + dest[2][1] = mat[2][1]; + dest[2][2] = mat[2][2]; +} + +/*! + * @brief make given matrix identity. It is identical with below, + * but it is more easy to do that with this func especially for members + * e.g. glm_mat3_identity(aStruct->aMatrix); + * + * @code + * glm_mat3_copy(GLM_MAT3_IDENTITY, mat); // C only + * + * // or + * mat3 mat = GLM_MAT3_IDENTITY_INIT; + * @endcode + * + * @param[in, out] mat destination + */ +CGLM_INLINE +void +glm_mat3_identity(mat3 mat) { + CGLM_ALIGN_MAT mat3 t = GLM_MAT3_IDENTITY_INIT; + glm_mat3_copy(t, mat); +} + +/*! + * @brief make given matrix array's each element identity matrix + * + * @param[in, out] mat matrix array (must be aligned (16/32) + * if alignment is not disabled) + * + * @param[in] count count of matrices + */ +CGLM_INLINE +void +glm_mat3_identity_array(mat3 * __restrict mat, size_t count) { + CGLM_ALIGN_MAT mat3 t = GLM_MAT3_IDENTITY_INIT; + size_t i; + + for (i = 0; i < count; i++) { + glm_mat3_copy(t, mat[i]); + } +} + +/*! + * @brief make given matrix zero. + * + * @param[in, out] mat matrix + */ +CGLM_INLINE +void +glm_mat3_zero(mat3 mat) { + CGLM_ALIGN_MAT mat3 t = GLM_MAT3_ZERO_INIT; + glm_mat3_copy(t, mat); +} + +/*! + * @brief multiply m1 and m2 to dest + * + * m1, m2 and dest matrices can be same matrix, it is possible to write this: + * + * @code + * mat3 m = GLM_MAT3_IDENTITY_INIT; + * glm_mat3_mul(m, m, m); + * @endcode + * + * @param[in] m1 left matrix + * @param[in] m2 right matrix + * @param[out] dest destination matrix + */ +CGLM_INLINE +void +glm_mat3_mul(mat3 m1, mat3 m2, mat3 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glm_mat3_mul_wasm(m1, m2, dest); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glm_mat3_mul_sse2(m1, m2, dest); +#else + float a00 = m1[0][0], a01 = m1[0][1], a02 = m1[0][2], + a10 = m1[1][0], a11 = m1[1][1], a12 = m1[1][2], + a20 = m1[2][0], a21 = m1[2][1], a22 = m1[2][2], + + b00 = m2[0][0], b01 = m2[0][1], b02 = m2[0][2], + b10 = m2[1][0], b11 = m2[1][1], b12 = m2[1][2], + b20 = m2[2][0], b21 = m2[2][1], b22 = m2[2][2]; + + dest[0][0] = a00 * b00 + a10 * b01 + a20 * b02; + dest[0][1] = a01 * b00 + a11 * b01 + a21 * b02; + dest[0][2] = a02 * b00 + a12 * b01 + a22 * b02; + dest[1][0] = a00 * b10 + a10 * b11 + a20 * b12; + dest[1][1] = a01 * b10 + a11 * b11 + a21 * b12; + dest[1][2] = a02 * b10 + a12 * b11 + a22 * b12; + dest[2][0] = a00 * b20 + a10 * b21 + a20 * b22; + dest[2][1] = a01 * b20 + a11 * b21 + a21 * b22; + dest[2][2] = a02 * b20 + a12 * b21 + a22 * b22; +#endif +} + +/*! + * @brief transpose mat3 and store in dest + * + * source matrix will not be transposed unless dest is m + * + * @param[in] m matrix + * @param[out] dest result + */ +CGLM_INLINE +void +glm_mat3_transpose_to(mat3 m, mat3 dest) { + dest[0][0] = m[0][0]; + dest[0][1] = m[1][0]; + dest[0][2] = m[2][0]; + dest[1][0] = m[0][1]; + dest[1][1] = m[1][1]; + dest[1][2] = m[2][1]; + dest[2][0] = m[0][2]; + dest[2][1] = m[1][2]; + dest[2][2] = m[2][2]; +} + +/*! + * @brief transpose mat3 and store result in same matrix + * + * @param[in, out] m source and dest + */ +CGLM_INLINE +void +glm_mat3_transpose(mat3 m) { + CGLM_ALIGN_MAT mat3 tmp; + + tmp[0][1] = m[1][0]; + tmp[0][2] = m[2][0]; + tmp[1][0] = m[0][1]; + tmp[1][2] = m[2][1]; + tmp[2][0] = m[0][2]; + tmp[2][1] = m[1][2]; + + m[0][1] = tmp[0][1]; + m[0][2] = tmp[0][2]; + m[1][0] = tmp[1][0]; + m[1][2] = tmp[1][2]; + m[2][0] = tmp[2][0]; + m[2][1] = tmp[2][1]; +} + +/*! + * @brief multiply mat3 with vec3 (column vector) and store in dest vector + * + * @param[in] m mat3 (left) + * @param[in] v vec3 (right, column vector) + * @param[out] dest vec3 (result, column vector) + */ +CGLM_INLINE +void +glm_mat3_mulv(mat3 m, vec3 v, vec3 dest) { + vec3 res; + res[0] = m[0][0] * v[0] + m[1][0] * v[1] + m[2][0] * v[2]; + res[1] = m[0][1] * v[0] + m[1][1] * v[1] + m[2][1] * v[2]; + res[2] = m[0][2] * v[0] + m[1][2] * v[1] + m[2][2] * v[2]; + glm_vec3_copy(res, dest); +} + +/*! + * @brief trace of matrix + * + * sum of the elements on the main diagonal from upper left to the lower right + * + * @param[in] m matrix + */ +CGLM_INLINE +float +glm_mat3_trace(mat3 m) { + return m[0][0] + m[1][1] + m[2][2]; +} + +/*! + * @brief convert mat3 to quaternion + * + * @param[in] m rotation matrix + * @param[out] dest destination quaternion + */ +CGLM_INLINE +void +glm_mat3_quat(mat3 m, versor dest) { + float trace, r, rinv; + + /* it seems using like m12 instead of m[1][2] causes extra instructions */ + + trace = m[0][0] + m[1][1] + m[2][2]; + if (trace >= 0.0f) { + r = sqrtf(1.0f + trace); + rinv = 0.5f / r; + + dest[0] = rinv * (m[1][2] - m[2][1]); + dest[1] = rinv * (m[2][0] - m[0][2]); + dest[2] = rinv * (m[0][1] - m[1][0]); + dest[3] = r * 0.5f; + } else if (m[0][0] >= m[1][1] && m[0][0] >= m[2][2]) { + r = sqrtf(1.0f - m[1][1] - m[2][2] + m[0][0]); + rinv = 0.5f / r; + + dest[0] = r * 0.5f; + dest[1] = rinv * (m[0][1] + m[1][0]); + dest[2] = rinv * (m[0][2] + m[2][0]); + dest[3] = rinv * (m[1][2] - m[2][1]); + } else if (m[1][1] >= m[2][2]) { + r = sqrtf(1.0f - m[0][0] - m[2][2] + m[1][1]); + rinv = 0.5f / r; + + dest[0] = rinv * (m[0][1] + m[1][0]); + dest[1] = r * 0.5f; + dest[2] = rinv * (m[1][2] + m[2][1]); + dest[3] = rinv * (m[2][0] - m[0][2]); + } else { + r = sqrtf(1.0f - m[0][0] - m[1][1] + m[2][2]); + rinv = 0.5f / r; + + dest[0] = rinv * (m[0][2] + m[2][0]); + dest[1] = rinv * (m[1][2] + m[2][1]); + dest[2] = r * 0.5f; + dest[3] = rinv * (m[0][1] - m[1][0]); + } +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +void +glm_mat3_scale(mat3 m, float s) { + m[0][0] *= s; m[0][1] *= s; m[0][2] *= s; + m[1][0] *= s; m[1][1] *= s; m[1][2] *= s; + m[2][0] *= s; m[2][1] *= s; m[2][2] *= s; +} + +/*! + * @brief mat3 determinant + * + * @param[in] mat matrix + * + * @return determinant + */ +CGLM_INLINE +float +glm_mat3_det(mat3 mat) { + float a = mat[0][0], b = mat[0][1], c = mat[0][2], + d = mat[1][0], e = mat[1][1], f = mat[1][2], + g = mat[2][0], h = mat[2][1], i = mat[2][2]; + + return a * (e * i - h * f) - d * (b * i - h * c) + g * (b * f - e * c); +} + +/*! + * @brief inverse mat3 and store in dest + * + * @param[in] mat matrix + * @param[out] dest inverse matrix + */ +CGLM_INLINE +void +glm_mat3_inv(mat3 mat, mat3 dest) { + float a = mat[0][0], b = mat[0][1], c = mat[0][2], + d = mat[1][0], e = mat[1][1], f = mat[1][2], + g = mat[2][0], h = mat[2][1], i = mat[2][2], + + c1 = e * i - f * h, c2 = d * i - g * f, c3 = d * h - g * e, + idt = 1.0f / (a * c1 - b * c2 + c * c3), ndt = -idt; + + dest[0][0] = idt * c1; + dest[0][1] = ndt * (b * i - h * c); + dest[0][2] = idt * (b * f - e * c); + dest[1][0] = ndt * c2; + dest[1][1] = idt * (a * i - g * c); + dest[1][2] = ndt * (a * f - d * c); + dest[2][0] = idt * c3; + dest[2][1] = ndt * (a * h - g * b); + dest[2][2] = idt * (a * e - d * b); +} + +/*! + * @brief swap two matrix columns + * + * @param[in,out] mat matrix + * @param[in] col1 col1 + * @param[in] col2 col2 + */ +CGLM_INLINE +void +glm_mat3_swap_col(mat3 mat, int col1, int col2) { + vec3 tmp; + glm_vec3_copy(mat[col1], tmp); + glm_vec3_copy(mat[col2], mat[col1]); + glm_vec3_copy(tmp, mat[col2]); +} + +/*! + * @brief swap two matrix rows + * + * @param[in,out] mat matrix + * @param[in] row1 row1 + * @param[in] row2 row2 + */ +CGLM_INLINE +void +glm_mat3_swap_row(mat3 mat, int row1, int row2) { + vec3 tmp; + tmp[0] = mat[0][row1]; + tmp[1] = mat[1][row1]; + tmp[2] = mat[2][row1]; + + mat[0][row1] = mat[0][row2]; + mat[1][row1] = mat[1][row2]; + mat[2][row1] = mat[2][row2]; + + mat[0][row2] = tmp[0]; + mat[1][row2] = tmp[1]; + mat[2][row2] = tmp[2]; +} + +/*! + * @brief helper for R (row vector) * M (matrix) * C (column vector) + * + * rmc stands for Row * Matrix * Column + * + * the result is scalar because R * M = Matrix1x3 (row vector), + * then Matrix1x3 * Vec3 (column vector) = Matrix1x1 (Scalar) + * + * @param[in] r row vector or matrix1x3 + * @param[in] m matrix3x3 + * @param[in] c column vector or matrix3x1 + * + * @return scalar value e.g. Matrix1x1 + */ +CGLM_INLINE +float +glm_mat3_rmc(vec3 r, mat3 m, vec3 c) { + vec3 tmp; + glm_mat3_mulv(m, c, tmp); + return glm_vec3_dot(r, tmp); +} + +/*! + * @brief Create mat3 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @param[out] dest matrix + */ +CGLM_INLINE +void +glm_mat3_make(const float * __restrict src, mat3 dest) { + dest[0][0] = src[0]; + dest[0][1] = src[1]; + dest[0][2] = src[2]; + + dest[1][0] = src[3]; + dest[1][1] = src[4]; + dest[1][2] = src[5]; + + dest[2][0] = src[6]; + dest[2][1] = src[7]; + dest[2][2] = src[8]; +} + +#endif /* cglm_mat3_h */ diff --git a/cglm/include/cglm/mat3x2.h b/cglm/include/cglm/mat3x2.h new file mode 100644 index 0000000..75ed9c2 --- /dev/null +++ b/cglm/include/cglm/mat3x2.h @@ -0,0 +1,155 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_MAT3X2_ZERO_INIT + GLM_MAT3X2_ZERO + + Functions: + CGLM_INLINE void glm_mat3x2_copy(mat3x2 mat, mat3x2 dest); + CGLM_INLINE void glm_mat3x2_zero(mat3x2 mat); + CGLM_INLINE void glm_mat3x2_make(const float * __restrict src, mat3x2 dest); + CGLM_INLINE void glm_mat3x2_mul(mat3x2 m1, mat2x3 m2, mat2 dest); + CGLM_INLINE void glm_mat3x2_mulv(mat3x2 m, vec3 v, vec2 dest); + CGLM_INLINE void glm_mat3x2_transpose(mat3x2 m, mat2x3 dest); + CGLM_INLINE void glm_mat3x2_scale(mat3x2 m, float s); + */ + +#ifndef cglm_mat3x2_h +#define cglm_mat3x2_h + +#include "common.h" + +#define GLM_MAT3X2_ZERO_INIT {{0.0f, 0.0f}, {0.0f, 0.0f}, {0.0f, 0.0f}} + +/* for C only */ +#define GLM_MAT3X2_ZERO GLM_MAT3X2_ZERO_INIT + +/*! + * @brief copy all members of [mat] to [dest] + * + * @param[in] mat source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_mat3x2_copy(mat3x2 mat, mat3x2 dest) { + dest[0][0] = mat[0][0]; + dest[0][1] = mat[0][1]; + + dest[1][0] = mat[1][0]; + dest[1][1] = mat[1][1]; + + dest[2][0] = mat[2][0]; + dest[2][1] = mat[2][1]; +} + +/*! + * @brief make given matrix zero. + * + * @param[in, out] mat matrix + */ +CGLM_INLINE +void +glm_mat3x2_zero(mat3x2 mat) { + CGLM_ALIGN_MAT mat3x2 t = GLM_MAT3X2_ZERO_INIT; + glm_mat3x2_copy(t, mat); +} + +/*! + * @brief Create mat3x2 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @param[out] dest matrix + */ +CGLM_INLINE +void +glm_mat3x2_make(const float * __restrict src, mat3x2 dest) { + dest[0][0] = src[0]; + dest[0][1] = src[1]; + + dest[1][0] = src[2]; + dest[1][1] = src[3]; + + dest[2][0] = src[4]; + dest[2][1] = src[5]; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * @code + * glm_mat3x2_mul(mat3x2, mat2x3, mat2); + * @endcode + * + * @param[in] m1 left matrix (mat3x2) + * @param[in] m2 right matrix (mat2x3) + * @param[out] dest destination matrix (mat2) + */ +CGLM_INLINE +void +glm_mat3x2_mul(mat3x2 m1, mat2x3 m2, mat2 dest) { + float a00 = m1[0][0], a01 = m1[0][1], + a10 = m1[1][0], a11 = m1[1][1], + a20 = m1[2][0], a21 = m1[2][1], + + b00 = m2[0][0], b01 = m2[0][1], b02 = m2[0][2], + b10 = m2[1][0], b11 = m2[1][1], b12 = m2[1][2]; + + dest[0][0] = a00 * b00 + a10 * b01 + a20 * b02; + dest[0][1] = a01 * b00 + a11 * b01 + a21 * b02; + + dest[1][0] = a00 * b10 + a10 * b11 + a20 * b12; + dest[1][1] = a01 * b10 + a11 * b11 + a21 * b12; +} + +/*! + * @brief multiply matrix with column vector and store in dest column vector + * + * @param[in] m matrix (left) + * @param[in] v vector (right, column vector) + * @param[out] dest result vector + */ +CGLM_INLINE +void +glm_mat3x2_mulv(mat3x2 m, vec3 v, vec2 dest) { + float v0 = v[0], v1 = v[1], v2 = v[2]; + + dest[0] = m[0][0] * v0 + m[1][0] * v1 + m[2][0] * v2; + dest[1] = m[0][1] * v0 + m[1][1] * v1 + m[2][1] * v2; +} + +/*! + * @brief transpose matrix and store in dest + * + * @param[in] m matrix + * @param[out] dest result + */ +CGLM_INLINE +void +glm_mat3x2_transpose(mat3x2 m, mat2x3 dest) { + dest[0][0] = m[0][0]; dest[0][1] = m[1][0]; dest[0][2] = m[2][0]; + dest[1][0] = m[0][1]; dest[1][1] = m[1][1]; dest[1][2] = m[2][1]; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +void +glm_mat3x2_scale(mat3x2 m, float s) { + m[0][0] *= s; m[0][1] *= s; m[1][0] *= s; + m[1][1] *= s; m[2][0] *= s; m[2][1] *= s; +} + +#endif diff --git a/cglm/include/cglm/mat3x4.h b/cglm/include/cglm/mat3x4.h new file mode 100644 index 0000000..8d14523 --- /dev/null +++ b/cglm/include/cglm/mat3x4.h @@ -0,0 +1,179 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_MAT3X4_ZERO_INIT + GLM_MAT3X4_ZERO + + Functions: + CGLM_INLINE void glm_mat3x4_copy(mat3x4 mat, mat3x4 dest); + CGLM_INLINE void glm_mat3x4_zero(mat3x4 mat); + CGLM_INLINE void glm_mat3x4_make(const float * __restrict src, mat3x4 dest); + CGLM_INLINE void glm_mat3x4_mul(mat3x4 m1, mat4x3 m2, mat4 dest); + CGLM_INLINE void glm_mat3x4_mulv(mat3x4 m, vec3 v, vec4 dest); + CGLM_INLINE void glm_mat3x4_transpose(mat3x4 m, mat4x3 dest); + CGLM_INLINE void glm_mat3x4_scale(mat3x4 m, float s); + */ + +#ifndef cglm_mat3x4_h +#define cglm_mat3x4_h + +#include "common.h" + +#define GLM_MAT3X4_ZERO_INIT {{0.0f, 0.0f, 0.0f, 0.0f}, \ + {0.0f, 0.0f, 0.0f, 0.0f}, \ + {0.0f, 0.0f, 0.0f, 0.0f}} + +/* for C only */ +#define GLM_MAT3X4_ZERO GLM_MAT3X4_ZERO_INIT + +/*! + * @brief copy all members of [mat] to [dest] + * + * @param[in] mat source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_mat3x4_copy(mat3x4 mat, mat3x4 dest) { + glm_vec4_ucopy(mat[0], dest[0]); + glm_vec4_ucopy(mat[1], dest[1]); + glm_vec4_ucopy(mat[2], dest[2]); +} + +/*! + * @brief make given matrix zero. + * + * @param[in, out] mat matrix + */ +CGLM_INLINE +void +glm_mat3x4_zero(mat3x4 mat) { + CGLM_ALIGN_MAT mat3x4 t = GLM_MAT3X4_ZERO_INIT; + glm_mat3x4_copy(t, mat); +} + +/*! + * @brief Create mat3x4 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @param[out] dest matrix + */ +CGLM_INLINE +void +glm_mat3x4_make(const float * __restrict src, mat3x4 dest) { + dest[0][0] = src[0]; + dest[0][1] = src[1]; + dest[0][2] = src[2]; + dest[0][3] = src[3]; + + dest[1][0] = src[4]; + dest[1][1] = src[5]; + dest[1][2] = src[6]; + dest[1][3] = src[7]; + + dest[2][0] = src[8]; + dest[2][1] = src[9]; + dest[2][2] = src[10]; + dest[2][3] = src[11]; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * @code + * glm_mat3x4_mul(mat3x4, mat4x3, mat4); + * @endcode + * + * @param[in] m1 left matrix (mat3x4) + * @param[in] m2 right matrix (mat4x3) + * @param[out] dest destination matrix (mat4) + */ +CGLM_INLINE +void +glm_mat3x4_mul(mat3x4 m1, mat4x3 m2, mat4 dest) { + float a00 = m1[0][0], a01 = m1[0][1], a02 = m1[0][2], a03 = m1[0][3], + a10 = m1[1][0], a11 = m1[1][1], a12 = m1[1][2], a13 = m1[1][3], + a20 = m1[2][0], a21 = m1[2][1], a22 = m1[2][2], a23 = m1[2][3], + + b00 = m2[0][0], b01 = m2[0][1], b02 = m2[0][2], + b10 = m2[1][0], b11 = m2[1][1], b12 = m2[1][2], + b20 = m2[2][0], b21 = m2[2][1], b22 = m2[2][2], + b30 = m2[3][0], b31 = m2[3][1], b32 = m2[3][2]; + + dest[0][0] = a00 * b00 + a10 * b01 + a20 * b02; + dest[0][1] = a01 * b00 + a11 * b01 + a21 * b02; + dest[0][2] = a02 * b00 + a12 * b01 + a22 * b02; + dest[0][3] = a03 * b00 + a13 * b01 + a23 * b02; + + dest[1][0] = a00 * b10 + a10 * b11 + a20 * b12; + dest[1][1] = a01 * b10 + a11 * b11 + a21 * b12; + dest[1][2] = a02 * b10 + a12 * b11 + a22 * b12; + dest[1][3] = a03 * b10 + a13 * b11 + a23 * b12; + + dest[2][0] = a00 * b20 + a10 * b21 + a20 * b22; + dest[2][1] = a01 * b20 + a11 * b21 + a21 * b22; + dest[2][2] = a02 * b20 + a12 * b21 + a22 * b22; + dest[2][3] = a03 * b20 + a13 * b21 + a23 * b22; + + dest[3][0] = a00 * b30 + a10 * b31 + a20 * b32; + dest[3][1] = a01 * b30 + a11 * b31 + a21 * b32; + dest[3][2] = a02 * b30 + a12 * b31 + a22 * b32; + dest[3][3] = a03 * b30 + a13 * b31 + a23 * b32; +} + +/*! + * @brief multiply matrix with column vector and store in dest column vector + * + * @param[in] m matrix (left) + * @param[in] v vector (right, column vector) + * @param[out] dest result vector + */ +CGLM_INLINE +void +glm_mat3x4_mulv(mat3x4 m, vec3 v, vec4 dest) { + float v0 = v[0], v1 = v[1], v2 = v[2]; + + dest[0] = m[0][0] * v0 + m[1][0] * v1 + m[2][0] * v2; + dest[1] = m[0][1] * v0 + m[1][1] * v1 + m[2][1] * v2; + dest[2] = m[0][2] * v0 + m[1][2] * v1 + m[2][2] * v2; + dest[3] = m[0][3] * v0 + m[1][3] * v1 + m[2][3] * v2; +} + +/*! + * @brief transpose matrix and store in dest + * + * @param[in] m matrix + * @param[out] dest result + */ +CGLM_INLINE +void +glm_mat3x4_transpose(mat3x4 m, mat4x3 dest) { + dest[0][0] = m[0][0]; dest[0][1] = m[1][0]; dest[0][2] = m[2][0]; + dest[1][0] = m[0][1]; dest[1][1] = m[1][1]; dest[1][2] = m[2][1]; + dest[2][0] = m[0][2]; dest[2][1] = m[1][2]; dest[2][2] = m[2][2]; + dest[3][0] = m[0][3]; dest[3][1] = m[1][3]; dest[3][2] = m[2][3]; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +void +glm_mat3x4_scale(mat3x4 m, float s) { + m[0][0] *= s; m[0][1] *= s; m[0][2] *= s; m[0][3] *= s; + m[1][0] *= s; m[1][1] *= s; m[1][2] *= s; m[1][3] *= s; + m[2][0] *= s; m[2][1] *= s; m[2][2] *= s; m[2][3] *= s; +} + +#endif diff --git a/cglm/include/cglm/mat4.h b/cglm/include/cglm/mat4.h new file mode 100644 index 0000000..1c36c26 --- /dev/null +++ b/cglm/include/cglm/mat4.h @@ -0,0 +1,802 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/*! + * Most of functions in this header are optimized manually with SIMD + * if available. You dont need to call/incude SIMD headers manually + */ + +/* + Macros: + GLM_MAT4_IDENTITY_INIT + GLM_MAT4_ZERO_INIT + GLM_MAT4_IDENTITY + GLM_MAT4_ZERO + + Functions: + CGLM_INLINE void glm_mat4_ucopy(mat4 mat, mat4 dest); + CGLM_INLINE void glm_mat4_copy(mat4 mat, mat4 dest); + CGLM_INLINE void glm_mat4_identity(mat4 mat); + CGLM_INLINE void glm_mat4_identity_array(mat4 * restrict mat, size_t count); + CGLM_INLINE void glm_mat4_zero(mat4 mat); + CGLM_INLINE void glm_mat4_pick3(mat4 mat, mat3 dest); + CGLM_INLINE void glm_mat4_pick3t(mat4 mat, mat3 dest); + CGLM_INLINE void glm_mat4_ins3(mat3 mat, mat4 dest); + CGLM_INLINE void glm_mat4_mul(mat4 m1, mat4 m2, mat4 dest); + CGLM_INLINE void glm_mat4_mulN(mat4 *matrices[], int len, mat4 dest); + CGLM_INLINE void glm_mat4_mulv(mat4 m, vec4 v, vec4 dest); + CGLM_INLINE void glm_mat4_mulv3(mat4 m, vec3 v, float last, vec3 dest); + CGLM_INLINE float glm_mat4_trace(mat4 m); + CGLM_INLINE float glm_mat4_trace3(mat4 m); + CGLM_INLINE void glm_mat4_quat(mat4 m, versor dest) ; + CGLM_INLINE void glm_mat4_transpose_to(mat4 m, mat4 dest); + CGLM_INLINE void glm_mat4_transpose(mat4 m); + CGLM_INLINE void glm_mat4_scale_p(mat4 m, float s); + CGLM_INLINE void glm_mat4_scale(mat4 m, float s); + CGLM_INLINE float glm_mat4_det(mat4 mat); + CGLM_INLINE void glm_mat4_inv(mat4 mat, mat4 dest); + CGLM_INLINE void glm_mat4_inv_fast(mat4 mat, mat4 dest); + CGLM_INLINE void glm_mat4_swap_col(mat4 mat, int col1, int col2); + CGLM_INLINE void glm_mat4_swap_row(mat4 mat, int row1, int row2); + CGLM_INLINE float glm_mat4_rmc(vec4 r, mat4 m, vec4 c); + CGLM_INLINE void glm_mat4_make(float * restrict src, mat4 dest); + */ + +#ifndef cglm_mat_h +#define cglm_mat_h + +#include "common.h" +#include "vec4.h" +#include "vec3.h" + +#ifdef CGLM_SSE_FP +# include "simd/sse2/mat4.h" +#endif + +#ifdef CGLM_AVX_FP +# include "simd/avx/mat4.h" +#endif + +#ifdef CGLM_NEON_FP +# include "simd/neon/mat4.h" +#endif + +#ifdef CGLM_SIMD_WASM +# include "simd/wasm/mat4.h" +#endif + +#ifndef NDEBUG +# include +#endif + +#define GLM_MAT4_IDENTITY_INIT {{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}} + +#define GLM_MAT4_ZERO_INIT {{0.0f, 0.0f, 0.0f, 0.0f}, \ + {0.0f, 0.0f, 0.0f, 0.0f}, \ + {0.0f, 0.0f, 0.0f, 0.0f}, \ + {0.0f, 0.0f, 0.0f, 0.0f}} + +/* for C only */ +#define GLM_MAT4_IDENTITY ((mat4)GLM_MAT4_IDENTITY_INIT) +#define GLM_MAT4_ZERO ((mat4)GLM_MAT4_ZERO_INIT) + +/* DEPRECATED! use _copy, _ucopy versions */ +#define glm_mat4_udup(mat, dest) glm_mat4_ucopy(mat, dest) +#define glm_mat4_dup(mat, dest) glm_mat4_copy(mat, dest) + +/* DEPRECATED! default is precise now. */ +#define glm_mat4_inv_precise(mat, dest) glm_mat4_inv(mat, dest) + +/*! + * @brief copy all members of [mat] to [dest] + * + * matrix may not be aligned, u stands for unaligned, this may be useful when + * copying a matrix from external source e.g. asset importer... + * + * @param[in] mat source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_mat4_ucopy(mat4 mat, mat4 dest) { + dest[0][0] = mat[0][0]; dest[1][0] = mat[1][0]; + dest[0][1] = mat[0][1]; dest[1][1] = mat[1][1]; + dest[0][2] = mat[0][2]; dest[1][2] = mat[1][2]; + dest[0][3] = mat[0][3]; dest[1][3] = mat[1][3]; + + dest[2][0] = mat[2][0]; dest[3][0] = mat[3][0]; + dest[2][1] = mat[2][1]; dest[3][1] = mat[3][1]; + dest[2][2] = mat[2][2]; dest[3][2] = mat[3][2]; + dest[2][3] = mat[2][3]; dest[3][3] = mat[3][3]; +} + +/*! + * @brief copy all members of [mat] to [dest] + * + * @param[in] mat source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_mat4_copy(mat4 mat, mat4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest[0], glmm_load(mat[0])); + glmm_store(dest[1], glmm_load(mat[1])); + glmm_store(dest[2], glmm_load(mat[2])); + glmm_store(dest[3], glmm_load(mat[3])); +#elif defined(__AVX__) + glmm_store256(dest[0], glmm_load256(mat[0])); + glmm_store256(dest[2], glmm_load256(mat[2])); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest[0], glmm_load(mat[0])); + glmm_store(dest[1], glmm_load(mat[1])); + glmm_store(dest[2], glmm_load(mat[2])); + glmm_store(dest[3], glmm_load(mat[3])); +#elif defined(CGLM_NEON_FP) + vst1q_f32(dest[0], vld1q_f32(mat[0])); + vst1q_f32(dest[1], vld1q_f32(mat[1])); + vst1q_f32(dest[2], vld1q_f32(mat[2])); + vst1q_f32(dest[3], vld1q_f32(mat[3])); +#else + glm_mat4_ucopy(mat, dest); +#endif +} + +/*! + * @brief make given matrix identity. It is identical with below, + * but it is more easy to do that with this func especially for members + * e.g. glm_mat4_identity(aStruct->aMatrix); + * + * @code + * glm_mat4_copy(GLM_MAT4_IDENTITY, mat); // C only + * + * // or + * mat4 mat = GLM_MAT4_IDENTITY_INIT; + * @endcode + * + * @param[in, out] mat destination + */ +CGLM_INLINE +void +glm_mat4_identity(mat4 mat) { + CGLM_ALIGN_MAT mat4 t = GLM_MAT4_IDENTITY_INIT; + glm_mat4_copy(t, mat); +} + +/*! + * @brief make given matrix array's each element identity matrix + * + * @param[in, out] mat matrix array (must be aligned (16/32) + * if alignment is not disabled) + * + * @param[in] count count of matrices + */ +CGLM_INLINE +void +glm_mat4_identity_array(mat4 * __restrict mat, size_t count) { + CGLM_ALIGN_MAT mat4 t = GLM_MAT4_IDENTITY_INIT; + size_t i; + + for (i = 0; i < count; i++) { + glm_mat4_copy(t, mat[i]); + } +} + +/*! + * @brief make given matrix zero. + * + * @param[in, out] mat matrix + */ +CGLM_INLINE +void +glm_mat4_zero(mat4 mat) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_128 x0; + x0 = wasm_f32x4_const_splat(0.f); + glmm_store(mat[0], x0); + glmm_store(mat[1], x0); + glmm_store(mat[2], x0); + glmm_store(mat[3], x0); +#elif defined(__AVX__) + __m256 y0; + y0 = _mm256_setzero_ps(); + glmm_store256(mat[0], y0); + glmm_store256(mat[2], y0); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_128 x0; + x0 = _mm_setzero_ps(); + glmm_store(mat[0], x0); + glmm_store(mat[1], x0); + glmm_store(mat[2], x0); + glmm_store(mat[3], x0); +#elif defined(CGLM_NEON_FP) + glmm_128 x0; + x0 = vdupq_n_f32(0.0f); + vst1q_f32(mat[0], x0); + vst1q_f32(mat[1], x0); + vst1q_f32(mat[2], x0); + vst1q_f32(mat[3], x0); +#else + CGLM_ALIGN_MAT mat4 t = GLM_MAT4_ZERO_INIT; + glm_mat4_copy(t, mat); +#endif +} + +/*! + * @brief copy upper-left of mat4 to mat3 + * + * @param[in] mat source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_mat4_pick3(mat4 mat, mat3 dest) { + dest[0][0] = mat[0][0]; + dest[0][1] = mat[0][1]; + dest[0][2] = mat[0][2]; + + dest[1][0] = mat[1][0]; + dest[1][1] = mat[1][1]; + dest[1][2] = mat[1][2]; + + dest[2][0] = mat[2][0]; + dest[2][1] = mat[2][1]; + dest[2][2] = mat[2][2]; +} + +/*! + * @brief copy upper-left of mat4 to mat3 (transposed) + * + * the postfix t stands for transpose + * + * @param[in] mat source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_mat4_pick3t(mat4 mat, mat3 dest) { + dest[0][0] = mat[0][0]; + dest[0][1] = mat[1][0]; + dest[0][2] = mat[2][0]; + + dest[1][0] = mat[0][1]; + dest[1][1] = mat[1][1]; + dest[1][2] = mat[2][1]; + + dest[2][0] = mat[0][2]; + dest[2][1] = mat[1][2]; + dest[2][2] = mat[2][2]; +} + +/*! + * @brief copy mat3 to mat4's upper-left + * + * @param[in] mat source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_mat4_ins3(mat3 mat, mat4 dest) { + dest[0][0] = mat[0][0]; + dest[0][1] = mat[0][1]; + dest[0][2] = mat[0][2]; + + dest[1][0] = mat[1][0]; + dest[1][1] = mat[1][1]; + dest[1][2] = mat[1][2]; + + dest[2][0] = mat[2][0]; + dest[2][1] = mat[2][1]; + dest[2][2] = mat[2][2]; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * m1, m2 and dest matrices can be same matrix, it is possible to write this: + * + * @code + * mat4 m = GLM_MAT4_IDENTITY_INIT; + * glm_mat4_mul(m, m, m); + * @endcode + * + * @param[in] m1 left matrix + * @param[in] m2 right matrix + * @param[out] dest destination matrix + */ +CGLM_INLINE +void +glm_mat4_mul(mat4 m1, mat4 m2, mat4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glm_mat4_mul_wasm(m1, m2, dest); +#elif defined(__AVX__) + glm_mat4_mul_avx(m1, m2, dest); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glm_mat4_mul_sse2(m1, m2, dest); +#elif defined(CGLM_NEON_FP) + glm_mat4_mul_neon(m1, m2, dest); +#else + float a00 = m1[0][0], a01 = m1[0][1], a02 = m1[0][2], a03 = m1[0][3], + a10 = m1[1][0], a11 = m1[1][1], a12 = m1[1][2], a13 = m1[1][3], + a20 = m1[2][0], a21 = m1[2][1], a22 = m1[2][2], a23 = m1[2][3], + a30 = m1[3][0], a31 = m1[3][1], a32 = m1[3][2], a33 = m1[3][3], + + b00 = m2[0][0], b01 = m2[0][1], b02 = m2[0][2], b03 = m2[0][3], + b10 = m2[1][0], b11 = m2[1][1], b12 = m2[1][2], b13 = m2[1][3], + b20 = m2[2][0], b21 = m2[2][1], b22 = m2[2][2], b23 = m2[2][3], + b30 = m2[3][0], b31 = m2[3][1], b32 = m2[3][2], b33 = m2[3][3]; + + dest[0][0] = a00 * b00 + a10 * b01 + a20 * b02 + a30 * b03; + dest[0][1] = a01 * b00 + a11 * b01 + a21 * b02 + a31 * b03; + dest[0][2] = a02 * b00 + a12 * b01 + a22 * b02 + a32 * b03; + dest[0][3] = a03 * b00 + a13 * b01 + a23 * b02 + a33 * b03; + dest[1][0] = a00 * b10 + a10 * b11 + a20 * b12 + a30 * b13; + dest[1][1] = a01 * b10 + a11 * b11 + a21 * b12 + a31 * b13; + dest[1][2] = a02 * b10 + a12 * b11 + a22 * b12 + a32 * b13; + dest[1][3] = a03 * b10 + a13 * b11 + a23 * b12 + a33 * b13; + dest[2][0] = a00 * b20 + a10 * b21 + a20 * b22 + a30 * b23; + dest[2][1] = a01 * b20 + a11 * b21 + a21 * b22 + a31 * b23; + dest[2][2] = a02 * b20 + a12 * b21 + a22 * b22 + a32 * b23; + dest[2][3] = a03 * b20 + a13 * b21 + a23 * b22 + a33 * b23; + dest[3][0] = a00 * b30 + a10 * b31 + a20 * b32 + a30 * b33; + dest[3][1] = a01 * b30 + a11 * b31 + a21 * b32 + a31 * b33; + dest[3][2] = a02 * b30 + a12 * b31 + a22 * b32 + a32 * b33; + dest[3][3] = a03 * b30 + a13 * b31 + a23 * b32 + a33 * b33; +#endif +} + +/*! + * @brief mupliply N mat4 matrices and store result in dest + * + * this function lets you multiply multiple (more than two or more...) matrices + *

multiplication will be done in loop, this may reduce instructions + * size but if len is too small then compiler may unroll whole loop, + * usage: + * @code + * mat m1, m2, m3, m4, res; + * + * glm_mat4_mulN((mat4 *[]){&m1, &m2, &m3, &m4}, 4, res); + * @endcode + * + * @warning matrices parameter is pointer array not mat4 array! + * + * @param[in] matrices mat4 * array + * @param[in] len matrices count + * @param[out] dest result + */ +CGLM_INLINE +void +glm_mat4_mulN(mat4 * __restrict matrices[], uint32_t len, mat4 dest) { + uint32_t i; + +#ifndef NDEBUG + assert(len > 1 && "there must be least 2 matrices to go!"); +#endif + + glm_mat4_mul(*matrices[0], *matrices[1], dest); + + for (i = 2; i < len; i++) + glm_mat4_mul(dest, *matrices[i], dest); +} + +/*! + * @brief multiply mat4 with vec4 (column vector) and store in dest vector + * + * @param[in] m mat4 (left) + * @param[in] v vec4 (right, column vector) + * @param[out] dest vec4 (result, column vector) + */ +CGLM_INLINE +void +glm_mat4_mulv(mat4 m, vec4 v, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glm_mat4_mulv_wasm(m, v, dest); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glm_mat4_mulv_sse2(m, v, dest); +#elif defined(CGLM_NEON_FP) + glm_mat4_mulv_neon(m, v, dest); +#else + vec4 res; + res[0] = m[0][0] * v[0] + m[1][0] * v[1] + m[2][0] * v[2] + m[3][0] * v[3]; + res[1] = m[0][1] * v[0] + m[1][1] * v[1] + m[2][1] * v[2] + m[3][1] * v[3]; + res[2] = m[0][2] * v[0] + m[1][2] * v[1] + m[2][2] * v[2] + m[3][2] * v[3]; + res[3] = m[0][3] * v[0] + m[1][3] * v[1] + m[2][3] * v[2] + m[3][3] * v[3]; + glm_vec4_copy(res, dest); +#endif +} + +/*! + * @brief trace of matrix + * + * sum of the elements on the main diagonal from upper left to the lower right + * + * @param[in] m matrix + */ +CGLM_INLINE +float +glm_mat4_trace(mat4 m) { + return m[0][0] + m[1][1] + m[2][2] + m[3][3]; +} + +/*! + * @brief trace of matrix (rotation part) + * + * sum of the elements on the main diagonal from upper left to the lower right + * + * @param[in] m matrix + */ +CGLM_INLINE +float +glm_mat4_trace3(mat4 m) { + return m[0][0] + m[1][1] + m[2][2]; +} + +/*! + * @brief convert mat4's rotation part to quaternion + * + * @param[in] m affine matrix + * @param[out] dest destination quaternion + */ +CGLM_INLINE +void +glm_mat4_quat(mat4 m, versor dest) { + float trace, r, rinv; + + /* it seems using like m12 instead of m[1][2] causes extra instructions */ + + trace = m[0][0] + m[1][1] + m[2][2]; + if (trace >= 0.0f) { + r = sqrtf(1.0f + trace); + rinv = 0.5f / r; + + dest[0] = rinv * (m[1][2] - m[2][1]); + dest[1] = rinv * (m[2][0] - m[0][2]); + dest[2] = rinv * (m[0][1] - m[1][0]); + dest[3] = r * 0.5f; + } else if (m[0][0] >= m[1][1] && m[0][0] >= m[2][2]) { + r = sqrtf(1.0f - m[1][1] - m[2][2] + m[0][0]); + rinv = 0.5f / r; + + dest[0] = r * 0.5f; + dest[1] = rinv * (m[0][1] + m[1][0]); + dest[2] = rinv * (m[0][2] + m[2][0]); + dest[3] = rinv * (m[1][2] - m[2][1]); + } else if (m[1][1] >= m[2][2]) { + r = sqrtf(1.0f - m[0][0] - m[2][2] + m[1][1]); + rinv = 0.5f / r; + + dest[0] = rinv * (m[0][1] + m[1][0]); + dest[1] = r * 0.5f; + dest[2] = rinv * (m[1][2] + m[2][1]); + dest[3] = rinv * (m[2][0] - m[0][2]); + } else { + r = sqrtf(1.0f - m[0][0] - m[1][1] + m[2][2]); + rinv = 0.5f / r; + + dest[0] = rinv * (m[0][2] + m[2][0]); + dest[1] = rinv * (m[1][2] + m[2][1]); + dest[2] = r * 0.5f; + dest[3] = rinv * (m[0][1] - m[1][0]); + } +} + +/*! + * @brief multiply vector with mat4 + * + * actually the result is vec4, after multiplication the last component + * is trimmed. if you need it don't use this func. + * + * @param[in] m mat4(affine transform) + * @param[in] v vec3 + * @param[in] last 4th item to make it vec4 + * @param[out] dest result vector (vec3) + */ +CGLM_INLINE +void +glm_mat4_mulv3(mat4 m, vec3 v, float last, vec3 dest) { + vec4 res; + glm_vec4(v, last, res); + glm_mat4_mulv(m, res, res); + glm_vec3(res, dest); +} + +/*! + * @brief transpose mat4 and store in dest + * + * source matrix will not be transposed unless dest is m + * + * @param[in] m matrix + * @param[out] dest result + */ +CGLM_INLINE +void +glm_mat4_transpose_to(mat4 m, mat4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glm_mat4_transp_wasm(m, dest); +#elif defined(__AVX__) + glm_mat4_transp_avx(m, dest); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glm_mat4_transp_sse2(m, dest); +#elif defined(CGLM_NEON_FP) + glm_mat4_transp_neon(m, dest); +#else + dest[0][0] = m[0][0]; dest[1][0] = m[0][1]; + dest[0][1] = m[1][0]; dest[1][1] = m[1][1]; + dest[0][2] = m[2][0]; dest[1][2] = m[2][1]; + dest[0][3] = m[3][0]; dest[1][3] = m[3][1]; + dest[2][0] = m[0][2]; dest[3][0] = m[0][3]; + dest[2][1] = m[1][2]; dest[3][1] = m[1][3]; + dest[2][2] = m[2][2]; dest[3][2] = m[2][3]; + dest[2][3] = m[3][2]; dest[3][3] = m[3][3]; +#endif +} + +/*! + * @brief transpose mat4 and store result in same matrix + * + * @param[in, out] m source and dest + */ +CGLM_INLINE +void +glm_mat4_transpose(mat4 m) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glm_mat4_transp_wasm(m, m); +#elif defined(__AVX__) + glm_mat4_transp_avx(m, m); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glm_mat4_transp_sse2(m, m); +#elif defined(CGLM_NEON_FP) + glm_mat4_transp_neon(m, m); +#else + mat4 d; + glm_mat4_transpose_to(m, d); + glm_mat4_ucopy(d, m); +#endif +} + +/*! + * @brief scale (multiply with scalar) matrix without simd optimization + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +void +glm_mat4_scale_p(mat4 m, float s) { + m[0][0] *= s; m[0][1] *= s; m[0][2] *= s; m[0][3] *= s; + m[1][0] *= s; m[1][1] *= s; m[1][2] *= s; m[1][3] *= s; + m[2][0] *= s; m[2][1] *= s; m[2][2] *= s; m[2][3] *= s; + m[3][0] *= s; m[3][1] *= s; m[3][2] *= s; m[3][3] *= s; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +void +glm_mat4_scale(mat4 m, float s) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glm_mat4_scale_wasm(m, s); +#elif defined(__AVX__) + glm_mat4_scale_avx(m, s); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glm_mat4_scale_sse2(m, s); +#elif defined(CGLM_NEON_FP) + glm_mat4_scale_neon(m, s); +#else + glm_mat4_scale_p(m, s); +#endif +} + +/*! + * @brief mat4 determinant + * + * @param[in] mat matrix + * + * @return determinant + */ +CGLM_INLINE +float +glm_mat4_det(mat4 mat) { +#if defined(__wasm__) && defined(__wasm_simd128__) + return glm_mat4_det_wasm(mat); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + return glm_mat4_det_sse2(mat); +#elif defined(CGLM_NEON_FP) + return glm_mat4_det_neon(mat); +#else + /* [square] det(A) = det(At) */ + float t[6]; + float a = mat[0][0], b = mat[0][1], c = mat[0][2], d = mat[0][3], + e = mat[1][0], f = mat[1][1], g = mat[1][2], h = mat[1][3], + i = mat[2][0], j = mat[2][1], k = mat[2][2], l = mat[2][3], + m = mat[3][0], n = mat[3][1], o = mat[3][2], p = mat[3][3]; + + t[0] = k * p - o * l; + t[1] = j * p - n * l; + t[2] = j * o - n * k; + t[3] = i * p - m * l; + t[4] = i * o - m * k; + t[5] = i * n - m * j; + + return a * (f * t[0] - g * t[1] + h * t[2]) + - b * (e * t[0] - g * t[3] + h * t[4]) + + c * (e * t[1] - f * t[3] + h * t[5]) + - d * (e * t[2] - f * t[4] + g * t[5]); +#endif +} + +/*! + * @brief inverse mat4 and store in dest + * + * @param[in] mat matrix + * @param[out] dest inverse matrix + */ +CGLM_INLINE +void +glm_mat4_inv(mat4 mat, mat4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glm_mat4_inv_wasm(mat, dest); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glm_mat4_inv_sse2(mat, dest); +#elif defined(CGLM_NEON_FP) + glm_mat4_inv_neon(mat, dest); +#else + float a = mat[0][0], b = mat[0][1], c = mat[0][2], d = mat[0][3], + e = mat[1][0], f = mat[1][1], g = mat[1][2], h = mat[1][3], + i = mat[2][0], j = mat[2][1], k = mat[2][2], l = mat[2][3], + m = mat[3][0], n = mat[3][1], o = mat[3][2], p = mat[3][3], + + c1 = k * p - l * o, c2 = c * h - d * g, c3 = i * p - l * m, + c4 = a * h - d * e, c5 = j * p - l * n, c6 = b * h - d * f, + c7 = i * n - j * m, c8 = a * f - b * e, c9 = j * o - k * n, + c10 = b * g - c * f, c11 = i * o - k * m, c12 = a * g - c * e, + + idt = 1.0f/(c8*c1+c4*c9+c10*c3+c2*c7-c12*c5-c6*c11), ndt = -idt; + + dest[0][0] = (f * c1 - g * c5 + h * c9) * idt; + dest[0][1] = (b * c1 - c * c5 + d * c9) * ndt; + dest[0][2] = (n * c2 - o * c6 + p * c10) * idt; + dest[0][3] = (j * c2 - k * c6 + l * c10) * ndt; + + dest[1][0] = (e * c1 - g * c3 + h * c11) * ndt; + dest[1][1] = (a * c1 - c * c3 + d * c11) * idt; + dest[1][2] = (m * c2 - o * c4 + p * c12) * ndt; + dest[1][3] = (i * c2 - k * c4 + l * c12) * idt; + + dest[2][0] = (e * c5 - f * c3 + h * c7) * idt; + dest[2][1] = (a * c5 - b * c3 + d * c7) * ndt; + dest[2][2] = (m * c6 - n * c4 + p * c8) * idt; + dest[2][3] = (i * c6 - j * c4 + l * c8) * ndt; + + dest[3][0] = (e * c9 - f * c11 + g * c7) * ndt; + dest[3][1] = (a * c9 - b * c11 + c * c7) * idt; + dest[3][2] = (m * c10 - n * c12 + o * c8) * ndt; + dest[3][3] = (i * c10 - j * c12 + k * c8) * idt; +#endif +} + +/*! + * @brief inverse mat4 and store in dest + * + * this func uses reciprocal approximation without extra corrections + * e.g Newton-Raphson. this should work faster than normal, + * to get more precise use glm_mat4_inv version. + * + * NOTE: You will lose precision, glm_mat4_inv is more accurate + * + * @param[in] mat matrix + * @param[out] dest inverse matrix + */ +CGLM_INLINE +void +glm_mat4_inv_fast(mat4 mat, mat4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glm_mat4_inv_fast_wasm(mat, dest); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glm_mat4_inv_fast_sse2(mat, dest); +#else + glm_mat4_inv(mat, dest); +#endif +} + +/*! + * @brief swap two matrix columns + * + * @param[in,out] mat matrix + * @param[in] col1 col1 + * @param[in] col2 col2 + */ +CGLM_INLINE +void +glm_mat4_swap_col(mat4 mat, int col1, int col2) { + CGLM_ALIGN(16) vec4 tmp; + glm_vec4_copy(mat[col1], tmp); + glm_vec4_copy(mat[col2], mat[col1]); + glm_vec4_copy(tmp, mat[col2]); +} + +/*! + * @brief swap two matrix rows + * + * @param[in,out] mat matrix + * @param[in] row1 row1 + * @param[in] row2 row2 + */ +CGLM_INLINE +void +glm_mat4_swap_row(mat4 mat, int row1, int row2) { + CGLM_ALIGN(16) vec4 tmp; + tmp[0] = mat[0][row1]; + tmp[1] = mat[1][row1]; + tmp[2] = mat[2][row1]; + tmp[3] = mat[3][row1]; + + mat[0][row1] = mat[0][row2]; + mat[1][row1] = mat[1][row2]; + mat[2][row1] = mat[2][row2]; + mat[3][row1] = mat[3][row2]; + + mat[0][row2] = tmp[0]; + mat[1][row2] = tmp[1]; + mat[2][row2] = tmp[2]; + mat[3][row2] = tmp[3]; +} + +/*! + * @brief helper for R (row vector) * M (matrix) * C (column vector) + * + * rmc stands for Row * Matrix * Column + * + * the result is scalar because R * M = Matrix1x4 (row vector), + * then Matrix1x4 * Vec4 (column vector) = Matrix1x1 (Scalar) + * + * @param[in] r row vector or matrix1x4 + * @param[in] m matrix4x4 + * @param[in] c column vector or matrix4x1 + * + * @return scalar value e.g. B(s) + */ +CGLM_INLINE +float +glm_mat4_rmc(vec4 r, mat4 m, vec4 c) { + vec4 tmp; + glm_mat4_mulv(m, c, tmp); + return glm_vec4_dot(r, tmp); +} + +/*! + * @brief Create mat4 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @param[out] dest matrix + */ +CGLM_INLINE +void +glm_mat4_make(const float * __restrict src, mat4 dest) { + dest[0][0] = src[0]; dest[1][0] = src[4]; + dest[0][1] = src[1]; dest[1][1] = src[5]; + dest[0][2] = src[2]; dest[1][2] = src[6]; + dest[0][3] = src[3]; dest[1][3] = src[7]; + + dest[2][0] = src[8]; dest[3][0] = src[12]; + dest[2][1] = src[9]; dest[3][1] = src[13]; + dest[2][2] = src[10]; dest[3][2] = src[14]; + dest[2][3] = src[11]; dest[3][3] = src[15]; +} + +#endif /* cglm_mat_h */ diff --git a/cglm/include/cglm/mat4x2.h b/cglm/include/cglm/mat4x2.h new file mode 100644 index 0000000..d3f0ad5 --- /dev/null +++ b/cglm/include/cglm/mat4x2.h @@ -0,0 +1,168 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_MAT4X2_ZERO_INIT + GLM_MAT4X2_ZERO + + Functions: + CGLM_INLINE void glm_mat4x2_copy(mat4x2 mat, mat4x2 dest); + CGLM_INLINE void glm_mat4x2_zero(mat4x2 mat); + CGLM_INLINE void glm_mat4x2_make(const float * __restrict src, mat4x2 dest); + CGLM_INLINE void glm_mat4x2_mul(mat4x2 m1, mat2x4 m2, mat2 dest); + CGLM_INLINE void glm_mat4x2_mulv(mat4x2 m, vec4 v, vec2 dest); + CGLM_INLINE void glm_mat4x2_transpose(mat4x2 m, mat2x4 dest); + CGLM_INLINE void glm_mat4x2_scale(mat4x2 m, float s); + */ + +#ifndef cglm_mat4x2_h +#define cglm_mat4x2_h + +#include "common.h" + +#define GLM_MAT4X2_ZERO_INIT {{0.0f, 0.0f}, {0.0f, 0.0f}, {0.0f, 0.0f}, {0.0f, 0.0f}} + +/* for C only */ +#define GLM_MAT4X2_ZERO GLM_MAT4X2_ZERO_INIT + +/*! + * @brief copy all members of [mat] to [dest] + * + * @param[in] mat source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_mat4x2_copy(mat4x2 mat, mat4x2 dest) { + dest[0][0] = mat[0][0]; + dest[0][1] = mat[0][1]; + + dest[1][0] = mat[1][0]; + dest[1][1] = mat[1][1]; + + dest[2][0] = mat[2][0]; + dest[2][1] = mat[2][1]; + + dest[3][0] = mat[3][0]; + dest[3][1] = mat[3][1]; +} + +/*! + * @brief make given matrix zero. + * + * @param[in, out] mat matrix + */ +CGLM_INLINE +void +glm_mat4x2_zero(mat4x2 mat) { + CGLM_ALIGN_MAT mat4x2 t = GLM_MAT4X2_ZERO_INIT; + glm_mat4x2_copy(t, mat); +} + +/*! + * @brief Create mat4x2 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @param[out] dest matrix + */ +CGLM_INLINE +void +glm_mat4x2_make(const float * __restrict src, mat4x2 dest) { + dest[0][0] = src[0]; + dest[0][1] = src[1]; + + dest[1][0] = src[2]; + dest[1][1] = src[3]; + + dest[2][0] = src[4]; + dest[2][1] = src[5]; + + dest[3][0] = src[6]; + dest[3][1] = src[7]; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * @code + * glm_mat4x2_mul(mat4x2, mat2x4, mat2); + * @endcode + * + * @param[in] m1 left matrix (mat4x2) + * @param[in] m2 right matrix (mat2x4) + * @param[out] dest destination matrix (mat2) + */ +CGLM_INLINE +void +glm_mat4x2_mul(mat4x2 m1, mat2x4 m2, mat2 dest) { + float a00 = m1[0][0], a01 = m1[0][1], + a10 = m1[1][0], a11 = m1[1][1], + a20 = m1[2][0], a21 = m1[2][1], + a30 = m1[3][0], a31 = m1[3][1], + + b00 = m2[0][0], b01 = m2[0][1], b02 = m2[0][2], b03 = m2[0][3], + b10 = m2[1][0], b11 = m2[1][1], b12 = m2[1][2], b13 = m2[1][3]; + + dest[0][0] = a00 * b00 + a10 * b01 + a20 * b02 + a30 * b03; + dest[0][1] = a01 * b00 + a11 * b01 + a21 * b02 + a31 * b03; + + dest[1][0] = a00 * b10 + a10 * b11 + a20 * b12 + a30 * b13; + dest[1][1] = a01 * b10 + a11 * b11 + a21 * b12 + a31 * b13; +} + +/*! + * @brief multiply matrix with column vector and store in dest column vector + * + * @param[in] m matrix (left) + * @param[in] v vector (right, column vector) + * @param[out] dest result vector + */ +CGLM_INLINE +void +glm_mat4x2_mulv(mat4x2 m, vec4 v, vec2 dest) { + float v0 = v[0], v1 = v[1], v2 = v[2], v3 = v[3]; + + dest[0] = m[0][0] * v0 + m[1][0] * v1 + m[2][0] * v2 + m[3][0] * v3; + dest[1] = m[0][1] * v0 + m[1][1] * v1 + m[2][1] * v2 + m[3][1] * v3; +} + +/*! + * @brief transpose matrix and store in dest + * + * @param[in] m matrix + * @param[out] dest result + */ +CGLM_INLINE +void +glm_mat4x2_transpose(mat4x2 m, mat2x4 dest) { + dest[0][0] = m[0][0]; + dest[0][1] = m[1][0]; + dest[0][2] = m[2][0]; + dest[0][3] = m[3][0]; + dest[1][0] = m[0][1]; + dest[1][1] = m[1][1]; + dest[1][2] = m[2][1]; + dest[1][3] = m[3][1]; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +void +glm_mat4x2_scale(mat4x2 m, float s) { + m[0][0] *= s; m[0][1] *= s; m[1][0] *= s; m[1][1] *= s; + m[2][0] *= s; m[2][1] *= s; m[3][0] *= s; m[3][1] *= s; +} + +#endif diff --git a/cglm/include/cglm/mat4x3.h b/cglm/include/cglm/mat4x3.h new file mode 100644 index 0000000..2ef05c1 --- /dev/null +++ b/cglm/include/cglm/mat4x3.h @@ -0,0 +1,192 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_MAT4X3_ZERO_INIT + GLM_MAT4X3_ZERO + + Functions: + CGLM_INLINE void glm_mat4x3_copy(mat4x3 mat, mat4x3 dest); + CGLM_INLINE void glm_mat4x3_zero(mat4x3 mat); + CGLM_INLINE void glm_mat4x3_make(const float * __restrict src, mat4x3 dest); + CGLM_INLINE void glm_mat4x3_mul(mat4x3 m1, mat3x4 m2, mat3 dest); + CGLM_INLINE void glm_mat4x3_mulv(mat4x3 m, vec4 v, vec3 dest); + CGLM_INLINE void glm_mat4x3_transpose(mat4x3 m, mat3x4 dest); + CGLM_INLINE void glm_mat4x3_scale(mat4x3 m, float s); + */ + +#ifndef cglm_mat4x3_h +#define cglm_mat4x3_h + +#include "common.h" + +#define GLM_MAT4X3_ZERO_INIT {{0.0f, 0.0f, 0.0f}, {0.0f, 0.0f, 0.0f}, \ + {0.0f, 0.0f, 0.0f}, {0.0f, 0.0f, 0.0f}} + +/* for C only */ +#define GLM_MAT4X3_ZERO GLM_MAT4X3_ZERO_INIT + +/*! + * @brief copy all members of [mat] to [dest] + * + * @param[in] mat source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_mat4x3_copy(mat4x3 mat, mat4x3 dest) { + dest[0][0] = mat[0][0]; + dest[0][1] = mat[0][1]; + dest[0][2] = mat[0][2]; + + dest[1][0] = mat[1][0]; + dest[1][1] = mat[1][1]; + dest[1][2] = mat[1][2]; + + dest[2][0] = mat[2][0]; + dest[2][1] = mat[2][1]; + dest[2][2] = mat[2][2]; + + dest[3][0] = mat[3][0]; + dest[3][1] = mat[3][1]; + dest[3][2] = mat[3][2]; +} + +/*! + * @brief make given matrix zero. + * + * @param[in, out] mat matrix + */ +CGLM_INLINE +void +glm_mat4x3_zero(mat4x3 mat) { + CGLM_ALIGN_MAT mat4x3 t = GLM_MAT4X3_ZERO_INIT; + glm_mat4x3_copy(t, mat); +} + +/*! + * @brief Create mat4x3 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @param[out] dest matrix + */ +CGLM_INLINE +void +glm_mat4x3_make(const float * __restrict src, mat4x3 dest) { + dest[0][0] = src[0]; + dest[0][1] = src[1]; + dest[0][2] = src[2]; + + dest[1][0] = src[3]; + dest[1][1] = src[4]; + dest[1][2] = src[5]; + + dest[2][0] = src[6]; + dest[2][1] = src[7]; + dest[2][2] = src[8]; + + dest[3][0] = src[9]; + dest[3][1] = src[10]; + dest[3][2] = src[11]; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * @code + * glm_mat4x3_mul(mat4x3, mat3x4, mat3); + * @endcode + * + * @param[in] m1 left matrix (mat4x3) + * @param[in] m2 right matrix (mat3x4) + * @param[out] dest destination matrix (mat3) + */ +CGLM_INLINE +void +glm_mat4x3_mul(mat4x3 m1, mat3x4 m2, mat3 dest) { + float a00 = m1[0][0], a01 = m1[0][1], a02 = m1[0][2], + a10 = m1[1][0], a11 = m1[1][1], a12 = m1[1][2], + a20 = m1[2][0], a21 = m1[2][1], a22 = m1[2][2], + a30 = m1[3][0], a31 = m1[3][1], a32 = m1[3][2], + + b00 = m2[0][0], b01 = m2[0][1], b02 = m2[0][2], b03 = m2[0][3], + b10 = m2[1][0], b11 = m2[1][1], b12 = m2[1][2], b13 = m2[1][3], + b20 = m2[2][0], b21 = m2[2][1], b22 = m2[2][2], b23 = m2[2][3]; + + dest[0][0] = a00 * b00 + a10 * b01 + a20 * b02 + a30 * b03; + dest[0][1] = a01 * b00 + a11 * b01 + a21 * b02 + a31 * b03; + dest[0][2] = a02 * b00 + a12 * b01 + a22 * b02 + a32 * b03; + + dest[1][0] = a00 * b10 + a10 * b11 + a20 * b12 + a30 * b13; + dest[1][1] = a01 * b10 + a11 * b11 + a21 * b12 + a31 * b13; + dest[1][2] = a02 * b10 + a12 * b11 + a22 * b12 + a32 * b13; + + dest[2][0] = a00 * b20 + a10 * b21 + a20 * b22 + a30 * b23; + dest[2][1] = a01 * b20 + a11 * b21 + a21 * b22 + a31 * b23; + dest[2][2] = a02 * b20 + a12 * b21 + a22 * b22 + a32 * b23; +} + +/*! + * @brief multiply matrix with column vector and store in dest column vector + * + * @param[in] m matrix (left) + * @param[in] v vector (right, column vector) + * @param[out] dest result vector + */ +CGLM_INLINE +void +glm_mat4x3_mulv(mat4x3 m, vec4 v, vec3 dest) { + float v0 = v[0], v1 = v[1], v2 = v[2], v3 = v[3]; + + dest[0] = m[0][0] * v0 + m[1][0] * v1 + m[2][0] * v2 + m[3][0] * v3; + dest[1] = m[0][1] * v0 + m[1][1] * v1 + m[2][1] * v2 + m[3][1] * v3; + dest[2] = m[0][2] * v0 + m[1][2] * v1 + m[2][2] * v2 + m[3][2] * v3; +} + +/*! + * @brief transpose matrix and store in dest + * + * @param[in] m matrix + * @param[out] dest result + */ +CGLM_INLINE +void +glm_mat4x3_transpose(mat4x3 m, mat3x4 dest) { + dest[0][0] = m[0][0]; + dest[0][1] = m[1][0]; + dest[0][2] = m[2][0]; + dest[0][3] = m[3][0]; + + dest[1][0] = m[0][1]; + dest[1][1] = m[1][1]; + dest[1][2] = m[2][1]; + dest[1][3] = m[3][1]; + + dest[2][0] = m[0][2]; + dest[2][1] = m[1][2]; + dest[2][2] = m[2][2]; + dest[2][3] = m[3][2]; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +void +glm_mat4x3_scale(mat4x3 m, float s) { + m[0][0] *= s; m[0][1] *= s; m[0][2] *= s; m[1][0] *= s; + m[1][1] *= s; m[1][2] *= s; m[2][0] *= s; m[2][1] *= s; + m[2][2] *= s; m[3][0] *= s; m[3][1] *= s; m[3][2] *= s; +} + +#endif diff --git a/cglm/include/cglm/noise.h b/cglm/include/cglm/noise.h new file mode 100644 index 0000000..bec12e9 --- /dev/null +++ b/cglm/include/cglm/noise.h @@ -0,0 +1,734 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + * + * Based on the work of Stefan Gustavson and Ashima Arts on "webgl-noise": + * https://github.com/stegu/webgl-noise + * Following Stefan Gustavson's paper "Simplex noise demystified": + * http://www.itn.liu.se/~stegu/simplexnoise/simplexnoise.pdf + * + * Implementation based on glm::perlin function: + * https://github.com/g-truc/glm/blob/master/glm/gtc/noise.inl + */ + +#ifndef cglm_noise_h +#define cglm_noise_h + +#include "vec4.h" +#include "vec4-ext.h" + +#include "vec3.h" +#include "vec3-ext.h" + +#include "vec2.h" +#include "vec2-ext.h" + +#define glm__noiseDetail_mod289(x) (x - floorf(x * (1.0f / 289.0f)) * 289.0f) + +/* glm__noiseDetail_permute(vec4 x, vec4 dest) */ +#define glm__noiseDetail_permute(x, dest) { \ + dest[0] = glm__noiseDetail_mod289((x[0] * 34.0f + 1.0f) * x[0]); \ + dest[1] = glm__noiseDetail_mod289((x[1] * 34.0f + 1.0f) * x[1]); \ + dest[2] = glm__noiseDetail_mod289((x[2] * 34.0f + 1.0f) * x[2]); \ + dest[3] = glm__noiseDetail_mod289((x[3] * 34.0f + 1.0f) * x[3]); \ +} + +/* glm__noiseDetail_fade_vec4(vec4 t, vec4 dest) */ +#define glm__noiseDetail_fade_vec4(t, dest) { \ + /* dest = (t * t * t) * (t * (t * 6.0f - 15.0f) + 10.0f) */ \ + vec4 temp; \ + glm_vec4_mul(t, t, temp); \ + glm_vec4_mul(temp, t, temp); \ + /* dest = (t * (t * 6.0f - 15.0f) + 10.0f) */ \ + glm_vec4_scale(t, 6.0f, dest); \ + glm_vec4_subs(dest, 15.0f, dest); \ + glm_vec4_mul(t, dest, dest); \ + glm_vec4_adds(dest, 10.0f, dest); \ + /* dest = temp * dest */ \ + glm_vec4_mul(temp, dest, dest); \ +} + +/* glm__noiseDetail_fade_vec3(vec3 t, vec3 dest) */ +#define glm__noiseDetail_fade_vec3(t, dest) { \ + /* dest = (t * t * t) * (t * (t * 6.0f - 15.0f) + 10.0f) */ \ + /* temp = t * t * t */ \ + vec3 temp; \ + glm_vec3_mul(t, t, temp); \ + glm_vec3_mul(temp, t, temp); \ + /* dest = (t * (t * 6.0f - 15.0f) + 10.0f) */ \ + glm_vec3_scale(t, 6.0f, dest); \ + glm_vec3_subs(dest, 15.0f, dest); \ + glm_vec3_mul(t, dest, dest); \ + glm_vec3_adds(dest, 10.0f, dest); \ + /* dest = temp * dest */ \ + glm_vec3_mul(temp, dest, dest); \ +} + +/* glm__noiseDetail_fade_vec2(vec2 t, vec2 dest) */ +#define glm__noiseDetail_fade_vec2(t, dest) { \ + /* dest = (t * t * t) * (t * (t * 6.0f - 15.0f) + 10.0f) */ \ + /* temp = t * t * t */ \ + vec2 temp; \ + glm_vec2_mul(t, t, temp); \ + glm_vec2_mul(temp, t, temp); \ + /* dest = (t * (t * 6.0f - 15.0f) + 10.0f) */ \ + glm_vec2_scale(t, 6.0f, dest); \ + glm_vec2_subs(dest, 15.0f, dest); \ + glm_vec2_mul(t, dest, dest); \ + glm_vec2_adds(dest, 10.0f, dest); \ + /* dest = temp * dest */ \ + glm_vec2_mul(temp, dest, dest); \ +} + +/* glm__noiseDetail_taylorInvSqrt(vec4 x, vec4 dest) */ +#define glm__noiseDetail_taylorInvSqrt(x, dest) { \ + /* dest = 1.79284291400159f - 0.85373472095314f * x */ \ + vec4 temp; \ + glm_vec4_scale(x, 0.85373472095314f, temp); /* temp = 0.853...f * x */ \ + glm_vec4_fill(dest, 1.79284291400159f); /* dest = 1.792...f */ \ + glm_vec4_sub(dest, temp, dest); /* dest = 1.79284291400159f - temp */ \ +} + +/* norm = taylorInvSqrt(vec4( + * dot(g00__, g00__), + * dot(g01__, g01__), + * dot(g10__, g10__), + * dot(g11__, g11__) + * )); +*/ + +/* glm__noiseDetail_gradNorm_vec4(vec4 g00__, vec4 g01__, vec4 g10__, vec4 g11__) */ +#define glm__noiseDetail_gradNorm_vec4(g00__, g01__, g10__, g11__) { \ + vec4 norm; \ + norm[0] = glm_vec4_dot(g00__, g00__); /* norm.x = dot(g00__, g00__) */ \ + norm[1] = glm_vec4_dot(g01__, g01__); /* norm.y = dot(g01__, g01__) */ \ + norm[2] = glm_vec4_dot(g10__, g10__); /* norm.z = dot(g10__, g10__) */ \ + norm[3] = glm_vec4_dot(g11__, g11__); /* norm.w = dot(g11__, g11__) */ \ + glm__noiseDetail_taylorInvSqrt(norm, norm); /* norm = taylorInvSqrt(norm) */ \ + \ + glm_vec4_scale(g00__, norm[0], g00__); /* g00__ *= norm.x */ \ + glm_vec4_scale(g01__, norm[1], g01__); /* g01__ *= norm.y */ \ + glm_vec4_scale(g10__, norm[2], g10__); /* g10__ *= norm.z */ \ + glm_vec4_scale(g11__, norm[3], g11__); /* g11__ *= norm.w */ \ +} + +/* glm__noiseDetail_gradNorm_vec3(vec3 g00_, vec3 g01_, vec3 g10_, vec3 g11_) */ +#define glm__noiseDetail_gradNorm_vec3(g00_, g01_, g10_, g11_) { \ + vec4 norm; \ + norm[0] = glm_vec3_dot(g00_, g00_); /* norm.x = dot(g00_, g00_) */ \ + norm[1] = glm_vec3_dot(g01_, g01_); /* norm.y = dot(g01_, g01_) */ \ + norm[2] = glm_vec3_dot(g10_, g10_); /* norm.z = dot(g10_, g10_) */ \ + norm[3] = glm_vec3_dot(g11_, g11_); /* norm.w = dot(g11_, g11_) */ \ + glm__noiseDetail_taylorInvSqrt(norm, norm); /* norm = taylorInvSqrt(norm) */ \ + \ + glm_vec3_scale(g00_, norm[0], g00_); /* g00_ *= norm.x */ \ + glm_vec3_scale(g01_, norm[1], g01_); /* g01_ *= norm.y */ \ + glm_vec3_scale(g10_, norm[2], g10_); /* g10_ *= norm.z */ \ + glm_vec3_scale(g11_, norm[3], g11_); /* g11_ *= norm.w */ \ +} + +/* glm__noiseDetail_gradNorm_vec2(vec2 g00, vec2 g01, vec2 g10, vec2 g11) */ +#define glm__noiseDetail_gradNorm_vec2(g00, g01, g10, g11) { \ + vec4 norm; \ + norm[0] = glm_vec2_dot(g00, g00); /* norm.x = dot(g00, g00) */ \ + norm[1] = glm_vec2_dot(g01, g01); /* norm.y = dot(g01, g01) */ \ + norm[2] = glm_vec2_dot(g10, g10); /* norm.z = dot(g10, g10) */ \ + norm[3] = glm_vec2_dot(g11, g11); /* norm.w = dot(g11, g11) */ \ + glm__noiseDetail_taylorInvSqrt(norm, norm); /* norm = taylorInvSqrt(norm) */ \ + \ + glm_vec2_scale(g00, norm[0], g00); /* g00 *= norm.x */ \ + glm_vec2_scale(g01, norm[1], g01); /* g01 *= norm.y */ \ + glm_vec2_scale(g10, norm[2], g10); /* g10 *= norm.z */ \ + glm_vec2_scale(g11, norm[3], g11); /* g11 *= norm.w */ \ +} + +/* glm__noiseDetail_i2gxyzw(vec4 ixy, vec4 gx, vec4 gy, vec4 gz, vec4 gw) */ +#define glm__noiseDetail_i2gxyzw(ixy, gx, gy, gz, gw) { \ + /* gx = ixy / 7.0 */ \ + glm_vec4_divs(ixy, 7.0f, gx); /* gx = ixy / 7.0 */ \ + \ + /* gy = fract(gx) / 7.0 */ \ + glm_vec4_floor(gx, gy); /* gy = floor(gx) */ \ + glm_vec4_divs(gy, 7.0f, gy); /* gy /= 7.0 */ \ + \ + /* gz = floor(gy) / 6.0 */ \ + glm_vec4_floor(gy, gz); /* gz = floor(gy) */ \ + glm_vec4_divs(gz, 6.0f, gz); /* gz /= 6.0 */ \ + \ + /* gx = fract(gx) - 0.5f */ \ + glm_vec4_fract(gx, gx); /* gx = fract(gx) */ \ + glm_vec4_subs(gx, 0.5f, gx); /* gx -= 0.5f */ \ + \ + /* gy = fract(gy) - 0.5f */ \ + glm_vec4_fract(gy, gy); /* gy = fract(gy) */ \ + glm_vec4_subs(gy, 0.5f, gy); /* gy -= 0.5f */ \ + \ + /* gz = fract(gz) - 0.5f */ \ + glm_vec4_fract(gz, gz); /* gz = fract(gz) */ \ + glm_vec4_subs(gz, 0.5f, gz); /* gz -= 0.5f */ \ + \ + /* abs(gx), abs(gy), abs(gz) */ \ + vec4 gxa, gya, gza; \ + glm_vec4_abs(gx, gxa); /* gxa = abs(gx) */ \ + glm_vec4_abs(gy, gya); /* gya = abs(gy) */ \ + glm_vec4_abs(gz, gza); /* gza = abs(gz) */ \ + \ + /* gw = 0.75 - abs(gx) - abs(gy) - abs(gz) */ \ + glm_vec4_fill(gw, 0.75f); /* gw = 0.75 */ \ + glm_vec4_sub(gw, gxa, gw); /* gw -= gxa */ \ + glm_vec4_sub(gw, gza, gw); /* gw -= gza */ \ + glm_vec4_sub(gw, gya, gw); /* gw -= gya */ \ + \ + /* sw = step(gw, 0.0); */ \ + vec4 sw; \ + glm_vec4_stepr(gw, 0.0f, sw); /* sw = step(gw, 0.0) */ \ + \ + /* gx -= sw * (step(vec4(0), gx) - T(0.5)); */ \ + vec4 temp = {0.0f}; /* temp = 0.0 */ \ + glm_vec4_step(temp, gx, temp); /* temp = step(temp, gx) */ \ + glm_vec4_subs(temp, 0.5f, temp); /* temp -= 0.5 */ \ + glm_vec4_mul(sw, temp, temp); /* temp *= sw */ \ + glm_vec4_sub(gx, temp, gx); /* gx -= temp */ \ + \ + /* gy -= sw * (step(vec4(0), gy) - T(0.5)); */ \ + glm_vec4_zero(temp); /* reset temp */ \ + glm_vec4_step(temp, gy, temp); /* temp = step(temp, gy) */ \ + glm_vec4_subs(temp, 0.5f, temp); /* temp -= 0.5 */ \ + glm_vec4_mul(sw, temp, temp); /* temp *= sw */ \ + glm_vec4_sub(gy, temp, gy); /* gy -= temp */ \ +} + +/* NOTE: This function is not *quite* analogous to glm__noiseDetail_i2gxyzw + * to try to match the output of glm::perlin. I think it might be a bug in + * in the original implementation, but for now I'm keeping it consistent. -MK + * + * Follow up: The original implementation (glm v 1.0.1) does: + * + * vec<4, T, Q> gx0 = ixy0 * T(1.0 / 7.0); + * + * as opposed to: + * + * vec<4, T, Q> gx0 = ixy0 / T(7); + * + * This ends up mapping to different simd instructions, at least on AMD. + * The delta is tiny but it gets amplified by the rest of the noise function. + * Hence we too need to do `glm_vec4_scale` as opposed to `glm_vec4_divs`, to + * match it. -MK + */ + +/* glm__noiseDetail_i2gxyz(vec4 i, vec4 gx, vec4 gy, vec4 gz) */ +#define glm__noiseDetail_i2gxyz(ixy, gx, gy, gz) { \ + /* gx = ixy / 7.0 */ \ + glm_vec4_scale(ixy, 1.0f / 7.0f, gx); /* gx = ixy * (1/7.0) */\ + \ + /* gy = fract(floor(gx0) / 7.0)) - 0.5; */ \ + glm_vec4_floor(gx, gy); /* gy = floor(gx) */ \ + glm_vec4_scale(gy, 1.0f / 7.0f, gy); /* gy *= 1 / 7.0 */ \ + glm_vec4_fract(gy, gy); /* gy = fract(gy) */ \ + glm_vec4_subs(gy, 0.5f, gy); /* gy -= 0.5f */ \ + \ + /* gx = fract(gx); */ \ + glm_vec4_fract(gx, gx); /* gx = fract(gx) */ \ + \ + /* abs(gx), abs(gy) */ \ + vec4 gxa, gya; \ + glm_vec4_abs(gx, gxa); /* gxa = abs(gx) */ \ + glm_vec4_abs(gy, gya); /* gya = abs(gy) */ \ + \ + /* gz = vec4(0.5) - abs(gx0) - abs(gy0); */ \ + glm_vec4_fill(gz, 0.5f); /* gz = 0.5 */ \ + glm_vec4_sub(gz, gxa, gz); /* gz -= gxa */ \ + glm_vec4_sub(gz, gya, gz); /* gz -= gya */ \ + \ + /* sz = step(gw, 0.0); */ \ + vec4 sz; \ + glm_vec4_stepr(gz, 0.0f, sz); /* sz = step(gz, 0.0) */ \ + \ + /* gx0 -= sz0 * (step(0.0, gx0) - T(0.5)); */ \ + vec4 temp = {0.0f}; /* temp = 0.0 */ \ + glm_vec4_step(temp, gx, temp); /* temp = step(temp, gx) */ \ + glm_vec4_subs(temp, 0.5f, temp); /* temp -= 0.5 */ \ + glm_vec4_mul(sz, temp, temp); /* temp *= sz */ \ + glm_vec4_sub(gx, temp, gx); /* gx -= temp */ \ + \ + /* gy0 -= sz0 * (step(0.0, gy0) - T(0.5)); */ \ + glm_vec4_zero(temp); /* reset temp */ \ + glm_vec4_step(temp, gy, temp); /* temp = step(temp, gy) */ \ + glm_vec4_subs(temp, 0.5f, temp); /* temp -= 0.5 */ \ + glm_vec4_mul(sz, temp, temp); /* temp *= sz */ \ + glm_vec4_sub(gy, temp, gy); /* gy -= temp */ \ +} + +/* glm__noiseDetail_i2gxy(vec4 i, vec4 gx, vec4 gy) */ +#define glm__noiseDetail_i2gxy(i, gx, gy) { \ + /* gx = 2.0 * fract(i / 41.0) - 1.0; */ \ + glm_vec4_divs(i, 41.0f, gx); /* gx = i / 41.0 */ \ + glm_vec4_fract(gx, gx); /* gx = fract(gx) */ \ + glm_vec4_scale(gx, 2.0f, gx); /* gx *= 2.0 */ \ + glm_vec4_subs(gx, 1.0f, gx); /* gx -= 1.0 */ \ + \ + /* gy = abs(gx) - 0.5; */ \ + glm_vec4_abs(gx, gy); /* gy = abs(gx) */ \ + glm_vec4_subs(gy, 0.5f, gy); /* gy -= 0.5 */ \ + \ + /* tx = floor(gx + 0.5); */ \ + vec4 tx; \ + glm_vec4_adds(gx, 0.5f, tx); /* tx = gx + 0.5 */ \ + glm_vec4_floor(tx, tx); /* tx = floor(tx) */ \ + \ + /* gx = gx - tx; */ \ + glm_vec4_sub(gx, tx, gx); /* gx -= tx */ \ +} + +/* ============================================================================ + * Classic perlin noise + * ============================================================================ + */ + +/*! + * @brief Classic perlin noise + * + * @param[in] point 4D vector + * @returns perlin noise value + */ +CGLM_INLINE +float +glm_perlin_vec4(vec4 point) { + /* Integer part of p for indexing */ + vec4 Pi0; + glm_vec4_floor(point, Pi0); /* Pi0 = floor(point); */ + + /* Integer part + 1 */ + vec4 Pi1; + glm_vec4_adds(Pi0, 1.0f, Pi1); /* Pi1 = Pi0 + 1.0f; */ + + glm_vec4_mods(Pi0, 289.0f, Pi0); /* Pi0 = mod(Pi0, 289.0f); */ + glm_vec4_mods(Pi1, 289.0f, Pi1); /* Pi1 = mod(Pi1, 289.0f); */ + + /* Fractional part of p for interpolation */ + vec4 Pf0; + glm_vec4_fract(point, Pf0); + + /* Fractional part - 1.0 */ + vec4 Pf1; + glm_vec4_subs(Pf0, 1.0f, Pf1); + + vec4 ix = {Pi0[0], Pi1[0], Pi0[0], Pi1[0]}; + vec4 iy = {Pi0[1], Pi0[1], Pi1[1], Pi1[1]}; + vec4 iz0 = {Pi0[2], Pi0[2], Pi0[2], Pi0[2]}; /* iz0 = vec4(Pi0.z); */ + vec4 iz1 = {Pi1[2], Pi1[2], Pi1[2], Pi1[2]}; /* iz1 = vec4(Pi1.z); */ + vec4 iw0 = {Pi0[3], Pi0[3], Pi0[3], Pi0[3]}; /* iw0 = vec4(Pi0.w); */ + vec4 iw1 = {Pi1[3], Pi1[3], Pi1[3], Pi1[3]}; /* iw1 = vec4(Pi1.w); */ + + /* ------------ */ + + /* ixy = permute(permute(ix) + iy) */ + vec4 ixy; + glm__noiseDetail_permute(ix, ixy); /* ixy = permute(ix) */ + glm_vec4_add(ixy, iy, ixy); /* ixy += iy; */ + glm__noiseDetail_permute(ixy, ixy); /* ixy = permute(ixy) */ + + /* ixy0 = permute(ixy + iz0) */ + vec4 ixy0; + glm_vec4_add(ixy, iz0, ixy0); /* ixy0 = ixy + iz0 */ + glm__noiseDetail_permute(ixy0, ixy0); /* ixy0 = permute(ixy0) */ + + /* ixy1 = permute(ixy + iz1) */ + vec4 ixy1; + glm_vec4_add(ixy, iz1, ixy1); /* ixy1 = ixy, iz1 */ + glm__noiseDetail_permute(ixy1, ixy1); /* ixy1 = permute(ixy1) */ + + /* ixy00 = permute(ixy0 + iw0) */ + vec4 ixy00; + glm_vec4_add(ixy0, iw0, ixy00); /* ixy00 = ixy0 + iw0 */ + glm__noiseDetail_permute(ixy00, ixy00); /* ixy00 = permute(ixy00) */ + + /* ixy01 = permute(ixy0 + iw1) */ + vec4 ixy01; + glm_vec4_add(ixy0, iw1, ixy01); /* ixy01 = ixy0 + iw1 */ + glm__noiseDetail_permute(ixy01, ixy01); /* ixy01 = permute(ixy01) */ + + /* ixy10 = permute(ixy1 + iw0) */ + vec4 ixy10; + glm_vec4_add(ixy1, iw0, ixy10); /* ixy10 = ixy1 + iw0 */ + glm__noiseDetail_permute(ixy10, ixy10); /* ixy10 = permute(ixy10) */ + + /* ixy11 = permute(ixy1 + iw1) */ + vec4 ixy11; + glm_vec4_add(ixy1, iw1, ixy11); /* ixy11 = ixy1 + iw1 */ + glm__noiseDetail_permute(ixy11, ixy11); /* ixy11 = permute(ixy11) */ + + /* ------------ */ + + vec4 gx00, gy00, gz00, gw00; + glm__noiseDetail_i2gxyzw(ixy00, gx00, gy00, gz00, gw00); + + vec4 gx01, gy01, gz01, gw01; + glm__noiseDetail_i2gxyzw(ixy01, gx01, gy01, gz01, gw01); + + vec4 gx10, gy10, gz10, gw10; + glm__noiseDetail_i2gxyzw(ixy10, gx10, gy10, gz10, gw10); + + vec4 gx11, gy11, gz11, gw11; + glm__noiseDetail_i2gxyzw(ixy11, gx11, gy11, gz11, gw11); + + /* ------------ */ + + vec4 g0000 = {gx00[0], gy00[0], gz00[0], gw00[0]}; /* g0000 = vec4(gx00.x, gy00.x, gz00.x, gw00.x); */ + vec4 g0100 = {gx00[2], gy00[2], gz00[2], gw00[2]}; /* g0100 = vec4(gx00.z, gy00.z, gz00.z, gw00.z); */ + vec4 g1000 = {gx00[1], gy00[1], gz00[1], gw00[1]}; /* g1000 = vec4(gx00.y, gy00.y, gz00.y, gw00.y); */ + vec4 g1100 = {gx00[3], gy00[3], gz00[3], gw00[3]}; /* g1100 = vec4(gx00.w, gy00.w, gz00.w, gw00.w); */ + + vec4 g0001 = {gx01[0], gy01[0], gz01[0], gw01[0]}; /* g0001 = vec4(gx01.x, gy01.x, gz01.x, gw01.x); */ + vec4 g0101 = {gx01[2], gy01[2], gz01[2], gw01[2]}; /* g0101 = vec4(gx01.z, gy01.z, gz01.z, gw01.z); */ + vec4 g1001 = {gx01[1], gy01[1], gz01[1], gw01[1]}; /* g1001 = vec4(gx01.y, gy01.y, gz01.y, gw01.y); */ + vec4 g1101 = {gx01[3], gy01[3], gz01[3], gw01[3]}; /* g1101 = vec4(gx01.w, gy01.w, gz01.w, gw01.w); */ + + vec4 g0010 = {gx10[0], gy10[0], gz10[0], gw10[0]}; /* g0010 = vec4(gx10.x, gy10.x, gz10.x, gw10.x); */ + vec4 g0110 = {gx10[2], gy10[2], gz10[2], gw10[2]}; /* g0110 = vec4(gx10.z, gy10.z, gz10.z, gw10.z); */ + vec4 g1010 = {gx10[1], gy10[1], gz10[1], gw10[1]}; /* g1010 = vec4(gx10.y, gy10.y, gz10.y, gw10.y); */ + vec4 g1110 = {gx10[3], gy10[3], gz10[3], gw10[3]}; /* g1110 = vec4(gx10.w, gy10.w, gz10.w, gw10.w); */ + + vec4 g0011 = {gx11[0], gy11[0], gz11[0], gw11[0]}; /* g0011 = vec4(gx11.x, gy11.x, gz11.x, gw11.x); */ + vec4 g0111 = {gx11[2], gy11[2], gz11[2], gw11[2]}; /* g0111 = vec4(gx11.z, gy11.z, gz11.z, gw11.z); */ + vec4 g1011 = {gx11[1], gy11[1], gz11[1], gw11[1]}; /* g1011 = vec4(gx11.y, gy11.y, gz11.y, gw11.y); */ + vec4 g1111 = {gx11[3], gy11[3], gz11[3], gw11[3]}; /* g1111 = vec4(gx11.w, gy11.w, gz11.w, gw11.w); */ + + glm__noiseDetail_gradNorm_vec4(g0000, g0100, g1000, g1100); + glm__noiseDetail_gradNorm_vec4(g0001, g0101, g1001, g1101); + glm__noiseDetail_gradNorm_vec4(g0010, g0110, g1010, g1110); + glm__noiseDetail_gradNorm_vec4(g0011, g0111, g1011, g1111); + + /* ------------ */ + + float n0000 = glm_vec4_dot(g0000, Pf0); /* n0000 = dot(g0000, Pf0) */ + + /* n1000 = dot(g1000, vec4(Pf1.x, Pf0.y, Pf0.z, Pf0.w)) */ + vec4 n1000d = {Pf1[0], Pf0[1], Pf0[2], Pf0[3]}; + float n1000 = glm_vec4_dot(g1000, n1000d); + + /* n0100 = dot(g0100, vec4(Pf0.x, Pf1.y, Pf0.z, Pf0.w)) */ + vec4 n0100d = {Pf0[0], Pf1[1], Pf0[2], Pf0[3]}; + float n0100 = glm_vec4_dot(g0100, n0100d); + + /* n1100 = dot(g1100, vec4(Pf1.x, Pf1.y, Pf0.z, Pf0.w)) */ + vec4 n1100d = {Pf1[0], Pf1[1], Pf0[2], Pf0[3]}; + float n1100 = glm_vec4_dot(g1100, n1100d); + + /* n0010 = dot(g0010, vec4(Pf0.x, Pf0.y, Pf1.z, Pf0.w)) */ + vec4 n0010d = {Pf0[0], Pf0[1], Pf1[2], Pf0[3]}; + float n0010 = glm_vec4_dot(g0010, n0010d); + + /* n1010 = dot(g1010, vec4(Pf1.x, Pf0.y, Pf1.z, Pf0.w)) */ + vec4 n1010d = {Pf1[0], Pf0[1], Pf1[2], Pf0[3]}; + float n1010 = glm_vec4_dot(g1010, n1010d); + + /* n0110 = dot(g0110, vec4(Pf0.x, Pf1.y, Pf1.z, Pf0.w)) */ + vec4 n0110d = {Pf0[0], Pf1[1], Pf1[2], Pf0[3]}; + float n0110 = glm_vec4_dot(g0110, n0110d); + + /* n1110 = dot(g1110, vec4(Pf1.x, Pf1.y, Pf1.z, Pf0.w)) */ + vec4 n1110d = {Pf1[0], Pf1[1], Pf1[2], Pf0[3]}; + float n1110 = glm_vec4_dot(g1110, n1110d); + + /* n0001 = dot(g0001, vec4(Pf0.x, Pf0.y, Pf0.z, Pf1.w)) */ + vec4 n0001d = {Pf0[0], Pf0[1], Pf0[2], Pf1[3]}; + float n0001 = glm_vec4_dot(g0001, n0001d); + + /* n1001 = dot(g1001, vec4(Pf1.x, Pf0.y, Pf0.z, Pf1.w)) */ + vec4 n1001d = {Pf1[0], Pf0[1], Pf0[2], Pf1[3]}; + float n1001 = glm_vec4_dot(g1001, n1001d); + + /* n0101 = dot(g0101, vec4(Pf0.x, Pf1.y, Pf0.z, Pf1.w)) */ + vec4 n0101d = {Pf0[0], Pf1[1], Pf0[2], Pf1[3]}; + float n0101 = glm_vec4_dot(g0101, n0101d); + + /* n1101 = dot(g1101, vec4(Pf1.x, Pf1.y, Pf0.z, Pf1.w)) */ + vec4 n1101d = {Pf1[0], Pf1[1], Pf0[2], Pf1[3]}; + float n1101 = glm_vec4_dot(g1101, n1101d); + + /* n0011 = dot(g0011, vec4(Pf0.x, Pf0.y, Pf1.z, Pf1.w)) */ + vec4 n0011d = {Pf0[0], Pf0[1], Pf1[2], Pf1[3]}; + float n0011 = glm_vec4_dot(g0011, n0011d); + + /* n1011 = dot(g1011, vec4(Pf1.x, Pf0.y, Pf1.z, Pf1.w)) */ + vec4 n1011d = {Pf1[0], Pf0[1], Pf1[2], Pf1[3]}; + float n1011 = glm_vec4_dot(g1011, n1011d); + + /* n0111 = dot(g0111, vec4(Pf0.x, Pf1.y, Pf1.z, Pf1.w)) */ + vec4 n0111d = {Pf0[0], Pf1[1], Pf1[2], Pf1[3]}; + float n0111 = glm_vec4_dot(g0111, n0111d); + + float n1111 = glm_vec4_dot(g1111, Pf1); /* n1111 = dot(g1111, Pf1) */ + + /* ------------ */ + + vec4 fade_xyzw; + glm__noiseDetail_fade_vec4(Pf0, fade_xyzw); /* fade_xyzw = fade(Pf0) */ + + /* n_0w = lerp(vec4(n0000, n1000, n0100, n1100), vec4(n0001, n1001, n0101, n1101), fade_xyzw.w) */ + vec4 n_0w1 = {n0000, n1000, n0100, n1100}; + vec4 n_0w2 = {n0001, n1001, n0101, n1101}; + vec4 n_0w; + glm_vec4_lerp(n_0w1, n_0w2, fade_xyzw[3], n_0w); + + /* n_1w = lerp(vec4(n0010, n1010, n0110, n1110), vec4(n0011, n1011, n0111, n1111), fade_xyzw.w) */ + vec4 n_1w1 = {n0010, n1010, n0110, n1110}; + vec4 n_1w2 = {n0011, n1011, n0111, n1111}; + vec4 n_1w; + glm_vec4_lerp(n_1w1, n_1w2, fade_xyzw[3], n_1w); + + /* n_zw = lerp(n_0w, n_1w, fade_xyzw.z) */ + vec4 n_zw; + glm_vec4_lerp(n_0w, n_1w, fade_xyzw[2], n_zw); + + /* n_yzw = lerp(vec2(n_zw.x, n_zw.y), vec2(n_zw.z, n_zw.w), fade_xyzw.y) */ + vec2 n_yzw; + vec2 n_yzw1 = {n_zw[0], n_zw[1]}; + vec2 n_yzw2 = {n_zw[2], n_zw[3]}; + glm_vec2_lerp(n_yzw1, n_yzw2, fade_xyzw[1], n_yzw); + + /* n_xyzw = lerp(n_yzw.x, n_yzw.y, fade_xyzw.x) */ + float n_xyzw = glm_lerp(n_yzw[0], n_yzw[1], fade_xyzw[0]); + + return n_xyzw * 2.2f; +} + + +/*! + * @brief Classic perlin noise + * + * @param[in] point 3D vector + * @returns perlin noise value + */ +CGLM_INLINE +float +glm_perlin_vec3(vec3 point) { + /* Integer part of p for indexing */ + vec3 Pi0; + glm_vec3_floor(point, Pi0); /* Pi0 = floor(point); */ + + /* Integer part + 1 */ + vec3 Pi1; + glm_vec3_adds(Pi0, 1.0f, Pi1); /* Pi1 = Pi0 + 1.0f; */ + + glm_vec3_mods(Pi0, 289.0f, Pi0); /* Pi0 = mod(Pi0, 289.0f); */ + glm_vec3_mods(Pi1, 289.0f, Pi1); /* Pi1 = mod(Pi1, 289.0f); */ + + /* Fractional part of p for interpolation */ + vec3 Pf0; + glm_vec3_fract(point, Pf0); + + /* Fractional part - 1.0 */ + vec3 Pf1; + glm_vec3_subs(Pf0, 1.0f, Pf1); + + vec4 ix = {Pi0[0], Pi1[0], Pi0[0], Pi1[0]}; + vec4 iy = {Pi0[1], Pi0[1], Pi1[1], Pi1[1]}; + vec4 iz0 = {Pi0[2], Pi0[2], Pi0[2], Pi0[2]}; /* iz0 = vec4(Pi0.z); */ + vec4 iz1 = {Pi1[2], Pi1[2], Pi1[2], Pi1[2]}; /* iz1 = vec4(Pi1.z); */ + + /* ------------ */ + + /* ixy = permute(permute(ix) + iy) */ + vec4 ixy; + glm__noiseDetail_permute(ix, ixy); /* ixy = permute(ix) */ + glm_vec4_add(ixy, iy, ixy); /* ixy += iy; */ + glm__noiseDetail_permute(ixy, ixy); /* ixy = permute(ixy) */ + + /* ixy0 = permute(ixy + iz0) */ + vec4 ixy0; + glm_vec4_add(ixy, iz0, ixy0); /* ixy0 = ixy + iz0 */ + glm__noiseDetail_permute(ixy0, ixy0); /* ixy0 = permute(ixy0) */ + + /* ixy1 = permute(ixy + iz1) */ + vec4 ixy1; + glm_vec4_add(ixy, iz1, ixy1); /* ixy1 = ixy, iz1 */ + glm__noiseDetail_permute(ixy1, ixy1); /* ixy1 = permute(ixy1) */ + + /* ------------ */ + + vec4 gx0, gy0, gz0; + glm__noiseDetail_i2gxyz(ixy0, gx0, gy0, gz0); + + vec4 gx1, gy1, gz1; + glm__noiseDetail_i2gxyz(ixy1, gx1, gy1, gz1); + + /* ------------ */ + + vec3 g000 = {gx0[0], gy0[0], gz0[0]}; /* g000 = vec3(gx0.x, gy0.x, gz0.x); */ + vec3 g100 = {gx0[1], gy0[1], gz0[1]}; /* g100 = vec3(gx0.y, gy0.y, gz0.y); */ + vec3 g010 = {gx0[2], gy0[2], gz0[2]}; /* g010 = vec3(gx0.z, gy0.z, gz0.z); */ + vec3 g110 = {gx0[3], gy0[3], gz0[3]}; /* g110 = vec3(gx0.w, gy0.w, gz0.w); */ + + vec3 g001 = {gx1[0], gy1[0], gz1[0]}; /* g001 = vec3(gx1.x, gy1.x, gz1.x); */ + vec3 g101 = {gx1[1], gy1[1], gz1[1]}; /* g101 = vec3(gx1.y, gy1.y, gz1.y); */ + vec3 g011 = {gx1[2], gy1[2], gz1[2]}; /* g011 = vec3(gx1.z, gy1.z, gz1.z); */ + vec3 g111 = {gx1[3], gy1[3], gz1[3]}; /* g111 = vec3(gx1.w, gy1.w, gz1.w); */ + + glm__noiseDetail_gradNorm_vec3(g000, g010, g100, g110); + glm__noiseDetail_gradNorm_vec3(g001, g011, g101, g111); + + /* ------------ */ + + float n000 = glm_vec3_dot(g000, Pf0); /* n000 = dot(g000, Pf0) */ + + /* n100 = dot(g100, vec3(Pf1.x, Pf0.y, Pf0.z)) */ + vec3 n100d = {Pf1[0], Pf0[1], Pf0[2]}; + float n100 = glm_vec3_dot(g100, n100d); + + /* n010 = dot(g010, vec3(Pf0.x, Pf1.y, Pf0.z)) */ + vec3 n010d = {Pf0[0], Pf1[1], Pf0[2]}; + float n010 = glm_vec3_dot(g010, n010d); + + /* n110 = dot(g110, vec3(Pf1.x, Pf1.y, Pf0.z)) */ + vec3 n110d = {Pf1[0], Pf1[1], Pf0[2]}; + float n110 = glm_vec3_dot(g110, n110d); + + /* n001 = dot(g001, vec3(Pf0.x, Pf0.y, Pf1.z)) */ + vec3 n001d = {Pf0[0], Pf0[1], Pf1[2]}; + float n001 = glm_vec3_dot(g001, n001d); + + /* n101 = dot(g101, vec3(Pf1.x, Pf0.y, Pf1.z)) */ + vec3 n101d = {Pf1[0], Pf0[1], Pf1[2]}; + float n101 = glm_vec3_dot(g101, n101d); + + /* n011 = dot(g011, vec3(Pf0.x, Pf1.y, Pf1.z)) */ + vec3 n011d = {Pf0[0], Pf1[1], Pf1[2]}; + float n011 = glm_vec3_dot(g011, n011d); + + float n111 = glm_vec3_dot(g111, Pf1); /* n111 = dot(g111, Pf1) */ + + /* ------------ */ + + vec3 fade_xyz; + glm__noiseDetail_fade_vec3(Pf0, fade_xyz); /* fade_xyz = fade(Pf0) */ + + /* n_z = lerp(vec4(n000, n100, n010, n110), vec4(n001, n101, n011, n111), fade_xyz.z); */ + vec4 n_z; + vec4 n_z1 = {n000, n100, n010, n110}; + vec4 n_z2 = {n001, n101, n011, n111}; + glm_vec4_lerp(n_z1, n_z2, fade_xyz[2], n_z); + + /* vec2 n_yz = lerp(vec2(n_z.x, n_z.y), vec2(n_z.z, n_z.w), fade_xyz.y); */ + vec2 n_yz; + vec2 n_yz1 = {n_z[0], n_z[1]}; + vec2 n_yz2 = {n_z[2], n_z[3]}; + glm_vec2_lerp(n_yz1, n_yz2, fade_xyz[1], n_yz); + + /* n_xyz = lerp(n_yz.x, n_yz.y, fade_xyz.x); */ + float n_xyz = glm_lerp(n_yz[0], n_yz[1], fade_xyz[0]); + + return n_xyz * 2.2f; +} + +/*! + * @brief Classic perlin noise + * + * @param[in] point 2D vector + * @returns perlin noise value + */ +CGLM_INLINE +float +glm_perlin_vec2(vec2 point) { + + /* Integer part of p for indexing */ + /* Pi = floor(vec4(point.x, point.y, point.x, point.y)) + vec4(0.0, 0.0, 1.0, 1.0); */ + vec4 Pi = {point[0], point[1], point[0], point[1]}; /* Pi = vec4(point.x, point.y, point.x, point.y) */ + glm_vec4_floor(Pi, Pi); /* Pi = floor(Pi) */ + Pi[2] += 1.0f; /* Pi.z += 1.0 */ + Pi[3] += 1.0f; /* Pi.w += 1.0 */ + + /* Fractional part of p for interpolation */ + /* vec<4, T, Q> Pf = glm::fract(vec<4, T, Q>(Position.x, Position.y, Position.x, Position.y)) - vec<4, T, Q>(0.0, 0.0, 1.0, 1.0); */ + vec4 Pf = {point[0], point[1], point[0], point[1]}; /* Pf = vec4(point.x, point.y, point.x, point.y) */ + glm_vec4_fract(Pf, Pf); /* Pf = fract(Pf) */ + Pf[2] -= 1.0f; /* Pf.z -= 1.0 */ + Pf[3] -= 1.0f; /* Pf.w -= 1.0 */ + + /* Mod to avoid truncation effects in permutation */ + glm_vec4_mods(Pi, 289.0f, Pi); /* Pi = mod(Pi, 289.0f); */ + + vec4 ix = {Pi[0], Pi[2], Pi[0], Pi[2]}; /* ix = vec4(Pi.x, Pi.z, Pi.x, Pi.z) */ + vec4 iy = {Pi[1], Pi[1], Pi[3], Pi[3]}; /* iy = vec4(Pi.y, Pi.y, Pi.w, Pi.w) */ + vec4 fx = {Pf[0], Pf[2], Pf[0], Pf[2]}; /* fx = vec4(Pf.x, Pf.z, Pf.x, Pf.z) */ + vec4 fy = {Pf[1], Pf[1], Pf[3], Pf[3]}; /* fy = vec4(Pf.y, Pf.y, Pf.w, Pf.w) */ + + /* ------------ */ + + /* i = permute(permute(ix) + iy); */ + vec4 i; + glm__noiseDetail_permute(ix, i); /* i = permute(ix) */ + glm_vec4_add(i, iy, i); /* i += iy; */ + glm__noiseDetail_permute(i, i); /* i = permute(i) */ + + /* ------------ */ + + vec4 gx, gy; + glm__noiseDetail_i2gxy(i, gx, gy); + + /* ------------ */ + + vec2 g00 = {gx[0], gy[0]}; /* g00 = vec2(gx.x, gy.x) */ + vec2 g10 = {gx[1], gy[1]}; /* g10 = vec2(gx.y, gy.y) */ + vec2 g01 = {gx[2], gy[2]}; /* g01 = vec2(gx.z, gy.z) */ + vec2 g11 = {gx[3], gy[3]}; /* g11 = vec2(gx.w, gy.w) */ + + glm__noiseDetail_gradNorm_vec2(g00, g01, g10, g11); + + /* ------------ */ + + /* n00 = dot(g00, vec2(fx.x, fy.x)) */ + vec2 n00d = {fx[0], fy[0]}; /* n00d = vec2(fx.x, fy.x) */ + float n00 = glm_vec2_dot(g00, n00d); /* n00 = dot(g00, n00d) */ + + /* n10 = dot(g10, vec2(fx.y, fy.y)) */ + vec2 n10d = {fx[1], fy[1]}; /* n10d = vec2(fx.y, fy.y) */ + float n10 = glm_vec2_dot(g10, n10d); /* n10 = dot(g10, n10d) */ + + /* n01 = dot(g01, vec2(fx.z, fy.z)) */ + vec2 n01d = {fx[2], fy[2]}; /* n01d = vec2(fx.z, fy.z) */ + float n01 = glm_vec2_dot(g01, n01d); /* n01 = dot(g01, n01d) */ + + /* n11 = dot(g11, vec2(fx.w, fy.w)) */ + vec2 n11d = {fx[3], fy[3]}; /* n11d = vec2(fx.w, fy.w) */ + float n11 = glm_vec2_dot(g11, n11d); /* n11 = dot(g11, n11d) */ + + /* ------------ */ + + /* fade_xyz = fade(vec2(Pf.x, Pf.y)) */ + vec2 fade_xy; + vec2 temp2 = {Pf[0], Pf[1]}; /* temp = vec2(Pf.x, Pf.y) */ + glm__noiseDetail_fade_vec2(temp2, fade_xy); /* fade_xy = fade(temp) */ + + /* n_x = lerp(vec2(n00, n01), vec2(n10, n11), fade_xy.x); */ + vec2 n_x; + vec2 n_x1 = {n00, n01}; /* n_x1 = vec2(n00, n01) */ + vec2 n_x2 = {n10, n11}; /* n_x2 = vec2(n10, n11) */ + glm_vec2_lerp(n_x1, n_x2, fade_xy[0], n_x); /* n_x = lerp(n_x1, n_x2, fade_xy.x) */ + + /* T n_xy = mix(n_x.x, n_x.y, fade_xy.y); */ + /* n_xy = lerp(n_x.x, n_x.y, fade_xy.y); */ + float n_xy = glm_lerp(n_x[0], n_x[1], fade_xy[1]); + + return n_xy * 2.3f; +} + +/* Undefine all helper macros */ + +#undef glm__noiseDetail_mod289 +#undef glm__noiseDetail_permute +#undef glm__noiseDetail_fade_vec4 +#undef glm__noiseDetail_fade_vec3 +#undef glm__noiseDetail_fade_vec2 +#undef glm__noiseDetail_taylorInvSqrt +#undef glm__noiseDetail_gradNorm_vec4 +#undef glm__noiseDetail_gradNorm_vec3 +#undef glm__noiseDetail_gradNorm_vec2 +#undef glm__noiseDetail_i2gxyzw +#undef glm__noiseDetail_i2gxyz +#undef glm__noiseDetail_i2gxy + +#endif /* cglm_noise_h */ diff --git a/cglm/include/cglm/plane.h b/cglm/include/cglm/plane.h new file mode 100644 index 0000000..9efabb7 --- /dev/null +++ b/cglm/include/cglm/plane.h @@ -0,0 +1,44 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_plane_h +#define cglm_plane_h + +#include "common.h" +#include "vec3.h" +#include "vec4.h" + +/* + Plane equation: Ax + By + Cz + D = 0; + + It stored in vec4 as [A, B, C, D]. (A, B, C) is normal and D is distance +*/ + +/* + Functions: + CGLM_INLINE void glm_plane_normalize(vec4 plane); + */ + +/*! + * @brief normalizes a plane + * + * @param[in, out] plane plane to normalize + */ +CGLM_INLINE +void +glm_plane_normalize(vec4 plane) { + float norm; + + if (CGLM_UNLIKELY((norm = glm_vec3_norm(plane)) < FLT_EPSILON)) { + glm_vec4_zero(plane); + return; + } + + glm_vec4_scale(plane, 1.0f / norm, plane); +} + +#endif /* cglm_plane_h */ diff --git a/cglm/include/cglm/project.h b/cglm/include/cglm/project.h new file mode 100644 index 0000000..1d0a4e5 --- /dev/null +++ b/cglm/include/cglm/project.h @@ -0,0 +1,172 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_project_h +#define cglm_project_h + +#include "common.h" +#include "vec3.h" +#include "vec4.h" +#include "mat4.h" + +#ifndef CGLM_CLIPSPACE_INCLUDE_ALL +# if CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_ZO_BIT +# include "clipspace/project_zo.h" +# elif CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_NO_BIT +# include "clipspace/project_no.h" +# endif +#else +# include "clipspace/project_zo.h" +# include "clipspace/project_no.h" +#endif + +/*! + * @brief maps the specified viewport coordinates into specified space [1] + * the matrix should contain projection matrix. + * + * if you don't have ( and don't want to have ) an inverse matrix then use + * glm_unproject version. You may use existing inverse of matrix in somewhere + * else, this is why glm_unprojecti exists to save save inversion cost + * + * [1] space: + * 1- if m = invProj: View Space + * 2- if m = invViewProj: World Space + * 3- if m = invMVP: Object Space + * + * You probably want to map the coordinates into object space + * so use invMVP as m + * + * Computing viewProj: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * glm_mat4_inv(viewProj, invMVP); + * + * @param[in] pos point/position in viewport coordinates + * @param[in] invMat matrix (see brief) + * @param[in] vp viewport as [x, y, width, height] + * @param[out] dest unprojected coordinates + */ +CGLM_INLINE +void +glm_unprojecti(vec3 pos, mat4 invMat, vec4 vp, vec3 dest) { +#if CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_ZO_BIT + glm_unprojecti_zo(pos, invMat, vp, dest); +#elif CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_NO_BIT + glm_unprojecti_no(pos, invMat, vp, dest); +#endif +} + +/*! + * @brief maps the specified viewport coordinates into specified space [1] + * the matrix should contain projection matrix. + * + * this is same as glm_unprojecti except this function get inverse matrix for + * you. + * + * [1] space: + * 1- if m = proj: View Space + * 2- if m = viewProj: World Space + * 3- if m = MVP: Object Space + * + * You probably want to map the coordinates into object space + * so use MVP as m + * + * Computing viewProj and MVP: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * + * @param[in] pos point/position in viewport coordinates + * @param[in] m matrix (see brief) + * @param[in] vp viewport as [x, y, width, height] + * @param[out] dest unprojected coordinates + */ +CGLM_INLINE +void +glm_unproject(vec3 pos, mat4 m, vec4 vp, vec3 dest) { + mat4 inv; + glm_mat4_inv(m, inv); + glm_unprojecti(pos, inv, vp, dest); +} + +/*! + * @brief map object coordinates to window coordinates + * + * Computing MVP: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * + * @param[in] pos object coordinates + * @param[in] m MVP matrix + * @param[in] vp viewport as [x, y, width, height] + * @param[out] dest projected coordinates + */ +CGLM_INLINE +void +glm_project(vec3 pos, mat4 m, vec4 vp, vec3 dest) { +#if CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_ZO_BIT + glm_project_zo(pos, m, vp, dest); +#elif CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_NO_BIT + glm_project_no(pos, m, vp, dest); +#endif +} + +/*! + * @brief map object's z coordinate to window coordinates + * + * Computing MVP: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * + * @param[in] v object coordinates + * @param[in] m MVP matrix + * + * @returns projected z coordinate + */ +CGLM_INLINE +float +glm_project_z(vec3 v, mat4 m) { +#if CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_ZO_BIT + return glm_project_z_zo(v, m); +#elif CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_NO_BIT + return glm_project_z_no(v, m); +#endif +} + +/*! + * @brief define a picking region + * + * @param[in] center center [x, y] of a picking region in window coordinates + * @param[in] size size [width, height] of the picking region in window coordinates + * @param[in] vp viewport as [x, y, width, height] + * @param[out] dest projected coordinates + */ +CGLM_INLINE +void +glm_pickmatrix(vec2 center, vec2 size, vec4 vp, mat4 dest) { + mat4 res; + vec3 v; + + if (size[0] <= 0.0f || size[1] <= 0.0f) + return; + + /* Translate and scale the picked region to the entire window */ + v[0] = (vp[2] - 2.0f * (center[0] - vp[0])) / size[0]; + v[1] = (vp[3] - 2.0f * (center[1] - vp[1])) / size[1]; + v[2] = 0.0f; + + glm_translate_make(res, v); + + v[0] = vp[2] / size[0]; + v[1] = vp[3] / size[1]; + v[2] = 1.0f; + + glm_scale(res, v); + + glm_mat4_copy(res, dest); +} + +#endif /* cglm_project_h */ diff --git a/cglm/include/cglm/quat.h b/cglm/include/cglm/quat.h new file mode 100644 index 0000000..cf1f325 --- /dev/null +++ b/cglm/include/cglm/quat.h @@ -0,0 +1,949 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_QUAT_IDENTITY_INIT + GLM_QUAT_IDENTITY + + Functions: + CGLM_INLINE void glm_quat_identity(versor q); + CGLM_INLINE void glm_quat_init(versor q, float x, float y, float z, float w); + CGLM_INLINE void glm_quat(versor q, float angle, float x, float y, float z); + CGLM_INLINE void glm_quatv(versor q, float angle, vec3 axis); + CGLM_INLINE void glm_quat_copy(versor q, versor dest); + CGLM_INLINE void glm_quat_from_vecs(vec3 a, vec3 b, versor dest); + CGLM_INLINE float glm_quat_norm(versor q); + CGLM_INLINE void glm_quat_normalize(versor q); + CGLM_INLINE void glm_quat_normalize_to(versor q, versor dest); + CGLM_INLINE float glm_quat_dot(versor p, versor q); + CGLM_INLINE void glm_quat_conjugate(versor q, versor dest); + CGLM_INLINE void glm_quat_inv(versor q, versor dest); + CGLM_INLINE void glm_quat_add(versor p, versor q, versor dest); + CGLM_INLINE void glm_quat_sub(versor p, versor q, versor dest); + CGLM_INLINE float glm_quat_real(versor q); + CGLM_INLINE void glm_quat_imag(versor q, vec3 dest); + CGLM_INLINE void glm_quat_imagn(versor q, vec3 dest); + CGLM_INLINE float glm_quat_imaglen(versor q); + CGLM_INLINE float glm_quat_angle(versor q); + CGLM_INLINE void glm_quat_axis(versor q, vec3 dest); + CGLM_INLINE void glm_quat_mul(versor p, versor q, versor dest); + CGLM_INLINE void glm_quat_mat4(versor q, mat4 dest); + CGLM_INLINE void glm_quat_mat4t(versor q, mat4 dest); + CGLM_INLINE void glm_quat_mat3(versor q, mat3 dest); + CGLM_INLINE void glm_quat_mat3t(versor q, mat3 dest); + CGLM_INLINE void glm_quat_lerp(versor from, versor to, float t, versor dest); + CGLM_INLINE void glm_quat_lerpc(versor from, versor to, float t, versor dest); + CGLM_INLINE void glm_quat_slerp(versor q, versor r, float t, versor dest); + CGLM_INLINE void glm_quat_slerp_longest(versor q, versor r, float t, versor dest); + CGLM_INLINE void glm_quat_nlerp(versor q, versor r, float t, versor dest); + CGLM_INLINE void glm_quat_look(vec3 eye, versor ori, mat4 dest); + CGLM_INLINE void glm_quat_for(vec3 dir, vec3 fwd, vec3 up, versor dest); + CGLM_INLINE void glm_quat_forp(vec3 from, + vec3 to, + vec3 fwd, + vec3 up, + versor dest); + CGLM_INLINE void glm_quat_rotatev(versor q, vec3 v, vec3 dest); + CGLM_INLINE void glm_quat_rotate(mat4 m, versor q, mat4 dest); + CGLM_INLINE void glm_quat_make(float * restrict src, versor dest); + */ + +#ifndef cglm_quat_h +#define cglm_quat_h + +#include "common.h" +#include "vec3.h" +#include "vec4.h" +#include "mat4.h" +#include "mat3.h" +#include "affine-mat.h" +#include "affine.h" + +#ifdef CGLM_SSE_FP +# include "simd/sse2/quat.h" +#endif + +#ifdef CGLM_NEON_FP +# include "simd/neon/quat.h" +#endif + +#ifdef CGLM_SIMD_WASM +# include "simd/wasm/quat.h" +#endif + +CGLM_INLINE void glm_quat_normalize(versor q); + +/* + * IMPORTANT: + * ---------------------------------------------------------------------------- + * cglm stores quat as [x, y, z, w] since v0.3.6 + * + * it was [w, x, y, z] before v0.3.6 it has been changed to [x, y, z, w] + * with v0.3.6 version. + * ---------------------------------------------------------------------------- + */ + +#define GLM_QUAT_IDENTITY_INIT {0.0f, 0.0f, 0.0f, 1.0f} +#define GLM_QUAT_IDENTITY ((versor)GLM_QUAT_IDENTITY_INIT) + +/*! + * @brief makes given quat to identity + * + * @param[in, out] q quaternion + */ +CGLM_INLINE +void +glm_quat_identity(versor q) { + CGLM_ALIGN(16) versor v = GLM_QUAT_IDENTITY_INIT; + glm_vec4_copy(v, q); +} + +/*! + * @brief make given quaternion array's each element identity quaternion + * + * @param[in, out] q quat array (must be aligned (16) + * if alignment is not disabled) + * + * @param[in] count count of quaternions + */ +CGLM_INLINE +void +glm_quat_identity_array(versor * __restrict q, size_t count) { + CGLM_ALIGN(16) versor v = GLM_QUAT_IDENTITY_INIT; + size_t i; + + for (i = 0; i < count; i++) { + glm_vec4_copy(v, q[i]); + } +} + +/*! + * @brief inits quaternion with raw values + * + * @param[out] q quaternion + * @param[in] x x + * @param[in] y y + * @param[in] z z + * @param[in] w w (real part) + */ +CGLM_INLINE +void +glm_quat_init(versor q, float x, float y, float z, float w) { + q[0] = x; + q[1] = y; + q[2] = z; + q[3] = w; +} + +/*! + * @brief creates NEW quaternion with axis vector + * + * @param[out] q quaternion + * @param[in] angle angle (radians) + * @param[in] axis axis + */ +CGLM_INLINE +void +glm_quatv(versor q, float angle, vec3 axis) { + CGLM_ALIGN(8) vec3 k; + float a, c, s; + + a = angle * 0.5f; + c = cosf(a); + s = sinf(a); + + glm_normalize_to(axis, k); + + q[0] = s * k[0]; + q[1] = s * k[1]; + q[2] = s * k[2]; + q[3] = c; +} + +/*! + * @brief creates NEW quaternion with individual axis components + * + * @param[out] q quaternion + * @param[in] angle angle (radians) + * @param[in] x axis.x + * @param[in] y axis.y + * @param[in] z axis.z + */ +CGLM_INLINE +void +glm_quat(versor q, float angle, float x, float y, float z) { + CGLM_ALIGN(8) vec3 axis = {x, y, z}; + glm_quatv(q, angle, axis); +} + +/*! + * @brief copy quaternion to another one + * + * @param[in] q quaternion + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_quat_copy(versor q, versor dest) { + glm_vec4_copy(q, dest); +} + +/*! + * @brief compute quaternion rotating vector A to vector B + * + * @param[in] a vec3 (must have unit length) + * @param[in] b vec3 (must have unit length) + * @param[out] dest quaternion (of unit length) + */ +CGLM_INLINE +void +glm_quat_from_vecs(vec3 a, vec3 b, versor dest) { + CGLM_ALIGN(8) vec3 axis; + float cos_theta; + float cos_half_theta; + + cos_theta = glm_vec3_dot(a, b); + if (cos_theta >= 1.f - GLM_FLT_EPSILON) { /* a ∥ b */ + glm_quat_identity(dest); + return; + } + if (cos_theta < -1.f + GLM_FLT_EPSILON) { /* angle(a, b) = π */ + glm_vec3_ortho(a, axis); + cos_half_theta = 0.f; /* cos π/2 */ + } else { + glm_vec3_cross(a, b, axis); + cos_half_theta = 1.0f + cos_theta; /* cos 0 + cos θ */ + } + + glm_quat_init(dest, axis[0], axis[1], axis[2], cos_half_theta); + glm_quat_normalize(dest); +} + +/*! + * @brief returns norm (magnitude) of quaternion + * + * @param[in] q quaternion + */ +CGLM_INLINE +float +glm_quat_norm(versor q) { + return glm_vec4_norm(q); +} + +/*! + * @brief normalize quaternion and store result in dest + * + * @param[in] q quaternion to normalze + * @param[out] dest destination quaternion + */ +CGLM_INLINE +void +glm_quat_normalize_to(versor q, versor dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_128 xdot, x0; + float dot; + + x0 = glmm_load(q); + xdot = glmm_vdot(x0, x0); + /* dot = _mm_cvtss_f32(xdot); */ + dot = wasm_f32x4_extract_lane(xdot, 0); + + if (dot <= 0.0f) { + glm_quat_identity(dest); + return; + } + + glmm_store(dest, wasm_f32x4_div(x0, wasm_f32x4_sqrt(xdot))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + __m128 xdot, x0; + float dot; + + x0 = glmm_load(q); + xdot = glmm_vdot(x0, x0); + dot = _mm_cvtss_f32(xdot); + + if (dot <= 0.0f) { + glm_quat_identity(dest); + return; + } + + glmm_store(dest, _mm_div_ps(x0, _mm_sqrt_ps(xdot))); +#else + float dot; + + dot = glm_vec4_norm2(q); + + if (dot <= 0.0f) { + glm_quat_identity(dest); + return; + } + + glm_vec4_scale(q, 1.0f / sqrtf(dot), dest); +#endif +} + +/*! + * @brief normalize quaternion + * + * @param[in, out] q quaternion + */ +CGLM_INLINE +void +glm_quat_normalize(versor q) { + glm_quat_normalize_to(q, q); +} + +/*! + * @brief dot product of two quaternion + * + * @param[in] p quaternion 1 + * @param[in] q quaternion 2 + */ +CGLM_INLINE +float +glm_quat_dot(versor p, versor q) { + return glm_vec4_dot(p, q); +} + +/*! + * @brief conjugate of quaternion + * + * @param[in] q quaternion + * @param[out] dest conjugate + */ +CGLM_INLINE +void +glm_quat_conjugate(versor q, versor dest) { + glm_vec4_negate_to(q, dest); + dest[3] = -dest[3]; +} + +/*! + * @brief inverse of non-zero quaternion + * + * @param[in] q quaternion + * @param[out] dest inverse quaternion + */ +CGLM_INLINE +void +glm_quat_inv(versor q, versor dest) { + CGLM_ALIGN(16) versor conj; + glm_quat_conjugate(q, conj); + glm_vec4_scale(conj, 1.0f / glm_vec4_norm2(q), dest); +} + +/*! + * @brief add (componentwise) two quaternions and store result in dest + * + * @param[in] p quaternion 1 + * @param[in] q quaternion 2 + * @param[out] dest result quaternion + */ +CGLM_INLINE +void +glm_quat_add(versor p, versor q, versor dest) { + glm_vec4_add(p, q, dest); +} + +/*! + * @brief subtract (componentwise) two quaternions and store result in dest + * + * @param[in] p quaternion 1 + * @param[in] q quaternion 2 + * @param[out] dest result quaternion + */ +CGLM_INLINE +void +glm_quat_sub(versor p, versor q, versor dest) { + glm_vec4_sub(p, q, dest); +} + +/*! + * @brief returns real part of quaternion + * + * @param[in] q quaternion + */ +CGLM_INLINE +float +glm_quat_real(versor q) { + return q[3]; +} + +/*! + * @brief returns imaginary part of quaternion + * + * @param[in] q quaternion + * @param[out] dest imag + */ +CGLM_INLINE +void +glm_quat_imag(versor q, vec3 dest) { + dest[0] = q[0]; + dest[1] = q[1]; + dest[2] = q[2]; +} + +/*! + * @brief returns normalized imaginary part of quaternion + * + * @param[in] q quaternion + */ +CGLM_INLINE +void +glm_quat_imagn(versor q, vec3 dest) { + glm_normalize_to(q, dest); +} + +/*! + * @brief returns length of imaginary part of quaternion + * + * @param[in] q quaternion + */ +CGLM_INLINE +float +glm_quat_imaglen(versor q) { + return glm_vec3_norm(q); +} + +/*! + * @brief returns angle of quaternion + * + * @param[in] q quaternion + */ +CGLM_INLINE +float +glm_quat_angle(versor q) { + /* + sin(theta / 2) = length(x*x + y*y + z*z) + cos(theta / 2) = w + theta = 2 * atan(sin(theta / 2) / cos(theta / 2)) + */ + return 2.0f * atan2f(glm_quat_imaglen(q), glm_quat_real(q)); +} + +/*! + * @brief axis of quaternion + * + * @param[in] q quaternion + * @param[out] dest axis of quaternion + */ +CGLM_INLINE +void +glm_quat_axis(versor q, vec3 dest) { + glm_quat_imagn(q, dest); +} + +/*! + * @brief multiplies two quaternion and stores result in dest + * this is also called Hamilton Product + * + * According to WikiPedia: + * The product of two rotation quaternions [clarification needed] will be + * equivalent to the rotation q followed by the rotation p + * + * @param[in] p quaternion 1 + * @param[in] q quaternion 2 + * @param[out] dest result quaternion + */ +CGLM_INLINE +void +glm_quat_mul(versor p, versor q, versor dest) { + /* + + (a1 b2 + b1 a2 + c1 d2 − d1 c2)i + + (a1 c2 − b1 d2 + c1 a2 + d1 b2)j + + (a1 d2 + b1 c2 − c1 b2 + d1 a2)k + a1 a2 − b1 b2 − c1 c2 − d1 d2 + */ +#if defined(__wasm__) && defined(__wasm_simd128__) + glm_quat_mul_wasm(p, q, dest); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glm_quat_mul_sse2(p, q, dest); +#elif defined(CGLM_NEON_FP) + glm_quat_mul_neon(p, q, dest); +#else + dest[0] = p[3] * q[0] + p[0] * q[3] + p[1] * q[2] - p[2] * q[1]; + dest[1] = p[3] * q[1] - p[0] * q[2] + p[1] * q[3] + p[2] * q[0]; + dest[2] = p[3] * q[2] + p[0] * q[1] - p[1] * q[0] + p[2] * q[3]; + dest[3] = p[3] * q[3] - p[0] * q[0] - p[1] * q[1] - p[2] * q[2]; +#endif +} + +/*! + * @brief convert quaternion to mat4 + * + * @param[in] q quaternion + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_quat_mat4(versor q, mat4 dest) { + float w, x, y, z, + xx, yy, zz, + xy, yz, xz, + wx, wy, wz, norm, s; + + norm = glm_quat_norm(q); + s = norm > 0.0f ? 2.0f / norm : 0.0f; + + x = q[0]; + y = q[1]; + z = q[2]; + w = q[3]; + + xx = s * x * x; xy = s * x * y; wx = s * w * x; + yy = s * y * y; yz = s * y * z; wy = s * w * y; + zz = s * z * z; xz = s * x * z; wz = s * w * z; + + dest[0][0] = 1.0f - yy - zz; + dest[1][1] = 1.0f - xx - zz; + dest[2][2] = 1.0f - xx - yy; + + dest[0][1] = xy + wz; + dest[1][2] = yz + wx; + dest[2][0] = xz + wy; + + dest[1][0] = xy - wz; + dest[2][1] = yz - wx; + dest[0][2] = xz - wy; + + dest[0][3] = 0.0f; + dest[1][3] = 0.0f; + dest[2][3] = 0.0f; + dest[3][0] = 0.0f; + dest[3][1] = 0.0f; + dest[3][2] = 0.0f; + dest[3][3] = 1.0f; +} + +/*! + * @brief convert quaternion to mat4 (transposed) + * + * @param[in] q quaternion + * @param[out] dest result matrix as transposed + */ +CGLM_INLINE +void +glm_quat_mat4t(versor q, mat4 dest) { + float w, x, y, z, + xx, yy, zz, + xy, yz, xz, + wx, wy, wz, norm, s; + + norm = glm_quat_norm(q); + s = norm > 0.0f ? 2.0f / norm : 0.0f; + + x = q[0]; + y = q[1]; + z = q[2]; + w = q[3]; + + xx = s * x * x; xy = s * x * y; wx = s * w * x; + yy = s * y * y; yz = s * y * z; wy = s * w * y; + zz = s * z * z; xz = s * x * z; wz = s * w * z; + + dest[0][0] = 1.0f - yy - zz; + dest[1][1] = 1.0f - xx - zz; + dest[2][2] = 1.0f - xx - yy; + + dest[1][0] = xy + wz; + dest[2][1] = yz + wx; + dest[0][2] = xz + wy; + + dest[0][1] = xy - wz; + dest[1][2] = yz - wx; + dest[2][0] = xz - wy; + + dest[0][3] = 0.0f; + dest[1][3] = 0.0f; + dest[2][3] = 0.0f; + dest[3][0] = 0.0f; + dest[3][1] = 0.0f; + dest[3][2] = 0.0f; + dest[3][3] = 1.0f; +} + +/*! + * @brief convert quaternion to mat3 + * + * @param[in] q quaternion + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_quat_mat3(versor q, mat3 dest) { + float w, x, y, z, + xx, yy, zz, + xy, yz, xz, + wx, wy, wz, norm, s; + + norm = glm_quat_norm(q); + s = norm > 0.0f ? 2.0f / norm : 0.0f; + + x = q[0]; + y = q[1]; + z = q[2]; + w = q[3]; + + xx = s * x * x; xy = s * x * y; wx = s * w * x; + yy = s * y * y; yz = s * y * z; wy = s * w * y; + zz = s * z * z; xz = s * x * z; wz = s * w * z; + + dest[0][0] = 1.0f - yy - zz; + dest[1][1] = 1.0f - xx - zz; + dest[2][2] = 1.0f - xx - yy; + + dest[0][1] = xy + wz; + dest[1][2] = yz + wx; + dest[2][0] = xz + wy; + + dest[1][0] = xy - wz; + dest[2][1] = yz - wx; + dest[0][2] = xz - wy; +} + +/*! + * @brief convert quaternion to mat3 (transposed) + * + * @param[in] q quaternion + * @param[out] dest result matrix + */ +CGLM_INLINE +void +glm_quat_mat3t(versor q, mat3 dest) { + float w, x, y, z, + xx, yy, zz, + xy, yz, xz, + wx, wy, wz, norm, s; + + norm = glm_quat_norm(q); + s = norm > 0.0f ? 2.0f / norm : 0.0f; + + x = q[0]; + y = q[1]; + z = q[2]; + w = q[3]; + + xx = s * x * x; xy = s * x * y; wx = s * w * x; + yy = s * y * y; yz = s * y * z; wy = s * w * y; + zz = s * z * z; xz = s * x * z; wz = s * w * z; + + dest[0][0] = 1.0f - yy - zz; + dest[1][1] = 1.0f - xx - zz; + dest[2][2] = 1.0f - xx - yy; + + dest[1][0] = xy + wz; + dest[2][1] = yz + wx; + dest[0][2] = xz + wy; + + dest[0][1] = xy - wz; + dest[1][2] = yz - wx; + dest[2][0] = xz - wy; +} + +/*! + * @brief interpolates between two quaternions + * using linear interpolation (LERP) + * + * @param[in] from from + * @param[in] to to + * @param[in] t interpolant (amount) + * @param[out] dest result quaternion + */ +CGLM_INLINE +void +glm_quat_lerp(versor from, versor to, float t, versor dest) { + glm_vec4_lerp(from, to, t, dest); +} + +/*! + * @brief interpolates between two quaternions + * using linear interpolation (LERP) + * + * @param[in] from from + * @param[in] to to + * @param[in] t interpolant (amount) clamped between 0 and 1 + * @param[out] dest result quaternion + */ +CGLM_INLINE +void +glm_quat_lerpc(versor from, versor to, float t, versor dest) { + glm_vec4_lerpc(from, to, t, dest); +} + +/*! + * @brief interpolates between two quaternions + * taking the shortest rotation path using + * normalized linear interpolation (NLERP) + * + * @param[in] from from + * @param[in] to to + * @param[in] t interpolant (amount) + * @param[out] dest result quaternion + */ +CGLM_INLINE +void +glm_quat_nlerp(versor from, versor to, float t, versor dest) { + versor target; + float dot; + + dot = glm_vec4_dot(from, to); + + glm_vec4_scale(to, (dot >= 0) ? 1.0f : -1.0f, target); + glm_quat_lerp(from, target, t, dest); + glm_quat_normalize(dest); +} + +/*! + * @brief interpolates between two quaternions + * using spherical linear interpolation (SLERP) + * + * @param[in] from from + * @param[in] to to + * @param[in] t amount + * @param[out] dest result quaternion + */ +CGLM_INLINE +void +glm_quat_slerp(versor from, versor to, float t, versor dest) { + CGLM_ALIGN(16) vec4 q1, q2; + float cosTheta, sinTheta, angle; + + cosTheta = glm_quat_dot(from, to); + glm_quat_copy(from, q1); + + if (fabsf(cosTheta) >= 1.0f) { + glm_quat_copy(q1, dest); + return; + } + + if (cosTheta < 0.0f) { + glm_vec4_negate(q1); + cosTheta = -cosTheta; + } + + sinTheta = sqrtf(1.0f - cosTheta * cosTheta); + + /* LERP to avoid zero division */ + if (fabsf(sinTheta) < 0.001f) { + glm_quat_lerp(from, to, t, dest); + return; + } + + /* SLERP */ + angle = acosf(cosTheta); + glm_vec4_scale(q1, sinf((1.0f - t) * angle), q1); + glm_vec4_scale(to, sinf(t * angle), q2); + + glm_vec4_add(q1, q2, q1); + glm_vec4_scale(q1, 1.0f / sinTheta, dest); +} + +/*! + * @brief interpolates between two quaternions + * using spherical linear interpolation (SLERP) and always takes the long path + * + * @param[in] from from + * @param[in] to to + * @param[in] t amount + * @param[out] dest result quaternion + */ +CGLM_INLINE +void +glm_quat_slerp_longest(versor from, versor to, float t, versor dest) { + CGLM_ALIGN(16) vec4 q1, q2; + float cosTheta, sinTheta, angle; + + cosTheta = glm_quat_dot(from, to); + glm_quat_copy(from, q1); + + if (fabsf(cosTheta) >= 1.0f) { + glm_quat_copy(q1, dest); + return; + } + + /* longest path */ + if (!(cosTheta < 0.0f)) { + glm_vec4_negate(q1); + cosTheta = -cosTheta; + } + + sinTheta = sqrtf(1.0f - cosTheta * cosTheta); + + /* LERP to avoid zero division */ + if (fabsf(sinTheta) < 0.001f) { + glm_quat_lerp(from, to, t, dest); + return; + } + + /* SLERP */ + angle = acosf(cosTheta); + glm_vec4_scale(q1, sinf((1.0f - t) * angle), q1); + glm_vec4_scale(to, sinf(t * angle), q2); + + glm_vec4_add(q1, q2, q1); + glm_vec4_scale(q1, 1.0f / sinTheta, dest); +} + +/*! + * @brief creates view matrix using quaternion as camera orientation + * + * @param[in] eye eye + * @param[in] ori orientation in world space as quaternion + * @param[out] dest view matrix + */ +CGLM_INLINE +void +glm_quat_look(vec3 eye, versor ori, mat4 dest) { + /* orientation */ + glm_quat_mat4t(ori, dest); + + /* translate */ + glm_mat4_mulv3(dest, eye, 1.0f, dest[3]); + glm_vec3_negate(dest[3]); +} + +/*! + * @brief creates look rotation quaternion + * + * @param[in] dir direction to look + * @param[in] up up vector + * @param[out] dest destination quaternion + */ +CGLM_INLINE +void +glm_quat_for(vec3 dir, vec3 up, versor dest) { + CGLM_ALIGN_MAT mat3 m; + + glm_vec3_normalize_to(dir, m[2]); + + /* No need to negate in LH, but we use RH here */ + glm_vec3_negate(m[2]); + + glm_vec3_crossn(up, m[2], m[0]); + glm_vec3_cross(m[2], m[0], m[1]); + + glm_mat3_quat(m, dest); +} + +/*! + * @brief creates look rotation quaternion using source and + * destination positions p suffix stands for position + * + * @param[in] from source point + * @param[in] to destination point + * @param[in] up up vector + * @param[out] dest destination quaternion + */ +CGLM_INLINE +void +glm_quat_forp(vec3 from, vec3 to, vec3 up, versor dest) { + CGLM_ALIGN(8) vec3 dir; + glm_vec3_sub(to, from, dir); + glm_quat_for(dir, up, dest); +} + +/*! + * @brief rotate vector using using quaternion + * + * @param[in] q quaternion + * @param[in] v vector to rotate + * @param[out] dest rotated vector + */ +CGLM_INLINE +void +glm_quat_rotatev(versor q, vec3 v, vec3 dest) { + CGLM_ALIGN(16) versor p; + CGLM_ALIGN(8) vec3 u, v1, v2; + float s; + + glm_quat_normalize_to(q, p); + glm_quat_imag(p, u); + s = glm_quat_real(p); + + glm_vec3_scale(u, 2.0f * glm_vec3_dot(u, v), v1); + glm_vec3_scale(v, s * s - glm_vec3_dot(u, u), v2); + glm_vec3_add(v1, v2, v1); + + glm_vec3_cross(u, v, v2); + glm_vec3_scale(v2, 2.0f * s, v2); + + glm_vec3_add(v1, v2, dest); +} + +/*! + * @brief rotate existing transform matrix using quaternion + * + * @param[in] m existing transform matrix + * @param[in] q quaternion + * @param[out] dest rotated matrix/transform + */ +CGLM_INLINE +void +glm_quat_rotate(mat4 m, versor q, mat4 dest) { + CGLM_ALIGN_MAT mat4 rot; + glm_quat_mat4(q, rot); + glm_mul_rot(m, rot, dest); +} + +/*! + * @brief rotate existing transform matrix using quaternion at pivot point + * + * @param[in, out] m existing transform matrix + * @param[in] q quaternion + * @param[out] pivot pivot + */ +CGLM_INLINE +void +glm_quat_rotate_at(mat4 m, versor q, vec3 pivot) { + CGLM_ALIGN(8) vec3 pivotInv; + + glm_vec3_negate_to(pivot, pivotInv); + + glm_translate(m, pivot); + glm_quat_rotate(m, q, m); + glm_translate(m, pivotInv); +} + +/*! + * @brief rotate NEW transform matrix using quaternion at pivot point + * + * this creates rotation matrix, it assumes you don't have a matrix + * + * this should work faster than glm_quat_rotate_at because it reduces + * one glm_translate. + * + * @param[out] m existing transform matrix + * @param[in] q quaternion + * @param[in] pivot pivot + */ +CGLM_INLINE +void +glm_quat_rotate_atm(mat4 m, versor q, vec3 pivot) { + CGLM_ALIGN(8) vec3 pivotInv; + + glm_vec3_negate_to(pivot, pivotInv); + + glm_translate_make(m, pivot); + glm_quat_rotate(m, q, m); + glm_translate(m, pivotInv); +} + +/*! + * @brief Create quaternion from pointer + * + * @param[in] src pointer to an array of floats + * @param[out] dest quaternion + */ +CGLM_INLINE +void +glm_quat_make(const float * __restrict src, versor dest) { + dest[0] = src[0]; dest[1] = src[1]; + dest[2] = src[2]; dest[3] = src[3]; +} + +#endif /* cglm_quat_h */ diff --git a/cglm/include/cglm/ray.h b/cglm/include/cglm/ray.h new file mode 100644 index 0000000..d7831bc --- /dev/null +++ b/cglm/include/cglm/ray.h @@ -0,0 +1,174 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE bool glm_ray_triangle(vec3 origin, + vec3 direction, + vec3 v0, + vec3 v1, + vec3 v2, + float *d); + CGLM_INLINE bool glm_ray_sphere(vec3 origin, + vec3 dir, + vec4 s, + float * __restrict t1, + float * __restrict t2) + CGLM_INLINE void glm_ray_at(vec3 orig, vec3 dir, float t, vec3 point); +*/ + +#ifndef cglm_ray_h +#define cglm_ray_h + +#include "vec3.h" + +/*! + * @brief Möller–Trumbore ray-triangle intersection algorithm + * + * @param[in] origin origin of ray + * @param[in] direction direction of ray + * @param[in] v0 first vertex of triangle + * @param[in] v1 second vertex of triangle + * @param[in] v2 third vertex of triangle + * @param[in, out] d distance to intersection + * @return whether there is intersection + */ +CGLM_INLINE +bool +glm_ray_triangle(vec3 origin, + vec3 direction, + vec3 v0, + vec3 v1, + vec3 v2, + float *d) { + vec3 edge1, edge2, p, t, q; + float det, inv_det, u, v, dist; + const float epsilon = 0.000001f; + + glm_vec3_sub(v1, v0, edge1); + glm_vec3_sub(v2, v0, edge2); + glm_vec3_cross(direction, edge2, p); + + det = glm_vec3_dot(edge1, p); + if (det > -epsilon && det < epsilon) + return false; + + inv_det = 1.0f / det; + + glm_vec3_sub(origin, v0, t); + + u = inv_det * glm_vec3_dot(t, p); + if (u < 0.0f || u > 1.0f) + return false; + + glm_vec3_cross(t, edge1, q); + + v = inv_det * glm_vec3_dot(direction, q); + if (v < 0.0f || u + v > 1.0f) + return false; + + dist = inv_det * glm_vec3_dot(edge2, q); + + if (d) + *d = dist; + + return dist > epsilon; +} + +/*! + * @brief ray sphere intersection + * + * returns false if there is no intersection if true: + * + * - t1 > 0, t2 > 0: ray intersects the sphere at t1 and t2 both ahead of the origin + * - t1 < 0, t2 > 0: ray starts inside the sphere, exits at t2 + * - t1 < 0, t2 < 0: no intersection ahead of the ray ( returns false ) + * - the caller can check if the intersection points (t1 and t2) fall within a + * specific range (for example, tmin < t1, t2 < tmax) to determine if the + * intersections are within a desired segment of the ray + * + * @param[in] origin ray origin + * @param[out] dir normalized ray direction + * @param[in] s sphere [center.x, center.y, center.z, radii] + * @param[in] t1 near point1 (closer to origin) + * @param[in] t2 far point2 (farther from origin) + * + * @returns whether there is intersection + */ +CGLM_INLINE +bool +glm_ray_sphere(vec3 origin, + vec3 dir, + vec4 s, + float * __restrict t1, + float * __restrict t2) { + vec3 dp; + float r2, ddp, dpp, dscr, q, tmp, _t1, _t2; + + glm_vec3_sub(s, origin, dp); + + ddp = glm_vec3_dot(dir, dp); + dpp = glm_vec3_norm2(dp); + + /* compute the remedy term for numerical stability */ + glm_vec3_mulsubs(dir, ddp, dp); /* dp: remedy term */ + + r2 = s[3] * s[3]; + dscr = r2 - glm_vec3_norm2(dp); + + if (dscr < 0.0f) { + /* no intersection */ + return false; + } + + dscr = sqrtf(dscr); + q = (ddp >= 0.0f) ? (ddp + dscr) : (ddp - dscr); + + /* + include Press, William H., Saul A. Teukolsky, + William T. Vetterling, and Brian P. Flannery, + "Numerical Recipes in C," Cambridge University Press, 1992. + */ + _t1 = q; + _t2 = (dpp - r2) / q; + + /* adjust t1 and t2 to ensure t1 is the closer intersection */ + if (_t1 > _t2) { + tmp = _t1; + _t1 = _t2; + _t2 = tmp; + } + + *t1 = _t1; + *t2 = _t2; + + /* check if the closest intersection (t1) is behind the ray's origin */ + if (_t1 < 0.0f && _t2 < 0.0f) { + /* both intersections are behind the ray, no visible intersection */ + return false; + } + + return true; +} + +/*! + * @brief point using t by 𝐏(𝑡)=𝐀+𝑡𝐛 + * + * @param[in] orig origin of ray + * @param[in] dir direction of ray + * @param[in] t parameter + * @param[out] point point at t + */ +CGLM_INLINE +void +glm_ray_at(vec3 orig, vec3 dir, float t, vec3 point) { + vec3 dst; + glm_vec3_scale(dir, t, dst); + glm_vec3_add(orig, dst, point); +} + +#endif diff --git a/cglm/include/cglm/simd/arm.h b/cglm/include/cglm/simd/arm.h new file mode 100644 index 0000000..3e3bb22 --- /dev/null +++ b/cglm/include/cglm/simd/arm.h @@ -0,0 +1,204 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_simd_arm_h +#define cglm_simd_arm_h +#include "intrin.h" +#ifdef CGLM_SIMD_ARM + +#if defined(_M_ARM64) || defined(_M_HYBRID_X86_ARM64) || defined(_M_ARM64EC) || defined(__aarch64__) +# define CGLM_ARM64 1 +#else +# define CGLM_ARM64 0 +#endif + +#define glmm_load(p) vld1q_f32(p) +#define glmm_store(p, a) vst1q_f32(p, a) + +#define glmm_set1(x) vdupq_n_f32(x) +#define glmm_128 float32x4_t + +#define glmm_splat_x(x) vdupq_lane_f32(vget_low_f32(x), 0) +#define glmm_splat_y(x) vdupq_lane_f32(vget_low_f32(x), 1) +#define glmm_splat_z(x) vdupq_lane_f32(vget_high_f32(x), 0) +#define glmm_splat_w(x) vdupq_lane_f32(vget_high_f32(x), 1) + +#define glmm_xor(a, b) \ + vreinterpretq_f32_s32(veorq_s32(vreinterpretq_s32_f32(a), \ + vreinterpretq_s32_f32(b))) + +#define glmm_swplane(v) vextq_f32(v, v, 2) +#define glmm_low(x) vget_low_f32(x) +#define glmm_high(x) vget_high_f32(x) + +#define glmm_combine_ll(x, y) vcombine_f32(vget_low_f32(x), vget_low_f32(y)) +#define glmm_combine_hl(x, y) vcombine_f32(vget_high_f32(x), vget_low_f32(y)) +#define glmm_combine_lh(x, y) vcombine_f32(vget_low_f32(x), vget_high_f32(y)) +#define glmm_combine_hh(x, y) vcombine_f32(vget_high_f32(x), vget_high_f32(y)) + +#if defined(_WIN32) && defined(_MSC_VER) +/* # define glmm_float32x4_init(x, y, z, w) { .n128_f32 = { x, y, z, w } } */ +CGLM_INLINE +float32x4_t +glmm_float32x4_init(float x, float y, float z, float w) { + CGLM_ALIGN(16) float v[4] = {x, y, z, w}; + return vld1q_f32(v); +} +#else +# define glmm_float32x4_init(x, y, z, w) { x, y, z, w } +#endif + +#define glmm_float32x4_SIGNMASK_PNPN glmm_float32x4_init( 0.f, -0.f, 0.f, -0.f) +#define glmm_float32x4_SIGNMASK_NPNP glmm_float32x4_init(-0.f, 0.f, -0.f, 0.f) +#define glmm_float32x4_SIGNMASK_NPPN glmm_float32x4_init(-0.f, 0.f, 0.f, -0.f) + +static inline float32x4_t glmm_abs(float32x4_t v) { return vabsq_f32(v); } +static inline float32x4_t glmm_min(float32x4_t a, float32x4_t b) { return vminq_f32(a, b); } +static inline float32x4_t glmm_max(float32x4_t a, float32x4_t b) { return vmaxq_f32(a, b); } + +static inline +float32x4_t +glmm_vhadd(float32x4_t v) { +#if CGLM_ARM64 + float32x4_t p; + p = vpaddq_f32(v, v); /* [a+b, c+d, a+b, c+d] */ + return vpaddq_f32(p, p); /* [t, t, t, t] */; +#else + return vaddq_f32(vaddq_f32(glmm_splat_x(v), glmm_splat_y(v)), + vaddq_f32(glmm_splat_z(v), glmm_splat_w(v))); +#endif + /* TODO: measure speed of this compare to above */ + /* return vdupq_n_f32(vaddvq_f32(v)); */ + + /* + return vaddq_f32(vaddq_f32(glmm_splat_x(v), glmm_splat_y(v)), + vaddq_f32(glmm_splat_z(v), glmm_splat_w(v))); + */ + /* + this seems slower: + v = vaddq_f32(v, vrev64q_f32(v)); + return vaddq_f32(v, vcombine_f32(vget_high_f32(v), vget_low_f32(v))); + */ +} + +static inline +float +glmm_hadd(float32x4_t v) { +#if CGLM_ARM64 + return vaddvq_f32(v); +#else + v = vaddq_f32(v, vrev64q_f32(v)); + v = vaddq_f32(v, vcombine_f32(vget_high_f32(v), vget_low_f32(v))); + return vgetq_lane_f32(v, 0); +#endif +} + +static inline +float +glmm_hmin(float32x4_t v) { + float32x2_t t; + t = vpmin_f32(vget_low_f32(v), vget_high_f32(v)); + t = vpmin_f32(t, t); + return vget_lane_f32(t, 0); +} + +static inline +float +glmm_hmax(float32x4_t v) { + float32x2_t t; + t = vpmax_f32(vget_low_f32(v), vget_high_f32(v)); + t = vpmax_f32(t, t); + return vget_lane_f32(t, 0); +} + +static inline +float +glmm_dot(float32x4_t a, float32x4_t b) { + return glmm_hadd(vmulq_f32(a, b)); +} + +static inline +float32x4_t +glmm_vdot(float32x4_t a, float32x4_t b) { + return glmm_vhadd(vmulq_f32(a, b)); +} + +static inline +float +glmm_norm(float32x4_t a) { + return sqrtf(glmm_dot(a, a)); +} + +static inline +float +glmm_norm2(float32x4_t a) { + return glmm_dot(a, a); +} + +static inline +float +glmm_norm_one(float32x4_t a) { + return glmm_hadd(glmm_abs(a)); +} + +static inline +float +glmm_norm_inf(float32x4_t a) { + return glmm_hmax(glmm_abs(a)); +} + +static inline +float32x4_t +glmm_div(float32x4_t a, float32x4_t b) { +#if CGLM_ARM64 + return vdivq_f32(a, b); +#else + /* 2 iterations of Newton-Raphson refinement of reciprocal */ + float32x4_t r0, r1; + r0 = vrecpeq_f32(b); + r1 = vrecpsq_f32(r0, b); + r0 = vmulq_f32(r1, r0); + r1 = vrecpsq_f32(r0, b); + r0 = vmulq_f32(r1, r0); + return vmulq_f32(a, r0); +#endif +} + +static inline +float32x4_t +glmm_fmadd(float32x4_t a, float32x4_t b, float32x4_t c) { +#if CGLM_ARM64 + return vfmaq_f32(c, a, b); /* why vfmaq_f32 is slower than vmlaq_f32 ??? */ +#else + return vmlaq_f32(c, a, b); +#endif +} + +static inline +float32x4_t +glmm_fnmadd(float32x4_t a, float32x4_t b, float32x4_t c) { +#if CGLM_ARM64 + return vfmsq_f32(c, a, b); +#else + return vmlsq_f32(c, a, b); +#endif +} + +static inline +float32x4_t +glmm_fmsub(float32x4_t a, float32x4_t b, float32x4_t c) { + return glmm_fmadd(a, b, vnegq_f32(c)); +} + +static inline +float32x4_t +glmm_fnmsub(float32x4_t a, float32x4_t b, float32x4_t c) { + return vsubq_f32(vdupq_n_f32(0.0f), glmm_fmadd(a, b, c)); +} + +#endif +#endif /* cglm_simd_arm_h */ diff --git a/cglm/include/cglm/simd/avx/affine.h b/cglm/include/cglm/simd/avx/affine.h new file mode 100644 index 0000000..b02ff0c --- /dev/null +++ b/cglm/include/cglm/simd/avx/affine.h @@ -0,0 +1,66 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_affine_mat_avx_h +#define cglm_affine_mat_avx_h +#ifdef __AVX__ + +#include "../../common.h" +#include "../intrin.h" + +#include + +CGLM_INLINE +void +glm_mul_avx(mat4 m1, mat4 m2, mat4 dest) { + /* D = R * L (Column-Major) */ + + __m256 y0, y1, y2, y3, y4, y5, y6, y7, y8, y9; + + y0 = glmm_load256(m2[0]); /* h g f e d c b a */ + y1 = glmm_load256(m2[2]); /* p o n m l k j i */ + + y2 = glmm_load256(m1[0]); /* h g f e d c b a */ + y3 = glmm_load256(m1[2]); /* p o n m l k j i */ + + /* 0x03: 0b00000011 */ + y4 = _mm256_permute2f128_ps(y2, y2, 0x03); /* d c b a h g f e */ + y5 = _mm256_permute2f128_ps(y3, y3, 0x03); /* l k j i p o n m */ + + /* f f f f a a a a */ + /* h h h h c c c c */ + /* e e e e b b b b */ + /* g g g g d d d d */ + y6 = _mm256_permutevar_ps(y0, _mm256_set_epi32(1, 1, 1, 1, 0, 0, 0, 0)); + y7 = _mm256_permutevar_ps(y0, _mm256_set_epi32(3, 3, 3, 3, 2, 2, 2, 2)); + y8 = _mm256_permutevar_ps(y0, _mm256_set_epi32(0, 0, 0, 0, 1, 1, 1, 1)); + y9 = _mm256_permutevar_ps(y0, _mm256_set_epi32(2, 2, 2, 2, 3, 3, 3, 3)); + + glmm_store256(dest[0], + _mm256_add_ps(_mm256_add_ps(_mm256_mul_ps(y2, y6), + _mm256_mul_ps(y3, y7)), + _mm256_add_ps(_mm256_mul_ps(y4, y8), + _mm256_mul_ps(y5, y9)))); + + /* n n n n i i i i */ + /* p p p p k k k k */ + /* m m m m j j j j */ + /* o o o o l l l l */ + y6 = _mm256_permutevar_ps(y1, _mm256_set_epi32(1, 1, 1, 1, 0, 0, 0, 0)); + y7 = _mm256_permutevar_ps(y1, _mm256_set_epi32(3, 3, 3, 3, 2, 2, 2, 2)); + y8 = _mm256_permutevar_ps(y1, _mm256_set_epi32(0, 0, 0, 0, 1, 1, 1, 1)); + y9 = _mm256_permutevar_ps(y1, _mm256_set_epi32(2, 2, 2, 2, 3, 3, 3, 3)); + + glmm_store256(dest[2], + _mm256_add_ps(_mm256_add_ps(_mm256_mul_ps(y2, y6), + _mm256_mul_ps(y3, y7)), + _mm256_add_ps(_mm256_mul_ps(y4, y8), + _mm256_mul_ps(y5, y9)))); +} + +#endif +#endif /* cglm_affine_mat_avx_h */ diff --git a/cglm/include/cglm/simd/avx/mat4.h b/cglm/include/cglm/simd/avx/mat4.h new file mode 100644 index 0000000..33771c2 --- /dev/null +++ b/cglm/include/cglm/simd/avx/mat4.h @@ -0,0 +1,115 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_mat_simd_avx_h +#define cglm_mat_simd_avx_h +#ifdef __AVX__ + +#include "../../common.h" +#include "../intrin.h" + +CGLM_INLINE +void +glm_mat4_scale_avx(mat4 m, float s) { + __m256 y0, y1, y2, y3, y4; + + y0 = glmm_load256(m[0]); /* h g f e d c b a */ + y1 = glmm_load256(m[2]); /* p o n m l k j i */ + + y2 = _mm256_broadcast_ss(&s); + + y3 = _mm256_mul_ps(y0, y2); + y4 = _mm256_mul_ps(y1, y2); + + glmm_store256(m[0], y3); + glmm_store256(m[2], y4); +} + +/* TODO: this must be tested and compared to SSE version, may be slower!!! */ +CGLM_INLINE +void +glm_mat4_transp_avx(mat4 m, mat4 dest) { + __m256 y0, y1, y2, y3; + + y0 = glmm_load256(m[0]); /* h g f e d c b a */ + y1 = glmm_load256(m[2]); /* p o n m l k j i */ + + y2 = _mm256_unpacklo_ps(y0, y1); /* n f m e j b i a */ + y3 = _mm256_unpackhi_ps(y0, y1); /* p h o g l d k c */ + + y0 = _mm256_permute2f128_ps(y2, y3, 0x20); /* l d k c j b i a */ + y1 = _mm256_permute2f128_ps(y2, y3, 0x31); /* p h o g n f m e */ + + y2 = _mm256_unpacklo_ps(y0, y1); /* o k g c m i e a */ + y3 = _mm256_unpackhi_ps(y0, y1); /* p l h d n j f b */ + + y0 = _mm256_permute2f128_ps(y2, y3, 0x20); /* n j f b m i e a */ + y1 = _mm256_permute2f128_ps(y2, y3, 0x31); /* p l h d o k g c */ + + glmm_store256(dest[0], y0); + glmm_store256(dest[2], y1); +} + +CGLM_INLINE +void +glm_mat4_mul_avx(mat4 m1, mat4 m2, mat4 dest) { + /* D = R * L (Column-Major) */ + + __m256 y0, y1, y2, y3, y4, y5, y6, y7, y8, y9, y10, y11, y12, y13; + __m256i yi0, yi1, yi2, yi3; + + y0 = glmm_load256(m2[0]); /* h g f e d c b a */ + y1 = glmm_load256(m2[2]); /* p o n m l k j i */ + + y2 = glmm_load256(m1[0]); /* h g f e d c b a */ + y3 = glmm_load256(m1[2]); /* p o n m l k j i */ + + /* 0x03: 0b00000011 */ + y4 = _mm256_permute2f128_ps(y2, y2, 0x03); /* d c b a h g f e */ + y5 = _mm256_permute2f128_ps(y3, y3, 0x03); /* l k j i p o n m */ + + yi0 = _mm256_set_epi32(1, 1, 1, 1, 0, 0, 0, 0); + yi1 = _mm256_set_epi32(3, 3, 3, 3, 2, 2, 2, 2); + yi2 = _mm256_set_epi32(0, 0, 0, 0, 1, 1, 1, 1); + yi3 = _mm256_set_epi32(2, 2, 2, 2, 3, 3, 3, 3); + + /* f f f f a a a a */ + /* h h h h c c c c */ + /* e e e e b b b b */ + /* g g g g d d d d */ + y6 = _mm256_permutevar_ps(y0, yi0); + y7 = _mm256_permutevar_ps(y0, yi1); + y8 = _mm256_permutevar_ps(y0, yi2); + y9 = _mm256_permutevar_ps(y0, yi3); + + /* n n n n i i i i */ + /* p p p p k k k k */ + /* m m m m j j j j */ + /* o o o o l l l l */ + y10 = _mm256_permutevar_ps(y1, yi0); + y11 = _mm256_permutevar_ps(y1, yi1); + y12 = _mm256_permutevar_ps(y1, yi2); + y13 = _mm256_permutevar_ps(y1, yi3); + + y0 = _mm256_mul_ps(y2, y6); + y1 = _mm256_mul_ps(y2, y10); + + y0 = glmm256_fmadd(y3, y7, y0); + y1 = glmm256_fmadd(y3, y11, y1); + + y0 = glmm256_fmadd(y4, y8, y0); + y1 = glmm256_fmadd(y4, y12, y1); + + y0 = glmm256_fmadd(y5, y9, y0); + y1 = glmm256_fmadd(y5, y13, y1); + + glmm_store256(dest[0], y0); + glmm_store256(dest[2], y1); +} + +#endif +#endif /* cglm_mat_simd_avx_h */ diff --git a/cglm/include/cglm/simd/intrin.h b/cglm/include/cglm/simd/intrin.h new file mode 100644 index 0000000..c477f34 --- /dev/null +++ b/cglm/include/cglm/simd/intrin.h @@ -0,0 +1,153 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_intrin_h +#define cglm_intrin_h + +#if defined(_MSC_VER) && !defined(_M_ARM64EC) +# if (defined(_M_AMD64) || defined(_M_X64)) || _M_IX86_FP == 2 +# ifndef __SSE__ +# define __SSE__ +# endif +# ifndef __SSE2__ +# define __SSE2__ +# endif +# elif _M_IX86_FP == 1 +# ifndef __SSE__ +# define __SSE__ +# endif +# endif +/* do not use alignment for older visual studio versions */ +/* also ARM32 also causes similar error, disable it for now on ARM32 too */ +# if _MSC_VER < 1913 || _M_ARM /* Visual Studio 2017 version 15.6 */ +# define CGLM_ALL_UNALIGNED +# endif +#endif + +#ifdef __AVX__ +# include +# define CGLM_AVX_FP 1 +# ifndef __SSE2__ +# define __SSE2__ +# endif +# ifndef __SSE3__ +# define __SSE3__ +# endif +# ifndef __SSE4__ +# define __SSE4__ +# endif +# ifndef __SSE4_1__ +# define __SSE4_1__ +# endif +# ifndef __SSE4_2__ +# define __SSE4_2__ +# endif +# ifndef CGLM_SIMD_x86 +# define CGLM_SIMD_x86 +# endif +#endif + +#if defined(__SSE__) +# include +# define CGLM_SSE_FP 1 +# ifndef CGLM_SIMD_x86 +# define CGLM_SIMD_x86 +# endif +#endif + +#if defined(__SSE2__) +# include +# define CGLM_SSE2_FP 1 +# ifndef CGLM_SIMD_x86 +# define CGLM_SIMD_x86 +# endif +#endif + +#if defined(__SSE3__) +# include +# ifndef CGLM_SIMD_x86 +# define CGLM_SIMD_x86 +# endif +#endif + +#if defined(__SSE4_1__) +# include +# ifndef CGLM_SIMD_x86 +# define CGLM_SIMD_x86 +# endif +#endif + +#if defined(__SSE4_2__) +# include +# ifndef CGLM_SIMD_x86 +# define CGLM_SIMD_x86 +# endif +#endif + +/* ARM Neon */ +#if defined(_WIN32) && defined(_MSC_VER) +/* TODO: non-ARM stuff already inported, will this be better option */ +/* # include */ + +# if defined(_M_ARM64) || defined(_M_HYBRID_X86_ARM64) || defined(_M_ARM64EC) +# include +# include +# ifndef CGLM_NEON_FP +# define CGLM_NEON_FP 1 +# endif +# ifndef CGLM_SIMD_ARM +# define CGLM_SIMD_ARM +# endif +# elif defined(_M_ARM) +# include +# include +# ifndef CGLM_NEON_FP +# define CGLM_NEON_FP 1 +# endif +# ifndef CGLM_SIMD_ARM +# define CGLM_SIMD_ARM +# endif +# endif + +#else /* non-windows */ +# if defined(__ARM_NEON) || defined(__ARM_NEON__) +# include +# if defined(__ARM_NEON_FP) || defined(__ARM_FP) +# define CGLM_NEON_FP 1 +# endif +# ifndef CGLM_SIMD_ARM +# define CGLM_SIMD_ARM +# endif +# endif +#endif + +/* WebAssembly */ +#if defined(__wasm__) && defined(__wasm_simd128__) +# ifndef CGLM_SIMD_WASM +# define CGLM_SIMD_WASM +# endif +#endif + +#if defined(CGLM_SIMD_x86) || defined(CGLM_SIMD_ARM) || defined(CGLM_SIMD_WASM) +# ifndef CGLM_SIMD +# define CGLM_SIMD +# endif +#endif + +#if defined(CGLM_SIMD_x86) && !defined(CGLM_SIMD_WASM) +# include "x86.h" +#endif + +#if defined(CGLM_SIMD_ARM) +# include "arm.h" +#endif + +#if defined(CGLM_SIMD_WASM) +# include "wasm.h" +#endif + +#endif /* cglm_intrin_h */ diff --git a/cglm/include/cglm/simd/neon/affine.h b/cglm/include/cglm/simd/neon/affine.h new file mode 100644 index 0000000..b0a65a6 --- /dev/null +++ b/cglm/include/cglm/simd/neon/affine.h @@ -0,0 +1,121 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_affine_neon_h +#define cglm_affine_neon_h +#if defined(CGLM_NEON_FP) + +#include "../../common.h" +#include "../intrin.h" + +CGLM_INLINE +void +glm_mul_neon(mat4 m1, mat4 m2, mat4 dest) { + /* D = R * L (Column-Major) */ + + glmm_128 l, r0, r1, r2, r3, v0, v1, v2, v3; + + l = glmm_load(m1[0]); + r0 = glmm_load(m2[0]); + r1 = glmm_load(m2[1]); + r2 = glmm_load(m2[2]); + r3 = glmm_load(m2[3]); + + v0 = vmulq_f32(glmm_splat_x(r0), l); + v1 = vmulq_f32(glmm_splat_x(r1), l); + v2 = vmulq_f32(glmm_splat_x(r2), l); + v3 = vmulq_f32(glmm_splat_x(r3), l); + + l = glmm_load(m1[1]); + v0 = glmm_fmadd(glmm_splat_y(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_y(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_y(r2), l, v2); + v3 = glmm_fmadd(glmm_splat_y(r3), l, v3); + + l = glmm_load(m1[2]); + v0 = glmm_fmadd(glmm_splat_z(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_z(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_z(r2), l, v2); + v3 = glmm_fmadd(glmm_splat_z(r3), l, v3); + + v3 = glmm_fmadd(glmm_splat_w(r3), glmm_load(m1[3]), v3); + + glmm_store(dest[0], v0); + glmm_store(dest[1], v1); + glmm_store(dest[2], v2); + glmm_store(dest[3], v3); +} + +CGLM_INLINE +void +glm_mul_rot_neon(mat4 m1, mat4 m2, mat4 dest) { + /* D = R * L (Column-Major) */ + + glmm_128 l, r0, r1, r2, v0, v1, v2; + + l = glmm_load(m1[0]); + r0 = glmm_load(m2[0]); + r1 = glmm_load(m2[1]); + r2 = glmm_load(m2[2]); + + v0 = vmulq_f32(glmm_splat_x(r0), l); + v1 = vmulq_f32(glmm_splat_x(r1), l); + v2 = vmulq_f32(glmm_splat_x(r2), l); + + l = glmm_load(m1[1]); + v0 = glmm_fmadd(glmm_splat_y(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_y(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_y(r2), l, v2); + + l = glmm_load(m1[2]); + v0 = glmm_fmadd(glmm_splat_z(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_z(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_z(r2), l, v2); + + glmm_store(dest[0], v0); + glmm_store(dest[1], v1); + glmm_store(dest[2], v2); + glmm_store(dest[3], glmm_load(m1[3])); +} + +CGLM_INLINE +void +glm_inv_tr_neon(mat4 mat) { + float32x4x4_t vmat; + glmm_128 r0, r1, r2, x0; + + vmat = vld4q_f32(mat[0]); + r0 = vmat.val[0]; + r1 = vmat.val[1]; + r2 = vmat.val[2]; + + x0 = glmm_fmadd(r0, glmm_splat_w(r0), + glmm_fmadd(r1, glmm_splat_w(r1), + vmulq_f32(r2, glmm_splat_w(r2)))); + x0 = vnegq_f32(x0); + + glmm_store(mat[0], r0); + glmm_store(mat[1], r1); + glmm_store(mat[2], r2); + glmm_store(mat[3], x0); + + mat[0][3] = 0.0f; + mat[1][3] = 0.0f; + mat[2][3] = 0.0f; + mat[3][3] = 1.0f; + + /* TODO: ? + zo = vget_high_f32(r3); + vst1_lane_f32(&mat[0][3], zo, 0); + vst1_lane_f32(&mat[1][3], zo, 0); + vst1_lane_f32(&mat[2][3], zo, 0); + vst1_lane_f32(&mat[3][3], zo, 1); + */ +} + +#endif +#endif /* cglm_affine_neon_h */ diff --git a/cglm/include/cglm/simd/neon/mat2.h b/cglm/include/cglm/simd/neon/mat2.h new file mode 100644 index 0000000..7d0d9eb --- /dev/null +++ b/cglm/include/cglm/simd/neon/mat2.h @@ -0,0 +1,44 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_mat2_neon_h +#define cglm_mat2_neon_h +#if defined(CGLM_NEON_FP) + +#include "../../common.h" +#include "../intrin.h" + +CGLM_INLINE +void +glm_mat2_mul_neon(mat2 m1, mat2 m2, mat2 dest) { + float32x4x2_t a1; + glmm_128 x0, x1, x2; + float32x2_t dc, ba; + + x1 = glmm_load(m1[0]); /* d c b a */ + x2 = glmm_load(m2[0]); /* h g f e */ + + dc = vget_high_f32(x1); + ba = vget_low_f32(x1); + + /* g g e e, h h f f */ + a1 = vtrnq_f32(x2, x2); + + /* + dest[0][0] = a * e + c * f; + dest[0][1] = b * e + d * f; + dest[1][0] = a * g + c * h; + dest[1][1] = b * g + d * h; + */ + x0 = glmm_fmadd(vcombine_f32(ba, ba), a1.val[0], + vmulq_f32(vcombine_f32(dc, dc), a1.val[1])); + + glmm_store(dest[0], x0); +} + +#endif +#endif /* cglm_mat2_neon_h */ diff --git a/cglm/include/cglm/simd/neon/mat4.h b/cglm/include/cglm/simd/neon/mat4.h new file mode 100644 index 0000000..6cf9811 --- /dev/null +++ b/cglm/include/cglm/simd/neon/mat4.h @@ -0,0 +1,468 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_mat4_neon_h +#define cglm_mat4_neon_h +#if defined(CGLM_NEON_FP) + +#include "../../common.h" +#include "../intrin.h" + +CGLM_INLINE +void +glm_mat4_scale_neon(mat4 m, float s) { + float32x4_t v0; + + v0 = vdupq_n_f32(s); + + vst1q_f32(m[0], vmulq_f32(vld1q_f32(m[0]), v0)); + vst1q_f32(m[1], vmulq_f32(vld1q_f32(m[1]), v0)); + vst1q_f32(m[2], vmulq_f32(vld1q_f32(m[2]), v0)); + vst1q_f32(m[3], vmulq_f32(vld1q_f32(m[3]), v0)); +} + +CGLM_INLINE +void +glm_mat4_transp_neon(mat4 m, mat4 dest) { + float32x4x4_t vmat; + + vmat = vld4q_f32(m[0]); + + vst1q_f32(dest[0], vmat.val[0]); + vst1q_f32(dest[1], vmat.val[1]); + vst1q_f32(dest[2], vmat.val[2]); + vst1q_f32(dest[3], vmat.val[3]); +} + +CGLM_INLINE +void +glm_mat4_mul_neon(mat4 m1, mat4 m2, mat4 dest) { + /* D = R * L (Column-Major) */ + + glmm_128 l, r0, r1, r2, r3, v0, v1, v2, v3; + + l = glmm_load(m1[0]); + r0 = glmm_load(m2[0]); + r1 = glmm_load(m2[1]); + r2 = glmm_load(m2[2]); + r3 = glmm_load(m2[3]); + + v0 = vmulq_f32(glmm_splat_x(r0), l); + v1 = vmulq_f32(glmm_splat_x(r1), l); + v2 = vmulq_f32(glmm_splat_x(r2), l); + v3 = vmulq_f32(glmm_splat_x(r3), l); + + l = glmm_load(m1[1]); + v0 = glmm_fmadd(glmm_splat_y(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_y(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_y(r2), l, v2); + v3 = glmm_fmadd(glmm_splat_y(r3), l, v3); + + l = glmm_load(m1[2]); + v0 = glmm_fmadd(glmm_splat_z(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_z(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_z(r2), l, v2); + v3 = glmm_fmadd(glmm_splat_z(r3), l, v3); + + l = glmm_load(m1[3]); + v0 = glmm_fmadd(glmm_splat_w(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_w(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_w(r2), l, v2); + v3 = glmm_fmadd(glmm_splat_w(r3), l, v3); + + glmm_store(dest[0], v0); + glmm_store(dest[1], v1); + glmm_store(dest[2], v2); + glmm_store(dest[3], v3); +} + +CGLM_INLINE +void +glm_mat4_mulv_neon(mat4 m, vec4 v, vec4 dest) { + float32x4_t l0, l1, l2, l3; + float32x2_t vlo, vhi; + + l0 = vld1q_f32(m[0]); + l1 = vld1q_f32(m[1]); + l2 = vld1q_f32(m[2]); + l3 = vld1q_f32(m[3]); + + vlo = vld1_f32(&v[0]); + vhi = vld1_f32(&v[2]); + + l0 = vmulq_lane_f32(l0, vlo, 0); + l0 = vmlaq_lane_f32(l0, l1, vlo, 1); + l0 = vmlaq_lane_f32(l0, l2, vhi, 0); + l0 = vmlaq_lane_f32(l0, l3, vhi, 1); + + vst1q_f32(dest, l0); +} + +CGLM_INLINE +float +glm_mat4_det_neon(mat4 mat) { + float32x4_t r0, r1, r2, r3, x0, x1, x2; + float32x2_t ij, op, mn, kl, nn, mm, jj, ii, gh, ef, t12, t34; + float32x4x2_t a1; + float32x4_t x3 = glmm_float32x4_SIGNMASK_PNPN; + + /* 127 <- 0, [square] det(A) = det(At) */ + r0 = glmm_load(mat[0]); /* d c b a */ + r1 = vrev64q_f32(glmm_load(mat[1])); /* g h e f */ + r2 = vrev64q_f32(glmm_load(mat[2])); /* l k i j */ + r3 = vrev64q_f32(glmm_load(mat[3])); /* o p m n */ + + gh = vget_high_f32(r1); + ef = vget_low_f32(r1); + kl = vget_high_f32(r2); + ij = vget_low_f32(r2); + op = vget_high_f32(r3); + mn = vget_low_f32(r3); + mm = vdup_lane_f32(mn, 1); + nn = vdup_lane_f32(mn, 0); + ii = vdup_lane_f32(ij, 1); + jj = vdup_lane_f32(ij, 0); + + /* + t[1] = j * p - n * l; + t[2] = j * o - n * k; + t[3] = i * p - m * l; + t[4] = i * o - m * k; + */ + x0 = glmm_fnmadd(vcombine_f32(kl, kl), vcombine_f32(nn, mm), + vmulq_f32(vcombine_f32(op, op), vcombine_f32(jj, ii))); + + t12 = vget_low_f32(x0); + t34 = vget_high_f32(x0); + + /* 1 3 1 3 2 4 2 4 */ + a1 = vuzpq_f32(x0, x0); + + /* + t[0] = k * p - o * l; + t[0] = k * p - o * l; + t[5] = i * n - m * j; + t[5] = i * n - m * j; + */ + x1 = glmm_fnmadd(vcombine_f32(vdup_lane_f32(kl, 0), jj), + vcombine_f32(vdup_lane_f32(op, 1), mm), + vmulq_f32(vcombine_f32(vdup_lane_f32(op, 0), nn), + vcombine_f32(vdup_lane_f32(kl, 1), ii))); + + /* + a * (f * t[0] - g * t[1] + h * t[2]) + - b * (e * t[0] - g * t[3] + h * t[4]) + + c * (e * t[1] - f * t[3] + h * t[5]) + - d * (e * t[2] - f * t[4] + g * t[5]) + */ + x2 = glmm_fnmadd(vcombine_f32(vdup_lane_f32(gh, 1), vdup_lane_f32(ef, 0)), + vcombine_f32(vget_low_f32(a1.val[0]), t34), + vmulq_f32(vcombine_f32(ef, vdup_lane_f32(ef, 1)), + vcombine_f32(vget_low_f32(x1), t12))); + + x2 = glmm_fmadd(vcombine_f32(vdup_lane_f32(gh, 0), gh), + vcombine_f32(vget_low_f32(a1.val[1]), vget_high_f32(x1)), x2); + + x2 = glmm_xor(x2, x3); + + return glmm_hadd(vmulq_f32(x2, r0)); +} + +/* old one */ +#if 0 +CGLM_INLINE +void +glm_mat4_inv_neon(mat4 mat, mat4 dest) { + float32x4_t r0, r1, r2, r3, + v0, v1, v2, v3, + t0, t1, t2, t3, t4, t5, + x0, x1, x2, x3, x4, x5, x6, x7, x8; + float32x4x2_t a1; + float32x2_t lp, ko, hg, jn, im, fe, ae, bf, cg, dh; + float32x4_t x9 = glmm_float32x4_SIGNMASK_NPNP; + + x8 = vrev64q_f32(x9); + + /* 127 <- 0 */ + r0 = glmm_load(mat[0]); /* d c b a */ + r1 = glmm_load(mat[1]); /* h g f e */ + r2 = glmm_load(mat[2]); /* l k j i */ + r3 = glmm_load(mat[3]); /* p o n m */ + + /* l p k o, j n i m */ + a1 = vzipq_f32(r3, r2); + + jn = vget_high_f32(a1.val[0]); + im = vget_low_f32(a1.val[0]); + lp = vget_high_f32(a1.val[1]); + ko = vget_low_f32(a1.val[1]); + hg = vget_high_f32(r1); + + x1 = vcombine_f32(vdup_lane_f32(lp, 0), lp); /* l p p p */ + x2 = vcombine_f32(vdup_lane_f32(ko, 0), ko); /* k o o o */ + x0 = vcombine_f32(vdup_lane_f32(lp, 1), vdup_lane_f32(hg, 1)); /* h h l l */ + x3 = vcombine_f32(vdup_lane_f32(ko, 1), vdup_lane_f32(hg, 0)); /* g g k k */ + + /* t1[0] = k * p - o * l; + t1[0] = k * p - o * l; + t2[0] = g * p - o * h; + t3[0] = g * l - k * h; */ + t0 = glmm_fnmadd(x2, x0, vmulq_f32(x3, x1)); + + fe = vget_low_f32(r1); + x4 = vcombine_f32(vdup_lane_f32(jn, 0), jn); /* j n n n */ + x5 = vcombine_f32(vdup_lane_f32(jn, 1), vdup_lane_f32(fe, 1)); /* f f j j */ + + /* t1[1] = j * p - n * l; + t1[1] = j * p - n * l; + t2[1] = f * p - n * h; + t3[1] = f * l - j * h; */ + t1 = glmm_fnmadd(x4, x0, vmulq_f32(x5, x1)); + + /* t1[2] = j * o - n * k + t1[2] = j * o - n * k; + t2[2] = f * o - n * g; + t3[2] = f * k - j * g; */ + t2 = glmm_fnmadd(x4, x3, vmulq_f32(x5, x2)); + + x6 = vcombine_f32(vdup_lane_f32(im, 1), vdup_lane_f32(fe, 0)); /* e e i i */ + x7 = vcombine_f32(vdup_lane_f32(im, 0), im); /* i m m m */ + + /* t1[3] = i * p - m * l; + t1[3] = i * p - m * l; + t2[3] = e * p - m * h; + t3[3] = e * l - i * h; */ + t3 = glmm_fnmadd(x7, x0, vmulq_f32(x6, x1)); + + /* t1[4] = i * o - m * k; + t1[4] = i * o - m * k; + t2[4] = e * o - m * g; + t3[4] = e * k - i * g; */ + t4 = glmm_fnmadd(x7, x3, vmulq_f32(x6, x2)); + + /* t1[5] = i * n - m * j; + t1[5] = i * n - m * j; + t2[5] = e * n - m * f; + t3[5] = e * j - i * f; */ + t5 = glmm_fnmadd(x7, x5, vmulq_f32(x6, x4)); + + /* h d f b, g c e a */ + a1 = vtrnq_f32(r0, r1); + + x4 = vrev64q_f32(a1.val[0]); /* c g a e */ + x5 = vrev64q_f32(a1.val[1]); /* d h b f */ + + ae = vget_low_f32(x4); + cg = vget_high_f32(x4); + bf = vget_low_f32(x5); + dh = vget_high_f32(x5); + + x0 = vcombine_f32(ae, vdup_lane_f32(ae, 1)); /* a a a e */ + x1 = vcombine_f32(bf, vdup_lane_f32(bf, 1)); /* b b b f */ + x2 = vcombine_f32(cg, vdup_lane_f32(cg, 1)); /* c c c g */ + x3 = vcombine_f32(dh, vdup_lane_f32(dh, 1)); /* d d d h */ + + /* + dest[0][0] = f * t1[0] - g * t1[1] + h * t1[2]; + dest[0][1] =-(b * t1[0] - c * t1[1] + d * t1[2]); + dest[0][2] = b * t2[0] - c * t2[1] + d * t2[2]; + dest[0][3] =-(b * t3[0] - c * t3[1] + d * t3[2]); */ + v0 = glmm_xor(glmm_fmadd(x3, t2, glmm_fnmadd(x2, t1, vmulq_f32(x1, t0))), x8); + + /* + dest[2][0] = e * t1[1] - f * t1[3] + h * t1[5]; + dest[2][1] =-(a * t1[1] - b * t1[3] + d * t1[5]); + dest[2][2] = a * t2[1] - b * t2[3] + d * t2[5]; + dest[2][3] =-(a * t3[1] - b * t3[3] + d * t3[5]);*/ + v2 = glmm_xor(glmm_fmadd(x3, t5, glmm_fnmadd(x1, t3, vmulq_f32(x0, t1))), x8); + + /* + dest[1][0] =-(e * t1[0] - g * t1[3] + h * t1[4]); + dest[1][1] = a * t1[0] - c * t1[3] + d * t1[4]; + dest[1][2] =-(a * t2[0] - c * t2[3] + d * t2[4]); + dest[1][3] = a * t3[0] - c * t3[3] + d * t3[4]; */ + v1 = glmm_xor(glmm_fmadd(x3, t4, glmm_fnmadd(x2, t3, vmulq_f32(x0, t0))), x9); + + /* + dest[3][0] =-(e * t1[2] - f * t1[4] + g * t1[5]); + dest[3][1] = a * t1[2] - b * t1[4] + c * t1[5]; + dest[3][2] =-(a * t2[2] - b * t2[4] + c * t2[5]); + dest[3][3] = a * t3[2] - b * t3[4] + c * t3[5]; */ + v3 = glmm_xor(glmm_fmadd(x2, t5, glmm_fnmadd(x1, t4, vmulq_f32(x0, t2))), x9); + + /* determinant */ + x0 = vcombine_f32(vget_low_f32(vzipq_f32(v0, v1).val[0]), + vget_low_f32(vzipq_f32(v2, v3).val[0])); + + /* + x0 = glmm_div(glmm_set1_rval(1.0f), glmm_vhadd(vmulq_f32(x0, r0))); + + glmm_store(dest[0], vmulq_f32(v0, x0)); + glmm_store(dest[1], vmulq_f32(v1, x0)); + glmm_store(dest[2], vmulq_f32(v2, x0)); + glmm_store(dest[3], vmulq_f32(v3, x0)); + */ + + x0 = glmm_vhadd(vmulq_f32(x0, r0)); + + glmm_store(dest[0], glmm_div(v0, x0)); + glmm_store(dest[1], glmm_div(v1, x0)); + glmm_store(dest[2], glmm_div(v2, x0)); + glmm_store(dest[3], glmm_div(v3, x0)); +} +#endif + +CGLM_INLINE +void +glm_mat4_inv_neon(mat4 mat, mat4 dest) { + float32x4_t r0, r1, r2, r3, + v0, v1, v2, v3, v4, v5, + t0, t1, t2; + float32x4x2_t a0, a1, a2, a3, a4; + float32x4_t s1 = glmm_float32x4_SIGNMASK_PNPN, s2; + +#if !CGLM_ARM64 + float32x2_t l0, l1; +#endif + + s2 = vrev64q_f32(s1); + + /* 127 <- 0 */ + r0 = glmm_load(mat[0]); /* d c b a */ + r1 = glmm_load(mat[1]); /* h g f e */ + r2 = glmm_load(mat[2]); /* l k j i */ + r3 = glmm_load(mat[3]); /* p o n m */ + + a1 = vzipq_f32(r0, r2); /* l d k c, j b i a */ + a2 = vzipq_f32(r1, r3); /* p h o g, n f m e */ + a3 = vzipq_f32(a2.val[0], a1.val[0]); /* j n b f, i m a e */ + a4 = vzipq_f32(a2.val[1], a1.val[1]); /* l p d h, k o c g */ + + v0 = vextq_f32(a1.val[0], a1.val[1], 2); /* k c j b */ + v1 = vextq_f32(a2.val[0], a2.val[1], 2); /* o g n f */ + v2 = vextq_f32(a1.val[1], a2.val[0], 2); /* m e l d */ + v3 = vextq_f32(a2.val[1], a1.val[0], 2); /* i a p h */ + v4 = vextq_f32(v1, v2, 2); /* l d o g */ + v5 = vextq_f32(v0, v3, 2); /* p h k c */ + + /* c2 = c * h - g * d c12 = a * g - c * e c8 = a * f - b * e + c1 = k * p - o * l c11 = i * o - k * m c7 = i * n - j * m + c4 = h * a - d * e c6 = b * h - d * f c10 = b * g - c * f + c3 = p * i - l * m c5 = j * p - l * n c9 = j * o - k * n */ + t0 = vmulq_f32(v5, v3); + t1 = vmulq_f32(a1.val[0], a2.val[1]); + t2 = vmulq_f32(a1.val[0], v1); + + t0 = glmm_fnmadd(v4, v2, t0); + t1 = glmm_fnmadd(a1.val[1], a2.val[0], t1); + t2 = glmm_fnmadd(v0, a2.val[0], t2); + + t0 = vrev64q_f32(t0); + t1 = vrev64q_f32(t1); + t2 = vrev64q_f32(t2); + + /* det */ + v0 = vrev64q_f32(t2); + v1 = vextq_f32(t1, t1, 2); + v0 = vmulq_f32(t0, v0); + v1 = vrev64q_f32(v1); + v1 = vmulq_f32(v1, t1); + + /* c3 * c10 + c4 * c9 + c1 * c8 + c2 * c7 */ +#if CGLM_ARM64 + v0 = vpaddq_f32(v0, v0); + v0 = vpaddq_f32(v0, v0); +#else + l0 = vget_low_f32(v0); + l1 = vget_high_f32(v0); + + l0 = vpadd_f32(l0, l0); /* [a+b, a+b] */ + l1 = vpadd_f32(l1, l1); /* [c+d, c+d] */ + l0 = vadd_f32(l0, l1); /* [sum, sum] */ + + v0 = vcombine_f32(l0, l0); +#endif + + /* c5 * c12 + c6 * c11 */ +#if CGLM_ARM64 + v1 = vpaddq_f32(v1, v1); +#else + l0 = vget_low_f32(v1); + l1 = vget_high_f32(v1); + + l0 = vpadd_f32(l0, l0); /* [a+b, a+b] */ + l1 = vpadd_f32(l1, l1); /* [c+d, c+d] */ + + v1 = vcombine_f32(l0, l1); +#endif + + v0 = vsubq_f32(v0, v1); /* det */ + + /* inv div */ + v1 = vdupq_n_f32(1.0f); + v0 = glmm_div(v1, v0); /* inv div */ + + /* multiply t0,t1,t2 by idt to reduce 1mul below: 2eor+4mul vs 3mul+4eor */ + t0 = vmulq_f32(t0, v0); + t1 = vmulq_f32(t1, v0); + t2 = vmulq_f32(t2, v0); + + a0 = vzipq_f32(t0, t0); /* c4 c4 c3 c3, c2 c2 c1 c1 */ + a1 = vzipq_f32(t1, t1); /* c6 c6 c5 c5, c12 c12 c11 c11 */ + a2 = vzipq_f32(t2, t2); /* c10 c10 c9 c9, c8 c8 c7 c7 */ + + /* result */ + + /* dest[0][0] = (f * c1 - g * c5 + h * c9) * idt; + dest[0][1] = (b * c1 - c * c5 + d * c9) * ndt; + dest[0][2] = (n * c2 - o * c6 + p * c10) * idt; + dest[0][3] = (j * c2 - k * c6 + l * c10) * ndt; + + dest[1][0] = (e * c1 - g * c3 + h * c11) * ndt; + dest[1][1] = (a * c1 - c * c3 + d * c11) * idt; + dest[1][2] = (m * c2 - o * c4 + p * c12) * ndt; + dest[1][3] = (i * c2 - k * c4 + l * c12) * idt; + + dest[2][0] = (e * c5 - f * c3 + h * c7) * idt; + dest[2][1] = (a * c5 - b * c3 + d * c7) * ndt; + dest[2][2] = (m * c6 - n * c4 + p * c8) * idt; + dest[2][3] = (i * c6 - j * c4 + l * c8) * ndt; + + dest[3][0] = (e * c9 - f * c11 + g * c7) * ndt; + dest[3][1] = (a * c9 - b * c11 + c * c7) * idt; + dest[3][2] = (m * c10 - n * c12 + o * c8) * ndt; + dest[3][3] = (i * c10 - j * c12 + k * c8) * idt; */ + + r0 = vmulq_f32(a3.val[1], a0.val[0]); + r1 = vmulq_f32(a3.val[0], a0.val[0]); + r2 = vmulq_f32(a3.val[0], a1.val[1]); + r3 = vmulq_f32(a3.val[0], a2.val[1]); + + r0 = glmm_fnmadd(a4.val[0], a1.val[1], r0); + r1 = glmm_fnmadd(a4.val[0], a0.val[1], r1); + r2 = glmm_fnmadd(a3.val[1], a0.val[1], r2); + r3 = glmm_fnmadd(a3.val[1], a1.val[0], r3); + + r0 = glmm_fmadd(a4.val[1], a2.val[1], r0); + r1 = glmm_fmadd(a4.val[1], a1.val[0], r1); + r2 = glmm_fmadd(a4.val[1], a2.val[0], r2); + r3 = glmm_fmadd(a4.val[0], a2.val[0], r3); + + /* 4xor may be fastart then 4mul, see above */ + r0 = glmm_xor(r0, s1); + r1 = glmm_xor(r1, s2); + r2 = glmm_xor(r2, s1); + r3 = glmm_xor(r3, s2); + + glmm_store(dest[0], r0); + glmm_store(dest[1], r1); + glmm_store(dest[2], r2); + glmm_store(dest[3], r3); +} + +#endif +#endif /* cglm_mat4_neon_h */ diff --git a/cglm/include/cglm/simd/neon/quat.h b/cglm/include/cglm/simd/neon/quat.h new file mode 100644 index 0000000..55dc1da --- /dev/null +++ b/cglm/include/cglm/simd/neon/quat.h @@ -0,0 +1,57 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_quat_neon_h +#define cglm_quat_neon_h +#if defined(CGLM_NEON_FP) + +#include "../../common.h" +#include "../intrin.h" + +CGLM_INLINE +void +glm_quat_mul_neon(versor p, versor q, versor dest) { + /* + + (a1 b2 + b1 a2 + c1 d2 − d1 c2)i + + (a1 c2 − b1 d2 + c1 a2 + d1 b2)j + + (a1 d2 + b1 c2 − c1 b2 + d1 a2)k + a1 a2 − b1 b2 − c1 c2 − d1 d2 + */ + + glmm_128 xp, xq, xqr, r, x, y, z, s2, s3; + glmm_128 s1 = glmm_float32x4_SIGNMASK_NPPN; + + float32x2_t qh, ql; + + xp = glmm_load(p); /* 3 2 1 0 */ + xq = glmm_load(q); + + r = vmulq_f32(glmm_splat_w(xp), xq); + x = glmm_splat_x(xp); + y = glmm_splat_y(xp); + z = glmm_splat_z(xp); + + ql = vget_high_f32(s1); + s3 = vcombine_f32(ql, ql); + s2 = vzipq_f32(s3, s3).val[0]; + + xqr = vrev64q_f32(xq); + qh = vget_high_f32(xqr); + ql = vget_low_f32(xqr); + + r = glmm_fmadd(glmm_xor(x, s3), vcombine_f32(qh, ql), r); + + r = glmm_fmadd(glmm_xor(y, s2), vcombine_f32(vget_high_f32(xq), + vget_low_f32(xq)), r); + + r = glmm_fmadd(glmm_xor(z, s1), vcombine_f32(ql, qh), r); + + glmm_store(dest, r); +} + +#endif +#endif /* cglm_quat_neon_h */ diff --git a/cglm/include/cglm/simd/sse2/affine.h b/cglm/include/cglm/simd/sse2/affine.h new file mode 100644 index 0000000..0619995 --- /dev/null +++ b/cglm/include/cglm/simd/sse2/affine.h @@ -0,0 +1,115 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_affine_mat_sse2_h +#define cglm_affine_mat_sse2_h +#if defined( __SSE__ ) || defined( __SSE2__ ) + +#include "../../common.h" +#include "../intrin.h" + +CGLM_INLINE +void +glm_mul_sse2(mat4 m1, mat4 m2, mat4 dest) { + /* D = R * L (Column-Major) */ + glmm_128 l, r0, r1, r2, r3, v0, v1, v2, v3; + + l = glmm_load(m1[0]); + r0 = glmm_load(m2[0]); + r1 = glmm_load(m2[1]); + r2 = glmm_load(m2[2]); + r3 = glmm_load(m2[3]); + + v0 = _mm_mul_ps(glmm_splat_x(r0), l); + v1 = _mm_mul_ps(glmm_splat_x(r1), l); + v2 = _mm_mul_ps(glmm_splat_x(r2), l); + v3 = _mm_mul_ps(glmm_splat_x(r3), l); + + l = glmm_load(m1[1]); + v0 = glmm_fmadd(glmm_splat_y(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_y(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_y(r2), l, v2); + v3 = glmm_fmadd(glmm_splat_y(r3), l, v3); + + l = glmm_load(m1[2]); + v0 = glmm_fmadd(glmm_splat_z(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_z(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_z(r2), l, v2); + v3 = glmm_fmadd(glmm_splat_z(r3), l, v3); + + l = glmm_load(m1[3]); + v3 = glmm_fmadd(glmm_splat_w(r3), l, v3); + + glmm_store(dest[0], v0); + glmm_store(dest[1], v1); + glmm_store(dest[2], v2); + glmm_store(dest[3], v3); +} + +CGLM_INLINE +void +glm_mul_rot_sse2(mat4 m1, mat4 m2, mat4 dest) { + /* D = R * L (Column-Major) */ + + glmm_128 l, r0, r1, r2, v0, v1, v2; + + l = glmm_load(m1[0]); + r0 = glmm_load(m2[0]); + r1 = glmm_load(m2[1]); + r2 = glmm_load(m2[2]); + + v0 = _mm_mul_ps(glmm_splat_x(r0), l); + v1 = _mm_mul_ps(glmm_splat_x(r1), l); + v2 = _mm_mul_ps(glmm_splat_x(r2), l); + + l = glmm_load(m1[1]); + v0 = glmm_fmadd(glmm_splat_y(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_y(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_y(r2), l, v2); + + l = glmm_load(m1[2]); + v0 = glmm_fmadd(glmm_splat_z(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_z(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_z(r2), l, v2); + + glmm_store(dest[0], v0); + glmm_store(dest[1], v1); + glmm_store(dest[2], v2); + glmm_store(dest[3], glmm_load(m1[3])); +} + +CGLM_INLINE +void +glm_inv_tr_sse2(mat4 mat) { + __m128 r0, r1, r2, r3, x0, x1, x2, x3, x4, x5; + + r0 = glmm_load(mat[0]); + r1 = glmm_load(mat[1]); + r2 = glmm_load(mat[2]); + r3 = glmm_load(mat[3]); + x1 = _mm_set_ps(1.0f, 0.0f, 0.0f, 0.0f); + + _MM_TRANSPOSE4_PS(r0, r1, r2, x1); + + x2 = glmm_shuff1(r3, 0, 0, 0, 0); + x3 = glmm_shuff1(r3, 1, 1, 1, 1); + x4 = glmm_shuff1(r3, 2, 2, 2, 2); + x5 = glmm_float32x4_SIGNMASK_NEG; + + x0 = glmm_fmadd(r0, x2, glmm_fmadd(r1, x3, _mm_mul_ps(r2, x4))); + x0 = _mm_xor_ps(x0, x5); + + x0 = _mm_add_ps(x0, x1); + + glmm_store(mat[0], r0); + glmm_store(mat[1], r1); + glmm_store(mat[2], r2); + glmm_store(mat[3], x0); +} + +#endif +#endif /* cglm_affine_mat_sse2_h */ diff --git a/cglm/include/cglm/simd/sse2/mat2.h b/cglm/include/cglm/simd/sse2/mat2.h new file mode 100644 index 0000000..31b3a29 --- /dev/null +++ b/cglm/include/cglm/simd/sse2/mat2.h @@ -0,0 +1,48 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_mat2_sse_h +#define cglm_mat2_sse_h +#if defined( __SSE__ ) || defined( __SSE2__ ) + +#include "../../common.h" +#include "../intrin.h" + +CGLM_INLINE +void +glm_mat2_mul_sse2(mat2 m1, mat2 m2, mat2 dest) { + __m128 x0, x1, x2, x3, x4; + + x1 = glmm_load(m1[0]); /* d c b a */ + x2 = glmm_load(m2[0]); /* h g f e */ + + x3 = glmm_shuff1(x2, 2, 2, 0, 0); + x4 = glmm_shuff1(x2, 3, 3, 1, 1); + x0 = _mm_movelh_ps(x1, x1); + x2 = _mm_movehl_ps(x1, x1); + + /* + dest[0][0] = a * e + c * f; + dest[0][1] = b * e + d * f; + dest[1][0] = a * g + c * h; + dest[1][1] = b * g + d * h; + */ + x0 = glmm_fmadd(x0, x3, _mm_mul_ps(x2, x4)); + + glmm_store(dest[0], x0); +} + +CGLM_INLINE +void +glm_mat2_transp_sse2(mat2 m, mat2 dest) { + /* d c b a */ + /* d b c a */ + glmm_store(dest[0], glmm_shuff1(glmm_load(m[0]), 3, 1, 2, 0)); +} + +#endif +#endif /* cglm_mat2_sse_h */ diff --git a/cglm/include/cglm/simd/sse2/mat3.h b/cglm/include/cglm/simd/sse2/mat3.h new file mode 100644 index 0000000..f07320c --- /dev/null +++ b/cglm/include/cglm/simd/sse2/mat3.h @@ -0,0 +1,76 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_mat3_sse_h +#define cglm_mat3_sse_h +#if defined( __SSE__ ) || defined( __SSE2__ ) + +#include "../../common.h" +#include "../intrin.h" + +CGLM_INLINE +void +glm_mat3_mul_sse2(mat3 m1, mat3 m2, mat3 dest) { + __m128 l0, l1, l2, r0, r1, r2, x0, x1, x2, x3, x4, x5, x6, x7, x8, x9; + + l0 = _mm_loadu_ps(m1[0]); + l1 = _mm_loadu_ps(&m1[1][1]); + + r0 = _mm_loadu_ps(m2[0]); + r1 = _mm_loadu_ps(&m2[1][1]); + + x8 = glmm_shuff1(l0, 0, 2, 1, 0); /* a00 a02 a01 a00 */ + x1 = glmm_shuff1(r0, 3, 0, 0, 0); /* b10 b00 b00 b00 */ + x2 = _mm_shuffle_ps(l0, l1, _MM_SHUFFLE(1, 0, 3, 3)); /* a12 a11 a10 a10 */ + x3 = _mm_shuffle_ps(r0, r1, _MM_SHUFFLE(2, 0, 3, 1)); /* b20 b11 b10 b01 */ + x0 = _mm_mul_ps(x8, x1); + + x6 = glmm_shuff1(l0, 1, 0, 2, 1); /* a01 a00 a02 a01 */ + x7 = glmm_shuff1(x3, 3, 3, 1, 1); /* b20 b20 b10 b10 */ + l2 = _mm_load_ss(&m1[2][2]); + r2 = _mm_load_ss(&m2[2][2]); + x1 = _mm_mul_ps(x6, x7); + l2 = glmm_shuff1(l2, 0, 0, 1, 0); /* a22 a22 0.f a22 */ + r2 = glmm_shuff1(r2, 0, 0, 1, 0); /* b22 b22 0.f b22 */ + + x4 = glmm_shuff1(x2, 0, 3, 2, 0); /* a10 a12 a11 a10 */ + x5 = glmm_shuff1(x2, 2, 0, 3, 2); /* a11 a10 a12 a11 */ + x6 = glmm_shuff1(x3, 2, 0, 0, 0); /* b11 b01 b01 b01 */ + x2 = glmm_shuff1(r1, 3, 3, 0, 0); /* b21 b21 b11 b11 */ + + x8 = _mm_unpackhi_ps(x8, x4); /* a10 a00 a12 a02 */ + x9 = _mm_unpackhi_ps(x7, x2); /* b21 b20 b21 b20 */ + + x0 = glmm_fmadd(x4, x6, x0); + x1 = glmm_fmadd(x5, x2, x1); + + x2 = _mm_movehl_ps(l2, l1); /* a22 a22 a21 a20 */ + x3 = glmm_shuff1(x2, 0, 2, 1, 0); /* a20 a22 a21 a20 */ + x2 = glmm_shuff1(x2, 1, 0, 2, 1); /* a21 a20 a22 a21 */ + x4 = _mm_shuffle_ps(r0, r1, _MM_SHUFFLE(1, 1, 2, 2)); /* b12 b12 b02 b02 */ + + x5 = glmm_shuff1(x4, 3, 0, 0, 0); /* b12 b02 b02 b02 */ + x4 = _mm_movehl_ps(r2, x4); /* b22 b22 b12 b12 */ + x0 = glmm_fmadd(x3, x5, x0); + x1 = glmm_fmadd(x2, x4, x1); + + /* + Dot Product : dest[2][2] = a02 * b20 + + a12 * b21 + + a22 * b22 + + 0 * 00 */ + x2 = _mm_movelh_ps(x8, l2); /* 0.f a22 a12 a02 */ + x3 = _mm_movelh_ps(x9, r2); /* 0.f b22 b21 b20 */ + x2 = glmm_vdots(x2, x3); + + _mm_storeu_ps(&dest[0][0], x0); + _mm_storeu_ps(&dest[1][1], x1); + _mm_store_ss (&dest[2][2], x2); +} + +#endif +#endif /* cglm_mat3_sse_h */ diff --git a/cglm/include/cglm/simd/sse2/mat4.h b/cglm/include/cglm/simd/sse2/mat4.h new file mode 100644 index 0000000..2127e72 --- /dev/null +++ b/cglm/include/cglm/simd/sse2/mat4.h @@ -0,0 +1,573 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_mat_sse_h +#define cglm_mat_sse_h +#if defined( __SSE__ ) || defined( __SSE2__ ) + +#include "../../common.h" +#include "../intrin.h" + +#define glm_mat4_inv_precise_sse2(mat, dest) glm_mat4_inv_sse2(mat, dest) + +CGLM_INLINE +void +glm_mat4_scale_sse2(mat4 m, float s) { + __m128 x0; + x0 = glmm_set1(s); + + glmm_store(m[0], _mm_mul_ps(glmm_load(m[0]), x0)); + glmm_store(m[1], _mm_mul_ps(glmm_load(m[1]), x0)); + glmm_store(m[2], _mm_mul_ps(glmm_load(m[2]), x0)); + glmm_store(m[3], _mm_mul_ps(glmm_load(m[3]), x0)); +} + +CGLM_INLINE +void +glm_mat4_transp_sse2(mat4 m, mat4 dest) { + __m128 r0, r1, r2, r3; + + r0 = glmm_load(m[0]); + r1 = glmm_load(m[1]); + r2 = glmm_load(m[2]); + r3 = glmm_load(m[3]); + + _MM_TRANSPOSE4_PS(r0, r1, r2, r3); + + glmm_store(dest[0], r0); + glmm_store(dest[1], r1); + glmm_store(dest[2], r2); + glmm_store(dest[3], r3); +} + +CGLM_INLINE +void +glm_mat4_mul_sse2(mat4 m1, mat4 m2, mat4 dest) { + /* D = R * L (Column-Major) */ + + glmm_128 l, r0, r1, r2, r3, v0, v1, v2, v3; + + l = glmm_load(m1[0]); + r0 = glmm_load(m2[0]); + r1 = glmm_load(m2[1]); + r2 = glmm_load(m2[2]); + r3 = glmm_load(m2[3]); + + v0 = _mm_mul_ps(glmm_splat_x(r0), l); + v1 = _mm_mul_ps(glmm_splat_x(r1), l); + v2 = _mm_mul_ps(glmm_splat_x(r2), l); + v3 = _mm_mul_ps(glmm_splat_x(r3), l); + + l = glmm_load(m1[1]); + v0 = glmm_fmadd(glmm_splat_y(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_y(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_y(r2), l, v2); + v3 = glmm_fmadd(glmm_splat_y(r3), l, v3); + + l = glmm_load(m1[2]); + v0 = glmm_fmadd(glmm_splat_z(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_z(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_z(r2), l, v2); + v3 = glmm_fmadd(glmm_splat_z(r3), l, v3); + + l = glmm_load(m1[3]); + v0 = glmm_fmadd(glmm_splat_w(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_w(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_w(r2), l, v2); + v3 = glmm_fmadd(glmm_splat_w(r3), l, v3); + + glmm_store(dest[0], v0); + glmm_store(dest[1], v1); + glmm_store(dest[2], v2); + glmm_store(dest[3], v3); +} + +CGLM_INLINE +void +glm_mat4_mulv_sse2(mat4 m, vec4 v, vec4 dest) { + __m128 x0, x1, m0, m1, m2, m3, v0, v1, v2, v3; + + m0 = glmm_load(m[0]); + m1 = glmm_load(m[1]); + m2 = glmm_load(m[2]); + m3 = glmm_load(m[3]); + + x0 = glmm_load(v); + v0 = glmm_splat_x(x0); + v1 = glmm_splat_y(x0); + v2 = glmm_splat_z(x0); + v3 = glmm_splat_w(x0); + + x1 = _mm_mul_ps(m3, v3); + x1 = glmm_fmadd(m2, v2, x1); + x1 = glmm_fmadd(m1, v1, x1); + x1 = glmm_fmadd(m0, v0, x1); + + glmm_store(dest, x1); +} + +CGLM_INLINE +float +glm_mat4_det_sse2(mat4 mat) { + __m128 r0, r1, r2, r3, x0, x1, x2; + + /* 127 <- 0, [square] det(A) = det(At) */ + r0 = glmm_load(mat[0]); /* d c b a */ + r1 = glmm_load(mat[1]); /* h g f e */ + r2 = glmm_load(mat[2]); /* l k j i */ + r3 = glmm_load(mat[3]); /* p o n m */ + + /* + t[1] = j * p - n * l; + t[2] = j * o - n * k; + t[3] = i * p - m * l; + t[4] = i * o - m * k; + */ + x0 = glmm_fnmadd(glmm_shuff1(r3, 0, 0, 1, 1), glmm_shuff1(r2, 2, 3, 2, 3), + _mm_mul_ps(glmm_shuff1(r2, 0, 0, 1, 1), + glmm_shuff1(r3, 2, 3, 2, 3))); + /* + t[0] = k * p - o * l; + t[0] = k * p - o * l; + t[5] = i * n - m * j; + t[5] = i * n - m * j; + */ + x1 = glmm_fnmadd(glmm_shuff1(r3, 0, 0, 2, 2), glmm_shuff1(r2, 1, 1, 3, 3), + _mm_mul_ps(glmm_shuff1(r2, 0, 0, 2, 2), + glmm_shuff1(r3, 1, 1, 3, 3))); + + /* + a * (f * t[0] - g * t[1] + h * t[2]) + - b * (e * t[0] - g * t[3] + h * t[4]) + + c * (e * t[1] - f * t[3] + h * t[5]) + - d * (e * t[2] - f * t[4] + g * t[5]) + */ + x2 = glmm_fnmadd(glmm_shuff1(r1, 1, 1, 2, 2), glmm_shuff1(x0, 3, 2, 2, 0), + _mm_mul_ps(glmm_shuff1(r1, 0, 0, 0, 1), + _mm_shuffle_ps(x1, x0, _MM_SHUFFLE(1, 0, 0, 0)))); + x2 = glmm_fmadd(glmm_shuff1(r1, 2, 3, 3, 3), + _mm_shuffle_ps(x0, x1, _MM_SHUFFLE(2, 2, 3, 1)), + x2); + + x2 = _mm_xor_ps(x2, glmm_float32x4_SIGNMASK_NPNP); + + return glmm_hadd(_mm_mul_ps(x2, r0)); +} + +CGLM_INLINE +void +glm_mat4_inv_fast_sse2(mat4 mat, mat4 dest) { + __m128 r0, r1, r2, r3, + v0, v1, v2, v3, + t0, t1, t2, t3, t4, t5, + x0, x1, x2, x3, x4, x5, x6, x7, x8, x9; + + /* x8 = _mm_set_ps(-0.f, 0.f, -0.f, 0.f); */ + x8 = glmm_float32x4_SIGNMASK_NPNP; + x9 = glmm_shuff1(x8, 2, 1, 2, 1); + + /* 127 <- 0 */ + r0 = glmm_load(mat[0]); /* d c b a */ + r1 = glmm_load(mat[1]); /* h g f e */ + r2 = glmm_load(mat[2]); /* l k j i */ + r3 = glmm_load(mat[3]); /* p o n m */ + + x0 = _mm_movehl_ps(r3, r2); /* p o l k */ + x3 = _mm_movelh_ps(r2, r3); /* n m j i */ + x1 = glmm_shuff1(x0, 1, 3, 3 ,3); /* l p p p */ + x2 = glmm_shuff1(x0, 0, 2, 2, 2); /* k o o o */ + x4 = glmm_shuff1(x3, 1, 3, 3, 3); /* j n n n */ + x7 = glmm_shuff1(x3, 0, 2, 2, 2); /* i m m m */ + + x6 = _mm_shuffle_ps(r2, r1, _MM_SHUFFLE(0, 0, 0, 0)); /* e e i i */ + x5 = _mm_shuffle_ps(r2, r1, _MM_SHUFFLE(1, 1, 1, 1)); /* f f j j */ + x3 = _mm_shuffle_ps(r2, r1, _MM_SHUFFLE(2, 2, 2, 2)); /* g g k k */ + x0 = _mm_shuffle_ps(r2, r1, _MM_SHUFFLE(3, 3, 3, 3)); /* h h l l */ + + t0 = _mm_mul_ps(x3, x1); + t1 = _mm_mul_ps(x5, x1); + t2 = _mm_mul_ps(x5, x2); + t3 = _mm_mul_ps(x6, x1); + t4 = _mm_mul_ps(x6, x2); + t5 = _mm_mul_ps(x6, x4); + + /* t1[0] = k * p - o * l; + t1[0] = k * p - o * l; + t2[0] = g * p - o * h; + t3[0] = g * l - k * h; */ + t0 = glmm_fnmadd(x2, x0, t0); + + /* t1[1] = j * p - n * l; + t1[1] = j * p - n * l; + t2[1] = f * p - n * h; + t3[1] = f * l - j * h; */ + t1 = glmm_fnmadd(x4, x0, t1); + + /* t1[2] = j * o - n * k + t1[2] = j * o - n * k; + t2[2] = f * o - n * g; + t3[2] = f * k - j * g; */ + t2 = glmm_fnmadd(x4, x3, t2); + + /* t1[3] = i * p - m * l; + t1[3] = i * p - m * l; + t2[3] = e * p - m * h; + t3[3] = e * l - i * h; */ + t3 = glmm_fnmadd(x7, x0, t3); + + /* t1[4] = i * o - m * k; + t1[4] = i * o - m * k; + t2[4] = e * o - m * g; + t3[4] = e * k - i * g; */ + t4 = glmm_fnmadd(x7, x3, t4); + + /* t1[5] = i * n - m * j; + t1[5] = i * n - m * j; + t2[5] = e * n - m * f; + t3[5] = e * j - i * f; */ + t5 = glmm_fnmadd(x7, x5, t5); + + x4 = _mm_movelh_ps(r0, r1); /* f e b a */ + x5 = _mm_movehl_ps(r1, r0); /* h g d c */ + + x0 = glmm_shuff1(x4, 0, 0, 0, 2); /* a a a e */ + x1 = glmm_shuff1(x4, 1, 1, 1, 3); /* b b b f */ + x2 = glmm_shuff1(x5, 0, 0, 0, 2); /* c c c g */ + x3 = glmm_shuff1(x5, 1, 1, 1, 3); /* d d d h */ + + v2 = _mm_mul_ps(x0, t1); + v1 = _mm_mul_ps(x0, t0); + v3 = _mm_mul_ps(x0, t2); + v0 = _mm_mul_ps(x1, t0); + + v2 = glmm_fnmadd(x1, t3, v2); + v3 = glmm_fnmadd(x1, t4, v3); + v0 = glmm_fnmadd(x2, t1, v0); + v1 = glmm_fnmadd(x2, t3, v1); + + v3 = glmm_fmadd(x2, t5, v3); + v0 = glmm_fmadd(x3, t2, v0); + v2 = glmm_fmadd(x3, t5, v2); + v1 = glmm_fmadd(x3, t4, v1); + + /* + dest[0][0] = f * t1[0] - g * t1[1] + h * t1[2]; + dest[0][1] =-(b * t1[0] - c * t1[1] + d * t1[2]); + dest[0][2] = b * t2[0] - c * t2[1] + d * t2[2]; + dest[0][3] =-(b * t3[0] - c * t3[1] + d * t3[2]); */ + v0 = _mm_xor_ps(v0, x8); + + /* + dest[2][0] = e * t1[1] - f * t1[3] + h * t1[5]; + dest[2][1] =-(a * t1[1] - b * t1[3] + d * t1[5]); + dest[2][2] = a * t2[1] - b * t2[3] + d * t2[5]; + dest[2][3] =-(a * t3[1] - b * t3[3] + d * t3[5]);*/ + v2 = _mm_xor_ps(v2, x8); + + /* + dest[1][0] =-(e * t1[0] - g * t1[3] + h * t1[4]); + dest[1][1] = a * t1[0] - c * t1[3] + d * t1[4]; + dest[1][2] =-(a * t2[0] - c * t2[3] + d * t2[4]); + dest[1][3] = a * t3[0] - c * t3[3] + d * t3[4]; */ + v1 = _mm_xor_ps(v1, x9); + + /* + dest[3][0] =-(e * t1[2] - f * t1[4] + g * t1[5]); + dest[3][1] = a * t1[2] - b * t1[4] + c * t1[5]; + dest[3][2] =-(a * t2[2] - b * t2[4] + c * t2[5]); + dest[3][3] = a * t3[2] - b * t3[4] + c * t3[5]; */ + v3 = _mm_xor_ps(v3, x9); + + /* determinant */ + x0 = _mm_shuffle_ps(v0, v1, _MM_SHUFFLE(0, 0, 0, 0)); + x1 = _mm_shuffle_ps(v2, v3, _MM_SHUFFLE(0, 0, 0, 0)); + x0 = _mm_shuffle_ps(x0, x1, _MM_SHUFFLE(2, 0, 2, 0)); + + x0 = _mm_rcp_ps(glmm_vhadd(_mm_mul_ps(x0, r0))); + + glmm_store(dest[0], _mm_mul_ps(v0, x0)); + glmm_store(dest[1], _mm_mul_ps(v1, x0)); + glmm_store(dest[2], _mm_mul_ps(v2, x0)); + glmm_store(dest[3], _mm_mul_ps(v3, x0)); +} + +/* old one */ +#if 0 +CGLM_INLINE +void +glm_mat4_inv_sse2(mat4 mat, mat4 dest) { + __m128 r0, r1, r2, r3, + v0, v1, v2, v3, + t0, t1, t2, t3, t4, t5, + x0, x1, x2, x3, x4, x5, x6, x7, x8, x9; + + /* x8 = _mm_set_ps(-0.f, 0.f, -0.f, 0.f); */ + x8 = glmm_float32x4_SIGNMASK_NPNP; + x9 = glmm_shuff1(x8, 2, 1, 2, 1); + + /* 127 <- 0 */ + r0 = glmm_load(mat[0]); /* d c b a */ + r1 = glmm_load(mat[1]); /* h g f e */ + r2 = glmm_load(mat[2]); /* l k j i */ + r3 = glmm_load(mat[3]); /* p o n m */ + + x0 = _mm_movehl_ps(r3, r2); /* p o l k */ + x3 = _mm_movelh_ps(r2, r3); /* n m j i */ + x1 = glmm_shuff1(x0, 1, 3, 3 ,3); /* l p p p */ + x2 = glmm_shuff1(x0, 0, 2, 2, 2); /* k o o o */ + x4 = glmm_shuff1(x3, 1, 3, 3, 3); /* j n n n */ + x7 = glmm_shuff1(x3, 0, 2, 2, 2); /* i m m m */ + + x6 = _mm_shuffle_ps(r2, r1, _MM_SHUFFLE(0, 0, 0, 0)); /* e e i i */ + x5 = _mm_shuffle_ps(r2, r1, _MM_SHUFFLE(1, 1, 1, 1)); /* f f j j */ + x3 = _mm_shuffle_ps(r2, r1, _MM_SHUFFLE(2, 2, 2, 2)); /* g g k k */ + x0 = _mm_shuffle_ps(r2, r1, _MM_SHUFFLE(3, 3, 3, 3)); /* h h l l */ + + t0 = _mm_mul_ps(x3, x1); + t1 = _mm_mul_ps(x5, x1); + t2 = _mm_mul_ps(x5, x2); + t3 = _mm_mul_ps(x6, x1); + t4 = _mm_mul_ps(x6, x2); + t5 = _mm_mul_ps(x6, x4); + + /* t1[0] = k * p - o * l; + t1[0] = k * p - o * l; + t2[0] = g * p - o * h; + t3[0] = g * l - k * h; */ + t0 = glmm_fnmadd(x2, x0, t0); + + /* t1[1] = j * p - n * l; + t1[1] = j * p - n * l; + t2[1] = f * p - n * h; + t3[1] = f * l - j * h; */ + t1 = glmm_fnmadd(x4, x0, t1); + + /* t1[2] = j * o - n * k + t1[2] = j * o - n * k; + t2[2] = f * o - n * g; + t3[2] = f * k - j * g; */ + t2 = glmm_fnmadd(x4, x3, t2); + + /* t1[3] = i * p - m * l; + t1[3] = i * p - m * l; + t2[3] = e * p - m * h; + t3[3] = e * l - i * h; */ + t3 = glmm_fnmadd(x7, x0, t3); + + /* t1[4] = i * o - m * k; + t1[4] = i * o - m * k; + t2[4] = e * o - m * g; + t3[4] = e * k - i * g; */ + t4 = glmm_fnmadd(x7, x3, t4); + + /* t1[5] = i * n - m * j; + t1[5] = i * n - m * j; + t2[5] = e * n - m * f; + t3[5] = e * j - i * f; */ + t5 = glmm_fnmadd(x7, x5, t5); + + x4 = _mm_movelh_ps(r0, r1); /* f e b a */ + x5 = _mm_movehl_ps(r1, r0); /* h g d c */ + + x0 = glmm_shuff1(x4, 0, 0, 0, 2); /* a a a e */ + x1 = glmm_shuff1(x4, 1, 1, 1, 3); /* b b b f */ + x2 = glmm_shuff1(x5, 0, 0, 0, 2); /* c c c g */ + x3 = glmm_shuff1(x5, 1, 1, 1, 3); /* d d d h */ + + v2 = _mm_mul_ps(x0, t1); + v1 = _mm_mul_ps(x0, t0); + v3 = _mm_mul_ps(x0, t2); + v0 = _mm_mul_ps(x1, t0); + + v2 = glmm_fnmadd(x1, t3, v2); + v3 = glmm_fnmadd(x1, t4, v3); + v0 = glmm_fnmadd(x2, t1, v0); + v1 = glmm_fnmadd(x2, t3, v1); + + v3 = glmm_fmadd(x2, t5, v3); + v0 = glmm_fmadd(x3, t2, v0); + v2 = glmm_fmadd(x3, t5, v2); + v1 = glmm_fmadd(x3, t4, v1); + + /* + dest[0][0] = f * t1[0] - g * t1[1] + h * t1[2]; + dest[0][1] =-(b * t1[0] - c * t1[1] + d * t1[2]); + dest[0][2] = b * t2[0] - c * t2[1] + d * t2[2]; + dest[0][3] =-(b * t3[0] - c * t3[1] + d * t3[2]); */ + v0 = _mm_xor_ps(v0, x8); + + /* + dest[2][0] = e * t1[1] - f * t1[3] + h * t1[5]; + dest[2][1] =-(a * t1[1] - b * t1[3] + d * t1[5]); + dest[2][2] = a * t2[1] - b * t2[3] + d * t2[5]; + dest[2][3] =-(a * t3[1] - b * t3[3] + d * t3[5]);*/ + v2 = _mm_xor_ps(v2, x8); + + /* + dest[1][0] =-(e * t1[0] - g * t1[3] + h * t1[4]); + dest[1][1] = a * t1[0] - c * t1[3] + d * t1[4]; + dest[1][2] =-(a * t2[0] - c * t2[3] + d * t2[4]); + dest[1][3] = a * t3[0] - c * t3[3] + d * t3[4]; */ + v1 = _mm_xor_ps(v1, x9); + + /* + dest[3][0] =-(e * t1[2] - f * t1[4] + g * t1[5]); + dest[3][1] = a * t1[2] - b * t1[4] + c * t1[5]; + dest[3][2] =-(a * t2[2] - b * t2[4] + c * t2[5]); + dest[3][3] = a * t3[2] - b * t3[4] + c * t3[5]; */ + v3 = _mm_xor_ps(v3, x9); + + /* determinant */ + x0 = _mm_shuffle_ps(v0, v1, _MM_SHUFFLE(0, 0, 0, 0)); + x1 = _mm_shuffle_ps(v2, v3, _MM_SHUFFLE(0, 0, 0, 0)); + x0 = _mm_shuffle_ps(x0, x1, _MM_SHUFFLE(2, 0, 2, 0)); + + x0 = _mm_div_ps(glmm_set1(1.0f), glmm_vhadd(_mm_mul_ps(x0, r0))); + + glmm_store(dest[0], _mm_mul_ps(v0, x0)); + glmm_store(dest[1], _mm_mul_ps(v1, x0)); + glmm_store(dest[2], _mm_mul_ps(v2, x0)); + glmm_store(dest[3], _mm_mul_ps(v3, x0)); +} +#endif + +CGLM_INLINE +void +glm_mat4_inv_sse2(mat4 mat, mat4 dest) { + __m128 r0, r1, r2, r3, s1, s2, + v0, v1, v2, v3, v4, v5, + t0, t1, t2, + x0, x1, x2, x3, x4, x5, x6, x7, x8, x9, x10, x11, x12, x13; + + /* s1 = _mm_set_ps(-0.f, 0.f, -0.f, 0.f); */ + s1 = glmm_float32x4_SIGNMASK_NPNP; + s2 = glmm_shuff1(s1, 2, 1, 2, 1); + + /* 127 <- 0 */ + r1 = glmm_load(mat[1]); /* h g f e */ + r0 = glmm_load(mat[0]); /* d c b a */ + r3 = glmm_load(mat[3]); /* p o n m */ + r2 = glmm_load(mat[2]); /* l k j i */ + + x4 = _mm_unpackhi_ps(r0, r2); /* l d k c */ + x5 = _mm_unpacklo_ps(r0, r2); /* j b i a */ + x6 = _mm_unpackhi_ps(r1, r3); /* p h o g */ + x7 = _mm_unpacklo_ps(r1, r3); /* n f m e */ + + x0 = _mm_unpackhi_ps(x7, x5); /* j n b f */ + x1 = _mm_unpacklo_ps(x7, x5); /* i m a e */ + x2 = _mm_unpackhi_ps(x6, x4); /* l p d h */ + x3 = _mm_unpacklo_ps(x6, x4); /* k o c g */ + + /* c2 = c * h - d * g c12 = a * g - c * e c8 = a * f - b * e + c1 = k * p - l * o c11 = i * o - k * m c7 = i * n - j * m + c4 = a * h - d * e c6 = b * h - d * f c10 = b * g - c * f + c3 = i * p - l * m c5 = j * p - l * n c9 = j * o - k * n */ + + x8 = _mm_shuffle_ps(x0, x3, _MM_SHUFFLE(3, 1, 3, 1)); /* k c j b */ + x9 = _mm_shuffle_ps(x0, x3, _MM_SHUFFLE(2, 0, 2, 0)); /* o g n f */ + + x10 = glmm_shuff1(x2, 2, 0, 2, 0); /* p h p h */ + x11 = glmm_shuff1(x2, 3, 1, 3, 1); /* l d l d */ + +#if 0 /* TODO measure both */ + x12 = _mm_shuffle_ps(x4, x5, _MM_SHUFFLE(1, 0, 1, 0)); /* i a k c */ + x13 = _mm_shuffle_ps(x6, x7, _MM_SHUFFLE(1, 0, 1, 0)); /* m e o g */ +#else + x12 = _mm_movelh_ps(x4, x5); /* i a k c */ + x13 = _mm_movelh_ps(x6, x7); /* m e o g */ +#endif + + t0 = _mm_mul_ps(x12, x10); + t1 = _mm_mul_ps(x5, x6); + t2 = _mm_mul_ps(x5, x9); + + t0 = glmm_fnmadd(x11, x13, t0); + t1 = glmm_fnmadd(x4, x7, t1); + t2 = glmm_fnmadd(x8, x7, t2); + + /* det */ + /* v0: c3 * c10 + c4 * c9 + c1 * c8 + c2 * c7 */ + /* v1: c5 * c12 + c6 * c11 */ + + v5 = glmm_set1_rval(1.0f); + v0 = glmm_shuff1(t2, 2, 3, 0, 1); + v1 = glmm_shuff1(t1, 0, 1, 2, 3); + v0 = _mm_mul_ps(t0, v0); + v1 = _mm_mul_ps(t1, v1); + v2 = glmm_shuff1(v1, 1, 0, 0, 1); + v3 = glmm_shuff1(v0, 0, 1, 2, 3); + v1 = _mm_add_ps(v1, v2); + v0 = _mm_add_ps(v0, v3); + v2 = glmm_shuff1(v0, 1, 0, 0, 1); + v0 = _mm_add_ps(v0, v2); + + v0 = _mm_sub_ps(v0, v1); /* det */ + v0 = _mm_div_ps(v5, v0); /* idt */ + + /* multiply t0,t1,t2 by idt to reduce 1mul below: 2eor+4mul vs 3mul+4eor */ + t0 = _mm_mul_ps(t0, v0); + t1 = _mm_mul_ps(t1, v0); + t2 = _mm_mul_ps(t2, v0); + + v0 = glmm_shuff1(t0, 0, 0, 1, 1); /* c2 c2 c1 c1 */ + v1 = glmm_shuff1(t0, 2, 2, 3, 3); /* c4 c4 c3 c3 */ + v2 = glmm_shuff1(t1, 0, 0, 1, 1); /* c12 c12 c11 c11 */ + v3 = glmm_shuff1(t1, 2, 2, 3, 3); /* c6 c6 c5 c5 */ + v4 = glmm_shuff1(t2, 0, 0, 1, 1); /* c8 c8 c7 c7 */ + v5 = glmm_shuff1(t2, 2, 2, 3, 3); /* c10 c10 c9 c9 */ + + /* result */ + + /* dest[0][0] = (f * c1 - g * c5 + h * c9) * idt; + dest[0][1] = (b * c1 - c * c5 + d * c9) * ndt; + dest[0][2] = (n * c2 - o * c6 + p * c10) * idt; + dest[0][3] = (j * c2 - k * c6 + l * c10) * ndt; + + dest[1][0] = (e * c1 - g * c3 + h * c11) * ndt; + dest[1][1] = (a * c1 - c * c3 + d * c11) * idt; + dest[1][2] = (m * c2 - o * c4 + p * c12) * ndt; + dest[1][3] = (i * c2 - k * c4 + l * c12) * idt; + + dest[2][0] = (e * c5 - f * c3 + h * c7) * idt; + dest[2][1] = (a * c5 - b * c3 + d * c7) * ndt; + dest[2][2] = (m * c6 - n * c4 + p * c8) * idt; + dest[2][3] = (i * c6 - j * c4 + l * c8) * ndt; + + dest[3][0] = (e * c9 - f * c11 + g * c7) * ndt; + dest[3][1] = (a * c9 - b * c11 + c * c7) * idt; + dest[3][2] = (m * c10 - n * c12 + o * c8) * ndt; + dest[3][3] = (i * c10 - j * c12 + k * c8) * idt; */ + + r0 = _mm_mul_ps(x0, v0); + r1 = _mm_mul_ps(x1, v0); + r2 = _mm_mul_ps(x1, v3); + r3 = _mm_mul_ps(x1, v5); + + r0 = glmm_fnmadd(x3, v3, r0); + r1 = glmm_fnmadd(x3, v1, r1); + r2 = glmm_fnmadd(x0, v1, r2); + r3 = glmm_fnmadd(x0, v2, r3); + + r0 = glmm_fmadd(x2, v5, r0); + r1 = glmm_fmadd(x2, v2, r1); + r2 = glmm_fmadd(x2, v4, r2); + r3 = glmm_fmadd(x3, v4, r3); + + /* 4xor may be fastart then 4mul, see above */ + r0 = _mm_xor_ps(r0, s1); + r1 = _mm_xor_ps(r1, s2); + r2 = _mm_xor_ps(r2, s1); + r3 = _mm_xor_ps(r3, s2); + + glmm_store(dest[0], r0); + glmm_store(dest[1], r1); + glmm_store(dest[2], r2); + glmm_store(dest[3], r3); +} +#endif +#endif /* cglm_mat_sse_h */ diff --git a/cglm/include/cglm/simd/sse2/quat.h b/cglm/include/cglm/simd/sse2/quat.h new file mode 100644 index 0000000..def0fe2 --- /dev/null +++ b/cglm/include/cglm/simd/sse2/quat.h @@ -0,0 +1,54 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_quat_simd_h +#define cglm_quat_simd_h +#if defined( __SSE__ ) || defined( __SSE2__ ) + +#include "../../common.h" +#include "../intrin.h" + +CGLM_INLINE +void +glm_quat_mul_sse2(versor p, versor q, versor dest) { + /* + + (a1 b2 + b1 a2 + c1 d2 − d1 c2)i + + (a1 c2 − b1 d2 + c1 a2 + d1 b2)j + + (a1 d2 + b1 c2 − c1 b2 + d1 a2)k + a1 a2 − b1 b2 − c1 c2 − d1 d2 + */ + + __m128 xp, xq, x1, x2, x3, r, x, y, z; + + xp = glmm_load(p); /* 3 2 1 0 */ + xq = glmm_load(q); + x1 = glmm_float32x4_SIGNMASK_NPNP; /* TODO: _mm_set1_ss() + shuff ? */ + r = _mm_mul_ps(glmm_splat_w(xp), xq); + + x2 = _mm_unpackhi_ps(x1, x1); + x3 = glmm_shuff1(x1, 3, 2, 0, 1); + x = glmm_splat_x(xp); + y = glmm_splat_y(xp); + z = glmm_splat_z(xp); + + x = _mm_xor_ps(x, x1); + y = _mm_xor_ps(y, x2); + z = _mm_xor_ps(z, x3); + + x1 = glmm_shuff1(xq, 0, 1, 2, 3); + x2 = glmm_shuff1(xq, 1, 0, 3, 2); + x3 = glmm_shuff1(xq, 2, 3, 0, 1); + + r = glmm_fmadd(x, x1, r); + r = glmm_fmadd(y, x2, r); + r = glmm_fmadd(z, x3, r); + + glmm_store(dest, r); +} + +#endif +#endif /* cglm_quat_simd_h */ diff --git a/cglm/include/cglm/simd/wasm.h b/cglm/include/cglm/simd/wasm.h new file mode 100644 index 0000000..69f8301 --- /dev/null +++ b/cglm/include/cglm/simd/wasm.h @@ -0,0 +1,197 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_simd_wasm_h +#define cglm_simd_wasm_h +#include "intrin.h" +#ifdef CGLM_SIMD_WASM +#include + +#define glmm_load(p) wasm_v128_load(p) +#define glmm_store(p, a) wasm_v128_store(p, (a)) + +#define glmm_set1(x) wasm_f32x4_splat(x) +#define glmm_set1_rval(x) wasm_f32x4_splat(x) +#define glmm_128 v128_t + +#define glmm_shuff1(xmm, z, y, x, w) wasm_i32x4_shuffle(xmm, xmm, w, x, y, z) + +#define glmm_splat(x, lane) glmm_shuff1(x, lane, lane, lane, lane) + +#define glmm_splat_x(x) glmm_splat(x, 0) +#define glmm_splat_y(x) glmm_splat(x, 1) +#define glmm_splat_z(x) glmm_splat(x, 2) +#define glmm_splat_w(x) glmm_splat(x, 3) + +#define GLMM_NEGZEROf 0x80000000 /* 0x80000000 ---> -0.0f */ + +/* _mm_set_ps(X, Y, Z, W); */ +#define GLMM__SIGNMASKf(X, Y, Z, W) wasm_i32x4_const(X, Y, Z, W) + +#define glmm_float32x4_SIGNMASK_PNPN GLMM__SIGNMASKf(0, GLMM_NEGZEROf, 0, GLMM_NEGZEROf) +#define glmm_float32x4_SIGNMASK_NPNP GLMM__SIGNMASKf(GLMM_NEGZEROf, 0, GLMM_NEGZEROf, 0) +#define glmm_float32x4_SIGNMASK_NPPN GLMM__SIGNMASKf(GLMM_NEGZEROf, 0, 0, GLMM_NEGZEROf) +#define glmm_float32x4_SIGNMASK_NEG wasm_i32x4_const_splat(GLMM_NEGZEROf) + +static inline glmm_128 glmm_abs(glmm_128 x) { return wasm_f32x4_abs(x); } +static inline glmm_128 glmm_min(glmm_128 a, glmm_128 b) { return wasm_f32x4_pmin(b, a); } +static inline glmm_128 glmm_max(glmm_128 a, glmm_128 b) { return wasm_f32x4_pmax(b, a); } + +static inline +glmm_128 +glmm_vhadd(glmm_128 v) { + glmm_128 x0; + x0 = wasm_f32x4_add(v, glmm_shuff1(v, 0, 1, 2, 3)); + x0 = wasm_f32x4_add(x0, glmm_shuff1(x0, 1, 0, 0, 1)); + return x0; +} + +static inline +glmm_128 +glmm_vhadds(glmm_128 v) { + glmm_128 shuf, sums; + shuf = glmm_shuff1(v, 2, 3, 0, 1); + sums = wasm_f32x4_add(v, shuf); + /* shuf = _mm_movehl_ps(shuf, sums); */ + shuf = wasm_i32x4_shuffle(shuf, sums, 6, 7, 2, 3); + sums = wasm_i32x4_shuffle(sums, wasm_f32x4_add(sums, shuf), 4, 1, 2, 3); + return sums; +} + +static inline +float +glmm_hadd(glmm_128 v) { + return wasm_f32x4_extract_lane(glmm_vhadds(v), 0); +} + +static inline +glmm_128 +glmm_vhmin(glmm_128 v) { + glmm_128 x0, x1, x2; + x0 = glmm_shuff1(v, 2, 3, 2, 3); /* [2, 3, 2, 3] */ + x1 = wasm_f32x4_pmin(x0, v); /* [0|2, 1|3, 2|2, 3|3] */ + x2 = glmm_splat(x1, 1); /* [1|3, 1|3, 1|3, 1|3] */ + return wasm_f32x4_pmin(x1, x2); +} + +static inline +float +glmm_hmin(glmm_128 v) { + return wasm_f32x4_extract_lane(glmm_vhmin(v), 0); +} + +static inline +glmm_128 +glmm_vhmax(glmm_128 v) { + glmm_128 x0, x1, x2; + x0 = glmm_shuff1(v, 2, 3, 2, 3); /* [2, 3, 2, 3] */ + x1 = wasm_f32x4_pmax(x0, v); /* [0|2, 1|3, 2|2, 3|3] */ + x2 = glmm_splat(x1, 1); /* [1|3, 1|3, 1|3, 1|3] */ + /* _mm_max_ss */ + return wasm_i32x4_shuffle(x1, wasm_f32x4_pmax(x1, x2), 4, 1, 2, 3); +} + +static inline +float +glmm_hmax(glmm_128 v) { + return wasm_f32x4_extract_lane(glmm_vhmax(v), 0); +} + +static inline +glmm_128 +glmm_vdots(glmm_128 a, glmm_128 b) { + return glmm_vhadds(wasm_f32x4_mul(a, b)); +} + +static inline +glmm_128 +glmm_vdot(glmm_128 a, glmm_128 b) { + glmm_128 x0; + x0 = wasm_f32x4_mul(a, b); + x0 = wasm_f32x4_add(x0, glmm_shuff1(x0, 1, 0, 3, 2)); + return wasm_f32x4_add(x0, glmm_shuff1(x0, 0, 1, 0, 1)); +} + +static inline +float +glmm_dot(glmm_128 a, glmm_128 b) { + return wasm_f32x4_extract_lane(glmm_vdots(a, b), 0); +} + +static inline +float +glmm_norm(glmm_128 a) { + glmm_128 x0; + x0 = glmm_vhadds(wasm_f32x4_mul(a, a)); + return wasm_f32x4_extract_lane( + wasm_i32x4_shuffle(x0, wasm_f32x4_sqrt(x0),4, 1, 2, 3), 0); +} + +static inline +float +glmm_norm2(glmm_128 a) { + return wasm_f32x4_extract_lane(glmm_vhadds(wasm_f32x4_mul(a, a)), 0); +} + +static inline +float +glmm_norm_one(glmm_128 a) { + return wasm_f32x4_extract_lane(glmm_vhadds(glmm_abs(a)), 0); +} + +static inline +float +glmm_norm_inf(glmm_128 a) { + return wasm_f32x4_extract_lane(glmm_vhmax(glmm_abs(a)), 0); +} + +static inline +glmm_128 +glmm_load3(float v[3]) { + glmm_128 xy = wasm_v128_load64_zero(v); + return wasm_f32x4_replace_lane(xy, 2, v[2]); +} + +static inline +void +glmm_store3(float v[3], glmm_128 vx) { + wasm_v128_store64_lane(v, vx, 0); + wasm_v128_store32_lane(&v[2], vx, 2); +} + +static inline +glmm_128 +glmm_div(glmm_128 a, glmm_128 b) { + return wasm_f32x4_div(a, b); +} + +static inline +glmm_128 +glmm_fmadd(glmm_128 a, glmm_128 b, glmm_128 c) { + return wasm_f32x4_add(c, wasm_f32x4_mul(a, b)); +} + +static inline +glmm_128 +glmm_fnmadd(glmm_128 a, glmm_128 b, glmm_128 c) { + return wasm_f32x4_sub(c, wasm_f32x4_mul(a, b)); +} + +static inline +glmm_128 +glmm_fmsub(glmm_128 a, glmm_128 b, glmm_128 c) { + return wasm_f32x4_sub(wasm_f32x4_mul(a, b), c); +} + +static inline +glmm_128 +glmm_fnmsub(glmm_128 a, glmm_128 b, glmm_128 c) { + return wasm_f32x4_neg(wasm_f32x4_add(wasm_f32x4_mul(a, b), c)); +} + +#endif +#endif /* cglm_simd_wasm_h */ diff --git a/cglm/include/cglm/simd/wasm/affine.h b/cglm/include/cglm/simd/wasm/affine.h new file mode 100644 index 0000000..80b98fb --- /dev/null +++ b/cglm/include/cglm/simd/wasm/affine.h @@ -0,0 +1,127 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_affine_mat_wasm_h +#define cglm_affine_mat_wasm_h +#if defined(__wasm__) && defined(__wasm_simd128__) + +#include "../../common.h" +#include "../intrin.h" + +CGLM_INLINE +void +glm_mul_wasm(mat4 m1, mat4 m2, mat4 dest) { + /* D = R * L (Column-Major) */ + glmm_128 l, r0, r1, r2, r3, v0, v1, v2, v3; + + l = glmm_load(m1[0]); + r0 = glmm_load(m2[0]); + r1 = glmm_load(m2[1]); + r2 = glmm_load(m2[2]); + r3 = glmm_load(m2[3]); + + v0 = wasm_f32x4_mul(glmm_splat_x(r0), l); + v1 = wasm_f32x4_mul(glmm_splat_x(r1), l); + v2 = wasm_f32x4_mul(glmm_splat_x(r2), l); + v3 = wasm_f32x4_mul(glmm_splat_x(r3), l); + + l = glmm_load(m1[1]); + v0 = glmm_fmadd(glmm_splat_y(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_y(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_y(r2), l, v2); + v3 = glmm_fmadd(glmm_splat_y(r3), l, v3); + + l = glmm_load(m1[2]); + v0 = glmm_fmadd(glmm_splat_z(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_z(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_z(r2), l, v2); + v3 = glmm_fmadd(glmm_splat_z(r3), l, v3); + + l = glmm_load(m1[3]); + v3 = glmm_fmadd(glmm_splat_w(r3), l, v3); + + glmm_store(dest[0], v0); + glmm_store(dest[1], v1); + glmm_store(dest[2], v2); + glmm_store(dest[3], v3); +} + +CGLM_INLINE +void +glm_mul_rot_wasm(mat4 m1, mat4 m2, mat4 dest) { + /* D = R * L (Column-Major) */ + + glmm_128 l, r0, r1, r2, v0, v1, v2; + + l = glmm_load(m1[0]); + r0 = glmm_load(m2[0]); + r1 = glmm_load(m2[1]); + r2 = glmm_load(m2[2]); + + v0 = wasm_f32x4_mul(glmm_splat_x(r0), l); + v1 = wasm_f32x4_mul(glmm_splat_x(r1), l); + v2 = wasm_f32x4_mul(glmm_splat_x(r2), l); + + l = glmm_load(m1[1]); + v0 = glmm_fmadd(glmm_splat_y(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_y(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_y(r2), l, v2); + + l = glmm_load(m1[2]); + v0 = glmm_fmadd(glmm_splat_z(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_z(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_z(r2), l, v2); + + glmm_store(dest[0], v0); + glmm_store(dest[1], v1); + glmm_store(dest[2], v2); + glmm_store(dest[3], glmm_load(m1[3])); +} + +CGLM_INLINE +void +glm_inv_tr_wasm(mat4 mat) { + glmm_128 r0, r1, r2, r3, x0, x1, x2, x3, x4, x5; + + r0 = glmm_load(mat[0]); + r1 = glmm_load(mat[1]); + r2 = glmm_load(mat[2]); + r3 = glmm_load(mat[3]); + x1 = wasm_f32x4_const(0.0f, 0.0f, 0.0f, 1.0f); + + /* _MM_TRANSPOSE4_PS(r0, r1, r2, x1); */ + x2 = wasm_i32x4_shuffle(r0, r1, 0, 4, 1, 5); + x3 = wasm_i32x4_shuffle(r0, r1, 2, 6, 3, 7); + x4 = wasm_i32x4_shuffle(r2, x1, 0, 4, 1, 5); + x5 = wasm_i32x4_shuffle(r2, x1, 2, 6, 3, 7); + /* r0 = _mm_movelh_ps(x2, x4); */ + r0 = wasm_i32x4_shuffle(x2, x4, 0, 1, 4, 5); + /* r1 = _mm_movehl_ps(x4, x2); */ + r1 = wasm_i32x4_shuffle(x4, x2, 6, 7, 2, 3); + /* r2 = _mm_movelh_ps(x3, x5); */ + r2 = wasm_i32x4_shuffle(x3, x5, 0, 1, 4, 5); + /* x1 = _mm_movehl_ps(x5, x3); */ + x1 = wasm_i32x4_shuffle(x5, x3, 6, 7, 2, 3); + + x2 = glmm_shuff1(r3, 0, 0, 0, 0); + x3 = glmm_shuff1(r3, 1, 1, 1, 1); + x4 = glmm_shuff1(r3, 2, 2, 2, 2); + + x0 = glmm_fmadd(r0, x2, + glmm_fmadd(r1, x3, wasm_f32x4_mul(r2, x4))); + x0 = wasm_f32x4_neg(x0); + + x0 = wasm_f32x4_add(x0, x1); + + glmm_store(mat[0], r0); + glmm_store(mat[1], r1); + glmm_store(mat[2], r2); + glmm_store(mat[3], x0); +} + +#endif +#endif /* cglm_affine_mat_wasm_h */ diff --git a/cglm/include/cglm/simd/wasm/mat2.h b/cglm/include/cglm/simd/wasm/mat2.h new file mode 100644 index 0000000..80ce0fb --- /dev/null +++ b/cglm/include/cglm/simd/wasm/mat2.h @@ -0,0 +1,50 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_mat2_wasm_h +#define cglm_mat2_wasm_h +#if defined(__wasm__) && defined(__wasm_simd128__) + +#include "../../common.h" +#include "../intrin.h" + +CGLM_INLINE +void +glm_mat2_mul_wasm(mat2 m1, mat2 m2, mat2 dest) { + glmm_128 x0, x1, x2, x3, x4; + + x1 = glmm_load(m1[0]); /* d c b a */ + x2 = glmm_load(m2[0]); /* h g f e */ + + x3 = glmm_shuff1(x2, 2, 2, 0, 0); + x4 = glmm_shuff1(x2, 3, 3, 1, 1); + /* x0 = _mm_movelh_ps(x1, x1); */ + x0 = wasm_i32x4_shuffle(x1, x1, 0, 1, 4, 5); + /* x2 = _mm_movehl_ps(x1, x1); */ + x2 = wasm_i32x4_shuffle(x1, x1, 6, 7, 2, 3); + + /* + dest[0][0] = a * e + c * f; + dest[0][1] = b * e + d * f; + dest[1][0] = a * g + c * h; + dest[1][1] = b * g + d * h; + */ + x0 = glmm_fmadd(x0, x3, wasm_f32x4_mul(x2, x4)); + + glmm_store(dest[0], x0); +} + +CGLM_INLINE +void +glm_mat2_transp_wasm(mat2 m, mat2 dest) { + /* d c b a */ + /* d b c a */ + glmm_store(dest[0], glmm_shuff1(glmm_load(m[0]), 3, 1, 2, 0)); +} + +#endif +#endif /* cglm_mat2_wasm_h */ diff --git a/cglm/include/cglm/simd/wasm/mat3.h b/cglm/include/cglm/simd/wasm/mat3.h new file mode 100644 index 0000000..dfe192d --- /dev/null +++ b/cglm/include/cglm/simd/wasm/mat3.h @@ -0,0 +1,85 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_mat3_wasm_h +#define cglm_mat3_wasm_h +#if defined(__wasm__) && defined(__wasm_simd128__) + +#include "../../common.h" +#include "../intrin.h" + +CGLM_INLINE +void +glm_mat3_mul_wasm(mat3 m1, mat3 m2, mat3 dest) { + glmm_128 l0, l1, l2, r0, r1, r2, x0, x1, x2, x3, x4, x5, x6, x7, x8, x9; + + l0 = wasm_v128_load(m1[0]); + l1 = wasm_v128_load(&m1[1][1]); + + r0 = wasm_v128_load(m2[0]); + r1 = wasm_v128_load(&m2[1][1]); + + x8 = glmm_shuff1(l0, 0, 2, 1, 0); /* a00 a02 a01 a00 */ + x1 = glmm_shuff1(r0, 3, 0, 0, 0); /* b10 b00 b00 b00 */ + x2 = wasm_i32x4_shuffle(l0, l1, 3, 3, 4, 5); /* a12 a11 a10 a10 */ + x3 = wasm_i32x4_shuffle(r0, r1, 1, 3, 4, 6); /* b20 b11 b10 b01 */ + x0 = wasm_f32x4_mul(x8, x1); + + x6 = glmm_shuff1(l0, 1, 0, 2, 1); /* a01 a00 a02 a01 */ + x7 = glmm_shuff1(x3, 3, 3, 1, 1); /* b20 b20 b10 b10 */ + l2 = wasm_v128_load32_zero(&m1[2][2]); + r2 = wasm_v128_load32_zero(&m2[2][2]); + x1 = wasm_f32x4_mul(x6, x7); + l2 = glmm_shuff1(l2, 0, 0, 1, 0); /* a22 a22 0.f a22 */ + r2 = glmm_shuff1(r2, 0, 0, 1, 0); /* b22 b22 0.f b22 */ + + x4 = glmm_shuff1(x2, 0, 3, 2, 0); /* a10 a12 a11 a10 */ + x5 = glmm_shuff1(x2, 2, 0, 3, 2); /* a11 a10 a12 a11 */ + x6 = glmm_shuff1(x3, 2, 0, 0, 0); /* b11 b01 b01 b01 */ + x2 = glmm_shuff1(r1, 3, 3, 0, 0); /* b21 b21 b11 b11 */ + + /* x8 = _mm_unpackhi_ps(x8, x4); */ + /* x9 = _mm_unpackhi_ps(x7, x2); */ + x8 = wasm_i32x4_shuffle(x8, x4, 2, 6, 3, 7); /* a10 a00 a12 a02 */ + x9 = wasm_i32x4_shuffle(x7, x2, 2, 6, 3, 7); /* b21 b20 b21 b20 */ + + x0 = glmm_fmadd(x4, x6, x0); + x1 = glmm_fmadd(x5, x2, x1); + + /* x2 = _mm_movehl_ps(l2, l1); */ + x2 = wasm_i32x4_shuffle(l2, l1, 6, 7, 2, 3); /* a22 a22 a21 a20 */ + x3 = glmm_shuff1(x2, 0, 2, 1, 0); /* a20 a22 a21 a20 */ + x2 = glmm_shuff1(x2, 1, 0, 2, 1); /* a21 a20 a22 a21 */ + x4 = wasm_i32x4_shuffle(r0, r1, 2, 2, 5, 5); /* b12 b12 b02 b02 */ + + x5 = glmm_shuff1(x4, 3, 0, 0, 0); /* b12 b02 b02 b02 */ + /* x4 = _mm_movehl_ps(r2, x4); */ + x4 = wasm_i32x4_shuffle(r2, x4, 6, 7, 2, 3); /* b22 b22 b12 b12 */ + x0 = glmm_fmadd(x3, x5, x0); + x1 = glmm_fmadd(x2, x4, x1); + + /* + Dot Product : dest[2][2] = a02 * b20 + + a12 * b21 + + a22 * b22 + + 0 * 00 */ + /* x2 = _mm_movelh_ps(x8, l2); */ + /* x3 = _mm_movelh_ps(x9, r2); */ + x2 = wasm_i32x4_shuffle(x8, l2, 0, 1, 4, 5); /* 0.f a22 a12 a02 */ + x3 = wasm_i32x4_shuffle(x9, r2, 0, 1, 4, 5); /* 0.f b22 b21 b20 */ + x2 = glmm_vdots(x2, x3); + + /* _mm_storeu_ps(&dest[0][0], x0); */ + wasm_v128_store(&dest[0][0], x0); + /* _mm_storeu_ps(&dest[1][1], x1); */ + wasm_v128_store(&dest[1][1], x1); + /* _mm_store_ss (&dest[2][2], x2); */ + wasm_v128_store32_lane(&dest[2][2], x2, 0); +} + +#endif +#endif /* cglm_mat3_wasm_h */ diff --git a/cglm/include/cglm/simd/wasm/mat4.h b/cglm/include/cglm/simd/wasm/mat4.h new file mode 100644 index 0000000..79ed688 --- /dev/null +++ b/cglm/include/cglm/simd/wasm/mat4.h @@ -0,0 +1,454 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_mat_wasm_h +#define cglm_mat_wasm_h +#if defined(__wasm__) && defined(__wasm_simd128__) + +#include "../../common.h" +#include "../intrin.h" + +#define glm_mat4_inv_precise_wasm(mat, dest) glm_mat4_inv_wasm(mat, dest) + +CGLM_INLINE +void +glm_mat4_scale_wasm(mat4 m, float s) { + glmm_128 x0; + x0 = wasm_f32x4_splat(s); + + glmm_store(m[0], wasm_f32x4_mul(glmm_load(m[0]), x0)); + glmm_store(m[1], wasm_f32x4_mul(glmm_load(m[1]), x0)); + glmm_store(m[2], wasm_f32x4_mul(glmm_load(m[2]), x0)); + glmm_store(m[3], wasm_f32x4_mul(glmm_load(m[3]), x0)); +} + +CGLM_INLINE +void +glm_mat4_transp_wasm(mat4 m, mat4 dest) { + glmm_128 r0, r1, r2, r3, tmp0, tmp1, tmp2, tmp3; + + r0 = glmm_load(m[0]); + r1 = glmm_load(m[1]); + r2 = glmm_load(m[2]); + r3 = glmm_load(m[3]); + + /* _MM_TRANSPOSE4_PS(r0, r1, r2, r3); */ + tmp0 = wasm_i32x4_shuffle(r0, r1, 0, 4, 1, 5); + tmp1 = wasm_i32x4_shuffle(r0, r1, 2, 6, 3, 7); + tmp2 = wasm_i32x4_shuffle(r2, r3, 0, 4, 1, 5); + tmp3 = wasm_i32x4_shuffle(r2, r3, 2, 6, 3, 7); + /* r0 = _mm_movelh_ps(tmp0, tmp2); */ + r0 = wasm_i32x4_shuffle(tmp0, tmp2, 0, 1, 4, 5); + /* r1 = _mm_movehl_ps(tmp2, tmp0); */ + r1 = wasm_i32x4_shuffle(tmp2, tmp0, 6, 7, 2, 3); + /* r2 = _mm_movelh_ps(tmp1, tmp3); */ + r2 = wasm_i32x4_shuffle(tmp1, tmp3, 0, 1, 4, 5); + /* r3 = _mm_movehl_ps(tmp3, tmp1); */ + r3 = wasm_i32x4_shuffle(tmp3, tmp1, 6, 7, 2, 3); + + glmm_store(dest[0], r0); + glmm_store(dest[1], r1); + glmm_store(dest[2], r2); + glmm_store(dest[3], r3); +} + +CGLM_INLINE +void +glm_mat4_mul_wasm(mat4 m1, mat4 m2, mat4 dest) { + /* D = R * L (Column-Major) */ + + glmm_128 l, r0, r1, r2, r3, v0, v1, v2, v3; + + l = glmm_load(m1[0]); + r0 = glmm_load(m2[0]); + r1 = glmm_load(m2[1]); + r2 = glmm_load(m2[2]); + r3 = glmm_load(m2[3]); + + v0 = wasm_f32x4_mul(glmm_splat_x(r0), l); + v1 = wasm_f32x4_mul(glmm_splat_x(r1), l); + v2 = wasm_f32x4_mul(glmm_splat_x(r2), l); + v3 = wasm_f32x4_mul(glmm_splat_x(r3), l); + + l = glmm_load(m1[1]); + v0 = glmm_fmadd(glmm_splat_y(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_y(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_y(r2), l, v2); + v3 = glmm_fmadd(glmm_splat_y(r3), l, v3); + + l = glmm_load(m1[2]); + v0 = glmm_fmadd(glmm_splat_z(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_z(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_z(r2), l, v2); + v3 = glmm_fmadd(glmm_splat_z(r3), l, v3); + + l = glmm_load(m1[3]); + v0 = glmm_fmadd(glmm_splat_w(r0), l, v0); + v1 = glmm_fmadd(glmm_splat_w(r1), l, v1); + v2 = glmm_fmadd(glmm_splat_w(r2), l, v2); + v3 = glmm_fmadd(glmm_splat_w(r3), l, v3); + + glmm_store(dest[0], v0); + glmm_store(dest[1], v1); + glmm_store(dest[2], v2); + glmm_store(dest[3], v3); +} + +CGLM_INLINE +void +glm_mat4_mulv_wasm(mat4 m, vec4 v, vec4 dest) { + glmm_128 x0, x1, m0, m1, m2, m3, v0, v1, v2, v3; + + m0 = glmm_load(m[0]); + m1 = glmm_load(m[1]); + m2 = glmm_load(m[2]); + m3 = glmm_load(m[3]); + + x0 = glmm_load(v); + v0 = glmm_splat_x(x0); + v1 = glmm_splat_y(x0); + v2 = glmm_splat_z(x0); + v3 = glmm_splat_w(x0); + + x1 = wasm_f32x4_mul(m3, v3); + x1 = glmm_fmadd(m2, v2, x1); + x1 = glmm_fmadd(m1, v1, x1); + x1 = glmm_fmadd(m0, v0, x1); + + glmm_store(dest, x1); +} + +CGLM_INLINE +float +glm_mat4_det_wasm(mat4 mat) { + glmm_128 r0, r1, r2, r3, x0, x1, x2; + + /* 127 <- 0, [square] det(A) = det(At) */ + r0 = glmm_load(mat[0]); /* d c b a */ + r1 = glmm_load(mat[1]); /* h g f e */ + r2 = glmm_load(mat[2]); /* l k j i */ + r3 = glmm_load(mat[3]); /* p o n m */ + + /* + t[1] = j * p - n * l; + t[2] = j * o - n * k; + t[3] = i * p - m * l; + t[4] = i * o - m * k; + */ + x0 = glmm_fnmadd(glmm_shuff1(r3, 0, 0, 1, 1), glmm_shuff1(r2, 2, 3, 2, 3), + wasm_f32x4_mul(glmm_shuff1(r2, 0, 0, 1, 1), + glmm_shuff1(r3, 2, 3, 2, 3))); + /* + t[0] = k * p - o * l; + t[0] = k * p - o * l; + t[5] = i * n - m * j; + t[5] = i * n - m * j; + */ + x1 = glmm_fnmadd(glmm_shuff1(r3, 0, 0, 2, 2), glmm_shuff1(r2, 1, 1, 3, 3), + wasm_f32x4_mul(glmm_shuff1(r2, 0, 0, 2, 2), + glmm_shuff1(r3, 1, 1, 3, 3))); + + /* + a * (f * t[0] - g * t[1] + h * t[2]) + - b * (e * t[0] - g * t[3] + h * t[4]) + + c * (e * t[1] - f * t[3] + h * t[5]) + - d * (e * t[2] - f * t[4] + g * t[5]) + */ + x2 = glmm_fnmadd(glmm_shuff1(r1, 1, 1, 2, 2), glmm_shuff1(x0, 3, 2, 2, 0), + wasm_f32x4_mul(glmm_shuff1(r1, 0, 0, 0, 1), + wasm_i32x4_shuffle(x1, x0, 0, 0, 4, 5))); + x2 = glmm_fmadd(glmm_shuff1(r1, 2, 3, 3, 3), + wasm_i32x4_shuffle(x0, x1, 1, 3, 6, 6), + x2); + /* x2 = wasm_v128_xor(x2, wasm_f32x4_const(0.f, -0.f, 0.f, -0.f)); */ + x2 = wasm_v128_xor(x2, glmm_float32x4_SIGNMASK_PNPN); + + return glmm_hadd(wasm_f32x4_mul(x2, r0)); +} + +CGLM_INLINE +void +glm_mat4_inv_fast_wasm(mat4 mat, mat4 dest) { + glmm_128 r0, r1, r2, r3, + v0, v1, v2, v3, + t0, t1, t2, t3, t4, t5, + x0, x1, x2, x3, x4, x5, x6, x7, x8, x9; + + /* x8 = wasm_f32x4_const(0.f, -0.f, 0.f, -0.f); */ + x8 = glmm_float32x4_SIGNMASK_PNPN; + x9 = glmm_shuff1(x8, 2, 1, 2, 1); + + /* 127 <- 0 */ + r0 = glmm_load(mat[0]); /* d c b a */ + r1 = glmm_load(mat[1]); /* h g f e */ + r2 = glmm_load(mat[2]); /* l k j i */ + r3 = glmm_load(mat[3]); /* p o n m */ + /* x0 = _mm_movehl_ps(r3, r2); */ + x0 = wasm_i32x4_shuffle(r3, r2, 6, 7, 2, 3); /* p o l k */ + /* x3 = _mm_movelh_ps(r2, r3); */ + x3 = wasm_i32x4_shuffle(r2, r3, 0, 1, 4, 5); /* n m j i */ + x1 = glmm_shuff1(x0, 1, 3, 3 ,3); /* l p p p */ + x2 = glmm_shuff1(x0, 0, 2, 2, 2); /* k o o o */ + x4 = glmm_shuff1(x3, 1, 3, 3, 3); /* j n n n */ + x7 = glmm_shuff1(x3, 0, 2, 2, 2); /* i m m m */ + + x6 = wasm_i32x4_shuffle(r2, r1, 0, 0, 4, 4); /* e e i i */ + x5 = wasm_i32x4_shuffle(r2, r1, 1, 1, 5, 5); /* f f j j */ + x3 = wasm_i32x4_shuffle(r2, r1, 2, 2, 6, 6); /* g g k k */ + x0 = wasm_i32x4_shuffle(r2, r1, 3, 3, 7, 7); /* h h l l */ + + t0 = wasm_f32x4_mul(x3, x1); + t1 = wasm_f32x4_mul(x5, x1); + t2 = wasm_f32x4_mul(x5, x2); + t3 = wasm_f32x4_mul(x6, x1); + t4 = wasm_f32x4_mul(x6, x2); + t5 = wasm_f32x4_mul(x6, x4); + + /* t1[0] = k * p - o * l; + t1[0] = k * p - o * l; + t2[0] = g * p - o * h; + t3[0] = g * l - k * h; */ + t0 = glmm_fnmadd(x2, x0, t0); + + /* t1[1] = j * p - n * l; + t1[1] = j * p - n * l; + t2[1] = f * p - n * h; + t3[1] = f * l - j * h; */ + t1 = glmm_fnmadd(x4, x0, t1); + + /* t1[2] = j * o - n * k + t1[2] = j * o - n * k; + t2[2] = f * o - n * g; + t3[2] = f * k - j * g; */ + t2 = glmm_fnmadd(x4, x3, t2); + + /* t1[3] = i * p - m * l; + t1[3] = i * p - m * l; + t2[3] = e * p - m * h; + t3[3] = e * l - i * h; */ + t3 = glmm_fnmadd(x7, x0, t3); + + /* t1[4] = i * o - m * k; + t1[4] = i * o - m * k; + t2[4] = e * o - m * g; + t3[4] = e * k - i * g; */ + t4 = glmm_fnmadd(x7, x3, t4); + + /* t1[5] = i * n - m * j; + t1[5] = i * n - m * j; + t2[5] = e * n - m * f; + t3[5] = e * j - i * f; */ + t5 = glmm_fnmadd(x7, x5, t5); + /* x4 = _mm_movelh_ps(r0, r1); */ + x4 = wasm_i32x4_shuffle(r0, r1, 0, 1, 4, 5); /* f e b a */ + /* x5 = _mm_movehl_ps(r1, r0); */ + x5 = wasm_i32x4_shuffle(r1, r0, 6, 7, 2, 3); /* h g d c */ + + x0 = glmm_shuff1(x4, 0, 0, 0, 2); /* a a a e */ + x1 = glmm_shuff1(x4, 1, 1, 1, 3); /* b b b f */ + x2 = glmm_shuff1(x5, 0, 0, 0, 2); /* c c c g */ + x3 = glmm_shuff1(x5, 1, 1, 1, 3); /* d d d h */ + + v2 = wasm_f32x4_mul(x0, t1); + v1 = wasm_f32x4_mul(x0, t0); + v3 = wasm_f32x4_mul(x0, t2); + v0 = wasm_f32x4_mul(x1, t0); + + v2 = glmm_fnmadd(x1, t3, v2); + v3 = glmm_fnmadd(x1, t4, v3); + v0 = glmm_fnmadd(x2, t1, v0); + v1 = glmm_fnmadd(x2, t3, v1); + + v3 = glmm_fmadd(x2, t5, v3); + v0 = glmm_fmadd(x3, t2, v0); + v2 = glmm_fmadd(x3, t5, v2); + v1 = glmm_fmadd(x3, t4, v1); + + /* + dest[0][0] = f * t1[0] - g * t1[1] + h * t1[2]; + dest[0][1] =-(b * t1[0] - c * t1[1] + d * t1[2]); + dest[0][2] = b * t2[0] - c * t2[1] + d * t2[2]; + dest[0][3] =-(b * t3[0] - c * t3[1] + d * t3[2]); */ + v0 = wasm_v128_xor(v0, x8); + + /* + dest[2][0] = e * t1[1] - f * t1[3] + h * t1[5]; + dest[2][1] =-(a * t1[1] - b * t1[3] + d * t1[5]); + dest[2][2] = a * t2[1] - b * t2[3] + d * t2[5]; + dest[2][3] =-(a * t3[1] - b * t3[3] + d * t3[5]);*/ + v2 = wasm_v128_xor(v2, x8); + + /* + dest[1][0] =-(e * t1[0] - g * t1[3] + h * t1[4]); + dest[1][1] = a * t1[0] - c * t1[3] + d * t1[4]; + dest[1][2] =-(a * t2[0] - c * t2[3] + d * t2[4]); + dest[1][3] = a * t3[0] - c * t3[3] + d * t3[4]; */ + v1 = wasm_v128_xor(v1, x9); + + /* + dest[3][0] =-(e * t1[2] - f * t1[4] + g * t1[5]); + dest[3][1] = a * t1[2] - b * t1[4] + c * t1[5]; + dest[3][2] =-(a * t2[2] - b * t2[4] + c * t2[5]); + dest[3][3] = a * t3[2] - b * t3[4] + c * t3[5]; */ + v3 = wasm_v128_xor(v3, x9); + + /* determinant */ + x0 = wasm_i32x4_shuffle(v0, v1, 0, 0, 4, 4); + x1 = wasm_i32x4_shuffle(v2, v3, 0, 0, 4, 4); + x0 = wasm_i32x4_shuffle(x0, x1, 0, 2, 4, 6); + + /* x0 = _mm_rcp_ps(glmm_vhadd(wasm_f32x4_mul(x0, r0))); */ + x0 = wasm_f32x4_div(wasm_f32x4_const_splat(1.0f), + glmm_vhadd(wasm_f32x4_mul(x0, r0))); + + glmm_store(dest[0], wasm_f32x4_mul(v0, x0)); + glmm_store(dest[1], wasm_f32x4_mul(v1, x0)); + glmm_store(dest[2], wasm_f32x4_mul(v2, x0)); + glmm_store(dest[3], wasm_f32x4_mul(v3, x0)); +} + +CGLM_INLINE +void +glm_mat4_inv_wasm(mat4 mat, mat4 dest) { + glmm_128 r0, r1, r2, r3, + v0, v1, v2, v3, + t0, t1, t2, t3, t4, t5, + x0, x1, x2, x3, x4, x5, x6, x7, x8, x9; + + /* x8 = wasm_f32x4_const(0.f, -0.f, 0.f, -0.f); */ + x8 = glmm_float32x4_SIGNMASK_PNPN; + x9 = glmm_shuff1(x8, 2, 1, 2, 1); + + /* 127 <- 0 */ + r0 = glmm_load(mat[0]); /* d c b a */ + r1 = glmm_load(mat[1]); /* h g f e */ + r2 = glmm_load(mat[2]); /* l k j i */ + r3 = glmm_load(mat[3]); /* p o n m */ + /* x0 = _mm_movehl_ps(r3, r2); */ + x0 = wasm_i32x4_shuffle(r3, r2, 6, 7, 2, 3); /* p o l k */ + /* x3 = _mm_movelh_ps(r2, r3); */ + x3 = wasm_i32x4_shuffle(r2, r3, 0, 1, 4, 5); /* n m j i */ + x1 = glmm_shuff1(x0, 1, 3, 3 ,3); /* l p p p */ + x2 = glmm_shuff1(x0, 0, 2, 2, 2); /* k o o o */ + x4 = glmm_shuff1(x3, 1, 3, 3, 3); /* j n n n */ + x7 = glmm_shuff1(x3, 0, 2, 2, 2); /* i m m m */ + + x6 = wasm_i32x4_shuffle(r2, r1, 0, 0, 4, 4); /* e e i i */ + x5 = wasm_i32x4_shuffle(r2, r1, 1, 1, 5, 5); /* f f j j */ + x3 = wasm_i32x4_shuffle(r2, r1, 2, 2, 6, 6); /* g g k k */ + x0 = wasm_i32x4_shuffle(r2, r1, 3, 3, 7, 7); /* h h l l */ + + t0 = wasm_f32x4_mul(x3, x1); + t1 = wasm_f32x4_mul(x5, x1); + t2 = wasm_f32x4_mul(x5, x2); + t3 = wasm_f32x4_mul(x6, x1); + t4 = wasm_f32x4_mul(x6, x2); + t5 = wasm_f32x4_mul(x6, x4); + + /* t1[0] = k * p - o * l; + t1[0] = k * p - o * l; + t2[0] = g * p - o * h; + t3[0] = g * l - k * h; */ + t0 = glmm_fnmadd(x2, x0, t0); + + /* t1[1] = j * p - n * l; + t1[1] = j * p - n * l; + t2[1] = f * p - n * h; + t3[1] = f * l - j * h; */ + t1 = glmm_fnmadd(x4, x0, t1); + + /* t1[2] = j * o - n * k + t1[2] = j * o - n * k; + t2[2] = f * o - n * g; + t3[2] = f * k - j * g; */ + t2 = glmm_fnmadd(x4, x3, t2); + + /* t1[3] = i * p - m * l; + t1[3] = i * p - m * l; + t2[3] = e * p - m * h; + t3[3] = e * l - i * h; */ + t3 = glmm_fnmadd(x7, x0, t3); + + /* t1[4] = i * o - m * k; + t1[4] = i * o - m * k; + t2[4] = e * o - m * g; + t3[4] = e * k - i * g; */ + t4 = glmm_fnmadd(x7, x3, t4); + + /* t1[5] = i * n - m * j; + t1[5] = i * n - m * j; + t2[5] = e * n - m * f; + t3[5] = e * j - i * f; */ + t5 = glmm_fnmadd(x7, x5, t5); + /* x4 = _mm_movelh_ps(r0, r1); */ + x4 = wasm_i32x4_shuffle(r0, r1, 0, 1, 4, 5); /* f e b a */ + /* x5 = _mm_movehl_ps(r1, r0); */ + x5 = wasm_i32x4_shuffle(r1, r0, 6, 7, 2, 3); /* h g d c */ + + x0 = glmm_shuff1(x4, 0, 0, 0, 2); /* a a a e */ + x1 = glmm_shuff1(x4, 1, 1, 1, 3); /* b b b f */ + x2 = glmm_shuff1(x5, 0, 0, 0, 2); /* c c c g */ + x3 = glmm_shuff1(x5, 1, 1, 1, 3); /* d d d h */ + + v2 = wasm_f32x4_mul(x0, t1); + v1 = wasm_f32x4_mul(x0, t0); + v3 = wasm_f32x4_mul(x0, t2); + v0 = wasm_f32x4_mul(x1, t0); + + v2 = glmm_fnmadd(x1, t3, v2); + v3 = glmm_fnmadd(x1, t4, v3); + v0 = glmm_fnmadd(x2, t1, v0); + v1 = glmm_fnmadd(x2, t3, v1); + + v3 = glmm_fmadd(x2, t5, v3); + v0 = glmm_fmadd(x3, t2, v0); + v2 = glmm_fmadd(x3, t5, v2); + v1 = glmm_fmadd(x3, t4, v1); + + /* + dest[0][0] = f * t1[0] - g * t1[1] + h * t1[2]; + dest[0][1] =-(b * t1[0] - c * t1[1] + d * t1[2]); + dest[0][2] = b * t2[0] - c * t2[1] + d * t2[2]; + dest[0][3] =-(b * t3[0] - c * t3[1] + d * t3[2]); */ + v0 = wasm_v128_xor(v0, x8); + + /* + dest[2][0] = e * t1[1] - f * t1[3] + h * t1[5]; + dest[2][1] =-(a * t1[1] - b * t1[3] + d * t1[5]); + dest[2][2] = a * t2[1] - b * t2[3] + d * t2[5]; + dest[2][3] =-(a * t3[1] - b * t3[3] + d * t3[5]);*/ + v2 = wasm_v128_xor(v2, x8); + + /* + dest[1][0] =-(e * t1[0] - g * t1[3] + h * t1[4]); + dest[1][1] = a * t1[0] - c * t1[3] + d * t1[4]; + dest[1][2] =-(a * t2[0] - c * t2[3] + d * t2[4]); + dest[1][3] = a * t3[0] - c * t3[3] + d * t3[4]; */ + v1 = wasm_v128_xor(v1, x9); + + /* + dest[3][0] =-(e * t1[2] - f * t1[4] + g * t1[5]); + dest[3][1] = a * t1[2] - b * t1[4] + c * t1[5]; + dest[3][2] =-(a * t2[2] - b * t2[4] + c * t2[5]); + dest[3][3] = a * t3[2] - b * t3[4] + c * t3[5]; */ + v3 = wasm_v128_xor(v3, x9); + + /* determinant */ + x0 = wasm_i32x4_shuffle(v0, v1, 0, 0, 4, 4); + x1 = wasm_i32x4_shuffle(v2, v3, 0, 0, 4, 4); + x0 = wasm_i32x4_shuffle(x0, x1, 0, 2, 4, 6); + + x0 = wasm_f32x4_div(wasm_f32x4_splat(1.0f), glmm_vhadd(wasm_f32x4_mul(x0, r0))); + + glmm_store(dest[0], wasm_f32x4_mul(v0, x0)); + glmm_store(dest[1], wasm_f32x4_mul(v1, x0)); + glmm_store(dest[2], wasm_f32x4_mul(v2, x0)); + glmm_store(dest[3], wasm_f32x4_mul(v3, x0)); +} + +#endif +#endif /* cglm_mat_wasm_h */ diff --git a/cglm/include/cglm/simd/wasm/quat.h b/cglm/include/cglm/simd/wasm/quat.h new file mode 100644 index 0000000..8d72546 --- /dev/null +++ b/cglm/include/cglm/simd/wasm/quat.h @@ -0,0 +1,55 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_quat_wasm_h +#define cglm_quat_wasm_h +#if defined(__wasm__) && defined(__wasm_simd128__) + +#include "../../common.h" +#include "../intrin.h" + +CGLM_INLINE +void +glm_quat_mul_wasm(versor p, versor q, versor dest) { + /* + + (a1 b2 + b1 a2 + c1 d2 − d1 c2)i + + (a1 c2 − b1 d2 + c1 a2 + d1 b2)j + + (a1 d2 + b1 c2 − c1 b2 + d1 a2)k + a1 a2 − b1 b2 − c1 c2 − d1 d2 + */ + + glmm_128 xp, xq, x1, x2, x3, r, x, y, z; + + xp = glmm_load(p); /* 3 2 1 0 */ + xq = glmm_load(q); + /* x1 = wasm_f32x4_const(0.f, -0.f, 0.f, -0.f); */ + x1 = glmm_float32x4_SIGNMASK_PNPN; /* TODO: _mm_set1_ss() + shuff ? */ + r = wasm_f32x4_mul(glmm_splat_w(xp), xq); + /* x2 = _mm_unpackhi_ps(x1, x1); */ + x2 = wasm_i32x4_shuffle(x1, x1, 2, 6, 3, 7); + x3 = glmm_shuff1(x1, 3, 2, 0, 1); + x = glmm_splat_x(xp); + y = glmm_splat_y(xp); + z = glmm_splat_z(xp); + + x = wasm_v128_xor(x, x1); + y = wasm_v128_xor(y, x2); + z = wasm_v128_xor(z, x3); + + x1 = glmm_shuff1(xq, 0, 1, 2, 3); + x2 = glmm_shuff1(xq, 1, 0, 3, 2); + x3 = glmm_shuff1(xq, 2, 3, 0, 1); + + r = glmm_fmadd(x, x1, r); + r = glmm_fmadd(y, x2, r); + r = glmm_fmadd(z, x3, r); + + glmm_store(dest, r); +} + +#endif +#endif /* cglm_quat_wasm_h */ diff --git a/cglm/include/cglm/simd/x86.h b/cglm/include/cglm/simd/x86.h new file mode 100644 index 0000000..2410d0f --- /dev/null +++ b/cglm/include/cglm/simd/x86.h @@ -0,0 +1,365 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_simd_x86_h +#define cglm_simd_x86_h +#include "intrin.h" +#ifdef CGLM_SIMD_x86 + +#ifdef CGLM_ALL_UNALIGNED +# define glmm_load(p) _mm_loadu_ps(p) +# define glmm_store(p, a) _mm_storeu_ps(p, a) +#else +# define glmm_load(p) _mm_load_ps(p) +# define glmm_store(p, a) _mm_store_ps(p, a) +#endif + +#define glmm_128 __m128 + +#ifdef __AVX__ +# define glmm_shuff1(xmm, z, y, x, w) \ + _mm_permute_ps((xmm), _MM_SHUFFLE(z, y, x, w)) +#else +# if !defined(CGLM_NO_INT_DOMAIN) && defined(__SSE2__) +# define glmm_shuff1(xmm, z, y, x, w) \ + _mm_castsi128_ps(_mm_shuffle_epi32(_mm_castps_si128(xmm), \ + _MM_SHUFFLE(z, y, x, w))) +# else +# define glmm_shuff1(xmm, z, y, x, w) \ + _mm_shuffle_ps(xmm, xmm, _MM_SHUFFLE(z, y, x, w)) +# endif +#endif + +#define glmm_splat(x, lane) glmm_shuff1(x, lane, lane, lane, lane) + +#ifdef __AVX__ +# define glmm_set1(x) _mm_broadcast_ss(&x) +# define glmm_set1_ptr(x) _mm_broadcast_ss(x) +# define glmm_set1_rval(x) _mm_set1_ps(x) +# ifdef __AVX2__ +# define glmm_splat_x(x) _mm_broadcastss_ps(x) +# else +# define glmm_splat_x(x) _mm_permute_ps(x, _MM_SHUFFLE(0, 0, 0, 0)) +# endif +# define glmm_splat_y(x) _mm_permute_ps(x, _MM_SHUFFLE(1, 1, 1, 1)) +# define glmm_splat_z(x) _mm_permute_ps(x, _MM_SHUFFLE(2, 2, 2, 2)) +# define glmm_splat_w(x) _mm_permute_ps(x, _MM_SHUFFLE(3, 3, 3, 3)) +#else +# define glmm_set1(x) _mm_set1_ps(x) +# define glmm_set1_ptr(x) _mm_set1_ps(*x) +# define glmm_set1_rval(x) _mm_set1_ps(x) + +# define glmm_splat_x(x) glmm_splat(x, 0) +# define glmm_splat_y(x) glmm_splat(x, 1) +# define glmm_splat_z(x) glmm_splat(x, 2) +# define glmm_splat_w(x) glmm_splat(x, 3) +#endif + +#ifdef __AVX__ +# ifdef CGLM_ALL_UNALIGNED +# define glmm_load256(p) _mm256_loadu_ps(p) +# define glmm_store256(p, a) _mm256_storeu_ps(p, a) +# else +# define glmm_load256(p) _mm256_load_ps(p) +# define glmm_store256(p, a) _mm256_store_ps(p, a) +# endif +#endif + +/* Note that `0x80000000` corresponds to `INT_MIN` for a 32-bit int. */ + +#if defined(__SSE2__) +# define GLMM_NEGZEROf ((int)0x80000000) /* 0x80000000 ---> -0.0f */ +# define GLMM_POSZEROf ((int)0x00000000) /* 0x00000000 ---> +0.0f */ +#else +# ifdef CGLM_FAST_MATH + union { int i; float f; } static GLMM_NEGZEROf_TU = { .i = (int)0x80000000 }; +# define GLMM_NEGZEROf GLMM_NEGZEROf_TU.f +# define GLMM_POSZEROf 0.0f +# else +# define GLMM_NEGZEROf -0.0f +# define GLMM_POSZEROf 0.0f +# endif +#endif + +#if defined(__SSE2__) +# define GLMM__SIGNMASKf(X, Y, Z, W) \ + _mm_castsi128_ps(_mm_set_epi32(X, Y, Z, W)) + /* _mm_set_ps(X, Y, Z, W); */ +#else +# define GLMM__SIGNMASKf(X, Y, Z, W) _mm_set_ps(X, Y, Z, W) +#endif + +#define glmm_float32x4_SIGNMASK_PNPN GLMM__SIGNMASKf(GLMM_POSZEROf, GLMM_NEGZEROf, GLMM_POSZEROf, GLMM_NEGZEROf) +#define glmm_float32x4_SIGNMASK_NPNP GLMM__SIGNMASKf(GLMM_NEGZEROf, GLMM_POSZEROf, GLMM_NEGZEROf, GLMM_POSZEROf) +#define glmm_float32x4_SIGNMASK_NPPN GLMM__SIGNMASKf(GLMM_NEGZEROf, GLMM_POSZEROf, GLMM_POSZEROf, GLMM_NEGZEROf) + +/* fasth math prevents -0.0f to work */ +#if defined(__SSE2__) +# define glmm_float32x4_SIGNMASK_NEG _mm_castsi128_ps(_mm_set1_epi32(GLMM_NEGZEROf)) /* _mm_set1_ps(-0.0f) */ +#else +# define glmm_float32x4_SIGNMASK_NEG glmm_set1(GLMM_NEGZEROf) +#endif + +#define glmm_float32x8_SIGNMASK_NEG _mm256_castsi256_ps(_mm256_set1_epi32(GLMM_NEGZEROf)) + +static inline +__m128 +glmm_abs(__m128 x) { + return _mm_andnot_ps(glmm_float32x4_SIGNMASK_NEG, x); +} + +static inline __m128 glmm_min(__m128 a, __m128 b) { return _mm_min_ps(a, b); } +static inline __m128 glmm_max(__m128 a, __m128 b) { return _mm_max_ps(a, b); } + +static inline +__m128 +glmm_vhadd(__m128 v) { + __m128 x0; + x0 = _mm_add_ps(v, glmm_shuff1(v, 0, 1, 2, 3)); + x0 = _mm_add_ps(x0, glmm_shuff1(x0, 1, 0, 0, 1)); + return x0; +} + +static inline +__m128 +glmm_vhadds(__m128 v) { +#if defined(__SSE3__) + __m128 shuf, sums; + shuf = _mm_movehdup_ps(v); + sums = _mm_add_ps(v, shuf); + shuf = _mm_movehl_ps(shuf, sums); + sums = _mm_add_ss(sums, shuf); + return sums; +#else + __m128 shuf, sums; + shuf = glmm_shuff1(v, 2, 3, 0, 1); + sums = _mm_add_ps(v, shuf); + shuf = _mm_movehl_ps(shuf, sums); + sums = _mm_add_ss(sums, shuf); + return sums; +#endif +} + +static inline +float +glmm_hadd(__m128 v) { + return _mm_cvtss_f32(glmm_vhadds(v)); +} + +static inline +__m128 +glmm_vhmin(__m128 v) { + __m128 x0, x1, x2; + x0 = _mm_movehl_ps(v, v); /* [2, 3, 2, 3] */ + x1 = _mm_min_ps(x0, v); /* [0|2, 1|3, 2|2, 3|3] */ + x2 = glmm_splat(x1, 1); /* [1|3, 1|3, 1|3, 1|3] */ + return _mm_min_ss(x1, x2); +} + +static inline +float +glmm_hmin(__m128 v) { + return _mm_cvtss_f32(glmm_vhmin(v)); +} + +static inline +__m128 +glmm_vhmax(__m128 v) { + __m128 x0, x1, x2; + x0 = _mm_movehl_ps(v, v); /* [2, 3, 2, 3] */ + x1 = _mm_max_ps(x0, v); /* [0|2, 1|3, 2|2, 3|3] */ + x2 = glmm_splat(x1, 1); /* [1|3, 1|3, 1|3, 1|3] */ + return _mm_max_ss(x1, x2); +} + +static inline +float +glmm_hmax(__m128 v) { + return _mm_cvtss_f32(glmm_vhmax(v)); +} + +static inline +__m128 +glmm_vdots(__m128 a, __m128 b) { +#if (defined(__SSE4_1__) || defined(__SSE4_2__)) && defined(CGLM_SSE4_DOT) + return _mm_dp_ps(a, b, 0xFF); +#elif defined(__SSE3__) && defined(CGLM_SSE3_DOT) + __m128 x0, x1; + x0 = _mm_mul_ps(a, b); + x1 = _mm_hadd_ps(x0, x0); + return _mm_hadd_ps(x1, x1); +#else + return glmm_vhadds(_mm_mul_ps(a, b)); +#endif +} + +static inline +__m128 +glmm_vdot(__m128 a, __m128 b) { +#if (defined(__SSE4_1__) || defined(__SSE4_2__)) && defined(CGLM_SSE4_DOT) + return _mm_dp_ps(a, b, 0xFF); +#elif defined(__SSE3__) && defined(CGLM_SSE3_DOT) + __m128 x0, x1; + x0 = _mm_mul_ps(a, b); + x1 = _mm_hadd_ps(x0, x0); + return _mm_hadd_ps(x1, x1); +#else + __m128 x0; + x0 = _mm_mul_ps(a, b); + x0 = _mm_add_ps(x0, glmm_shuff1(x0, 1, 0, 3, 2)); + return _mm_add_ps(x0, glmm_shuff1(x0, 0, 1, 0, 1)); +#endif +} + +static inline +float +glmm_dot(__m128 a, __m128 b) { + return _mm_cvtss_f32(glmm_vdots(a, b)); +} + +static inline +float +glmm_norm(__m128 a) { + return _mm_cvtss_f32(_mm_sqrt_ss(glmm_vhadds(_mm_mul_ps(a, a)))); +} + +static inline +float +glmm_norm2(__m128 a) { + return _mm_cvtss_f32(glmm_vhadds(_mm_mul_ps(a, a))); +} + +static inline +float +glmm_norm_one(__m128 a) { + return _mm_cvtss_f32(glmm_vhadds(glmm_abs(a))); +} + +static inline +float +glmm_norm_inf(__m128 a) { + return _mm_cvtss_f32(glmm_vhmax(glmm_abs(a))); +} + +#if defined(__SSE2__) +static inline +__m128 +glmm_load3(float v[3]) { + __m128i xy; + __m128 z; + + xy = _mm_loadl_epi64(CGLM_CASTPTR_ASSUME_ALIGNED(v, const __m128i)); + z = _mm_load_ss(&v[2]); + + return _mm_movelh_ps(_mm_castsi128_ps(xy), z); +} + +static inline +void +glmm_store3(float v[3], __m128 vx) { + _mm_storel_pi(CGLM_CASTPTR_ASSUME_ALIGNED(v, __m64), vx); + _mm_store_ss(&v[2], glmm_shuff1(vx, 2, 2, 2, 2)); +} +#endif + +static inline +__m128 +glmm_div(__m128 a, __m128 b) { + return _mm_div_ps(a, b); +} + +/* enable FMA macro for MSVC? */ +#if defined(_MSC_VER) && !defined(__FMA__) && defined(__AVX2__) +# define __FMA__ 1 +#endif + +static inline +__m128 +glmm_fmadd(__m128 a, __m128 b, __m128 c) { +#ifdef __FMA__ + return _mm_fmadd_ps(a, b, c); +#else + return _mm_add_ps(c, _mm_mul_ps(a, b)); +#endif +} + +static inline +__m128 +glmm_fnmadd(__m128 a, __m128 b, __m128 c) { +#ifdef __FMA__ + return _mm_fnmadd_ps(a, b, c); +#else + return _mm_sub_ps(c, _mm_mul_ps(a, b)); +#endif +} + +static inline +__m128 +glmm_fmsub(__m128 a, __m128 b, __m128 c) { +#ifdef __FMA__ + return _mm_fmsub_ps(a, b, c); +#else + return _mm_sub_ps(_mm_mul_ps(a, b), c); +#endif +} + +static inline +__m128 +glmm_fnmsub(__m128 a, __m128 b, __m128 c) { +#ifdef __FMA__ + return _mm_fnmsub_ps(a, b, c); +#else + return _mm_xor_ps(_mm_add_ps(_mm_mul_ps(a, b), c), + glmm_float32x4_SIGNMASK_NEG); +#endif +} + +#if defined(__AVX__) +static inline +__m256 +glmm256_fmadd(__m256 a, __m256 b, __m256 c) { +#ifdef __FMA__ + return _mm256_fmadd_ps(a, b, c); +#else + return _mm256_add_ps(c, _mm256_mul_ps(a, b)); +#endif +} + +static inline +__m256 +glmm256_fnmadd(__m256 a, __m256 b, __m256 c) { +#ifdef __FMA__ + return _mm256_fnmadd_ps(a, b, c); +#else + return _mm256_sub_ps(c, _mm256_mul_ps(a, b)); +#endif +} + +static inline +__m256 +glmm256_fmsub(__m256 a, __m256 b, __m256 c) { +#ifdef __FMA__ + return _mm256_fmsub_ps(a, b, c); +#else + return _mm256_sub_ps(_mm256_mul_ps(a, b), c); +#endif +} + +static inline +__m256 +glmm256_fnmsub(__m256 a, __m256 b, __m256 c) { +#ifdef __FMA__ + return _mm256_fmsub_ps(a, b, c); +#else + return _mm256_xor_ps(_mm256_sub_ps(_mm256_mul_ps(a, b), c), + glmm_float32x8_SIGNMASK_NEG); +#endif +} +#endif + +#endif +#endif /* cglm_simd_x86_h */ diff --git a/cglm/include/cglm/sphere.h b/cglm/include/cglm/sphere.h new file mode 100644 index 0000000..334b83a --- /dev/null +++ b/cglm/include/cglm/sphere.h @@ -0,0 +1,99 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_sphere_h +#define cglm_sphere_h + +#include "common.h" +#include "mat4.h" + +/* + Sphere Representation in cglm: [center.x, center.y, center.z, radii] + + You could use this representation or you can convert it to vec4 before call + any function + */ + +/*! + * @brief helper for getting sphere radius + * + * @param[in] s sphere + * + * @return returns radii + */ +CGLM_INLINE +float +glm_sphere_radii(vec4 s) { + return s[3]; +} + +/*! + * @brief apply transform to sphere, it is just wrapper for glm_mat4_mulv3 + * + * @param[in] s sphere + * @param[in] m transform matrix + * @param[out] dest transformed sphere + */ +CGLM_INLINE +void +glm_sphere_transform(vec4 s, mat4 m, vec4 dest) { + glm_mat4_mulv3(m, s, 1.0f, dest); + dest[3] = s[3]; +} + +/*! + * @brief merges two spheres and creates a new one + * + * two sphere must be in same space, for instance if one in world space then + * the other must be in world space too, not in local space. + * + * @param[in] s1 sphere 1 + * @param[in] s2 sphere 2 + * @param[out] dest merged/extended sphere + */ +CGLM_INLINE +void +glm_sphere_merge(vec4 s1, vec4 s2, vec4 dest) { + float dist, radii; + + dist = glm_vec3_distance(s1, s2); + radii = dist + s1[3] + s2[3]; + + radii = glm_max(radii, s1[3]); + radii = glm_max(radii, s2[3]); + + glm_vec3_center(s1, s2, dest); + dest[3] = radii; +} + +/*! + * @brief check if two sphere intersects + * + * @param[in] s1 sphere + * @param[in] s2 other sphere + */ +CGLM_INLINE +bool +glm_sphere_sphere(vec4 s1, vec4 s2) { + return glm_vec3_distance2(s1, s2) <= glm_pow2(s1[3] + s2[3]); +} + +/*! + * @brief check if sphere intersects with point + * + * @param[in] s sphere + * @param[in] point point + */ +CGLM_INLINE +bool +glm_sphere_point(vec4 s, vec3 point) { + float rr; + rr = s[3] * s[3]; + return glm_vec3_distance2(point, s) <= rr; +} + +#endif /* cglm_sphere_h */ diff --git a/cglm/include/cglm/struct.h b/cglm/include/cglm/struct.h new file mode 100644 index 0000000..31ca4e2 --- /dev/null +++ b/cglm/include/cglm/struct.h @@ -0,0 +1,50 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_structs_h +#define cglm_structs_h +#ifdef __cplusplus +extern "C" { +#endif + +#include "cglm.h" +#include "types-struct.h" +#include "struct/vec2.h" +#include "struct/vec3.h" +#include "struct/vec4.h" +#include "struct/ivec2.h" +#include "struct/ivec3.h" +#include "struct/ivec4.h" +#include "struct/mat2.h" +#include "struct/mat2x3.h" +#include "struct/mat2x4.h" +#include "struct/mat3.h" +#include "struct/mat3x2.h" +#include "struct/mat3x4.h" +#include "struct/mat4.h" +#include "struct/mat4x2.h" +#include "struct/mat4x3.h" +#include "struct/affine.h" +#include "struct/frustum.h" +#include "struct/plane.h" +#include "struct/noise.h" +#include "struct/box.h" +#include "struct/color.h" +#include "struct/io.h" +#include "struct/cam.h" +#include "struct/quat.h" +#include "struct/euler.h" +#include "struct/project.h" +#include "struct/sphere.h" +#include "struct/curve.h" +#include "struct/affine2d.h" +#include "struct/ray.h" + +#ifdef __cplusplus +} +#endif +#endif /* cglm_structs_h */ diff --git a/cglm/include/cglm/struct/aabb2d.h b/cglm/include/cglm/struct/aabb2d.h new file mode 100644 index 0000000..9077069 --- /dev/null +++ b/cglm/include/cglm/struct/aabb2d.h @@ -0,0 +1,253 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglms_aabb2ds_h +#define cglms_aabb2ds_h + +#include "../common.h" +#include "../types-struct.h" +#include "../aabb2d.h" +#include "vec2.h" +#include "vec4.h" +#include "mat4.h" + +/* api definition */ +#define glms_aabb2d_(NAME) CGLM_STRUCTAPI(aabb2d, NAME) + +/*! + * @brief apply transform to Axis-Aligned Bounding Box + * + * @param[in] aabb bounding box + * @param[in] m transform matrix + * @param[out] dest transformed bounding box + */ +CGLM_INLINE +void +glms_aabb2d_(transform)(vec2s aabb[2], mat3s m, vec2s dest[2]) { + vec2 rawAabb[2]; + vec2 rawDest[2]; + + glms_vec2_(unpack)(rawAabb, aabb, 2); + glm_aabb2d_transform(rawAabb, m.raw, rawDest); + glms_vec2_(pack)(dest, rawDest, 2); +} + +/*! + * @brief merges two AABB bounding box and creates new one + * + * two box must be in same space, if one of box is in different space then + * you should consider to convert it's space by glm_box_space + * + * @param[in] aabb1 bounding box 1 + * @param[in] aabb2 bounding box 2 + * @param[out] dest merged bounding box + */ +CGLM_INLINE +void +glms_aabb2d_(merge)(vec2s aabb1[2], vec2s aabb2[2], vec2s dest[2]) { + vec2 rawAabb1[2]; + vec2 rawAabb2[2]; + vec2 rawDest[2]; + + glms_vec2_(unpack)(rawAabb1, aabb1, 2); + glms_vec2_(unpack)(rawAabb2, aabb2, 2); + glm_aabb2d_merge(rawAabb1, rawAabb2, rawDest); + glms_vec2_(pack)(dest, rawDest, 2); +} + +/*! + * @brief crops a bounding box with another one. + * + * this could be useful for getting a bbox which fits with view frustum and + * object bounding boxes. In this case you crop view frustum box with objects + * box + * + * @param[in] aabb bounding box 1 + * @param[in] cropAabb crop box + * @param[out] dest cropped bounding box + */ +CGLM_INLINE +void +glms_aabb2d_(crop)(vec2s aabb[2], vec2s cropAabb[2], vec2s dest[2]) { + vec2 rawAabb[2]; + vec2 rawCropAabb[2]; + vec2 rawDest[2]; + + glms_vec2_(unpack)(rawAabb, aabb, 2); + glms_vec2_(unpack)(rawCropAabb, cropAabb, 2); + glm_aabb2d_crop(rawAabb, rawCropAabb, rawDest); + glms_vec2_(pack)(dest, rawDest, 2); +} + +/*! + * @brief crops a bounding box with another one. + * + * this could be useful for getting a bbox which fits with view frustum and + * object bounding boxes. In this case you crop view frustum box with objects + * box + * + * @param[in] aabb bounding box + * @param[in] cropAabb crop box + * @param[in] clampAabb minimum box + * @param[out] dest cropped bounding box + */ +CGLM_INLINE +void +glms_aabb2d_(crop_until)(vec2s aabb[2], + vec2s cropAabb[2], + vec2s clampAabb[2], + vec2s dest[2]) { + glms_aabb2d_(crop)(aabb, cropAabb, dest); + glms_aabb2d_(merge)(clampAabb, dest, dest); +} + +/*! + * @brief invalidate AABB min and max values + * + * @param[in, out] aabb bounding box + */ +CGLM_INLINE +void +glms_aabb2d_(invalidate)(vec2s box[2]) { + box[0] = glms_vec2_(fill)(FLT_MAX); + box[1] = glms_vec2_(fill)(-FLT_MAX); +} + +/*! + * @brief check if AABB is valid or not + * + * @param[in] aabb bounding box + */ +CGLM_INLINE +bool +glms_aabb2d_(isvalid)(vec2s aabb[2]) { + vec2 rawAabb[2]; + glms_vec2_(unpack)(rawAabb, aabb, 2); + return glm_aabb2d_isvalid(rawAabb); +} + +/*! + * @brief distance between of min and max + * + * @param[in] aabb bounding box + */ +CGLM_INLINE +float +glms_aabb2d_(diag)(vec2s aabb[2]) { + vec2 rawAabb[2]; + glms_vec2_(unpack)(rawAabb, aabb, 2); + return glm_aabb2d_diag(rawAabb); +} + + +/*! + * @brief size of aabb + * + * @param[in] aabb bounding aabb + * @param[out] dest size + */ +CGLM_INLINE +vec2s +glms_aabb2d_(sizev)(vec2s aabb[2]) { + vec2s size; + vec2 rawAabb[2]; + glms_vec2_(unpack)(rawAabb, aabb, 2); + glm_aabb2d_sizev(rawAabb, size.raw); + return size; +} + +/*! + * @brief radius of sphere which surrounds AABB + * + * @param[in] aabb bounding box + */ +CGLM_INLINE +float +glms_aabb2d_(radius)(vec2s aabb[2]) { + return glms_aabb2d_(size)(aabb) * 0.5f; +} + +/*! + * @brief computes center point of AABB + * + * @param[in] aabb bounding box + * @returns center of bounding box + */ +CGLM_INLINE +vec2s +glms_aabb2d_(center)(vec2s aabb[2]) { + return glms_vec2_(center)(aabb[0], aabb[1]); +} + +/*! + * @brief check if two AABB intersects + * + * @param[in] aabb bounding box + * @param[in] other other bounding box + */ +CGLM_INLINE +bool +glms_aabb2d_(aabb)(vec2s aabb[2], vec2s other[2]) { + vec2 rawAabb[2]; + vec2 rawOther[2]; + + glms_vec2_(unpack)(rawAabb, aabb, 2); + glms_vec2_(unpack)(rawOther, other, 2); + return glm_aabb2d_aabb(rawAabb, rawOther); +} + +/*! + * @brief check if AABB intersects with a circle + * + * https://github.com/erich666/GraphicsGems/blob/master/gems/BoxSphere.c + * Solid Box - Solid Sphere test. + * + * @param[in] aabb solid bounding box + * @param[in] s solid sphere + */ +CGLM_INLINE +bool +glms_aabb2d_(circle)(vec2s aabb[2], vec3s c) { + vec2 rawAabb[2]; + + glms_vec2_(unpack)(rawAabb, aabb, 2); + return glm_aabb2d_circle(rawAabb, c.raw); +} + +/*! + * @brief check if point is inside of AABB + * + * @param[in] aabb bounding box + * @param[in] point point + */ +CGLM_INLINE +bool +glms_aabb2d_(point)(vec2s aabb[2], vec2s point) { + vec2 rawAabb[2]; + + glms_vec2_(unpack)(rawAabb, aabb, 2); + return glm_aabb2d_point(rawAabb, point.raw); +} + +/*! + * @brief check if AABB contains other AABB + * + * @param[in] box bounding box + * @param[in] other other bounding box + */ +CGLM_INLINE +bool +glms_aabb2d_(contains)(vec2s aabb[2], vec2s other[2]) { + vec2 rawAabb[2]; + vec2 rawOther[2]; + + glms_vec2_(unpack)(rawAabb, aabb, 2); + glms_vec2_(unpack)(rawOther, other, 2); + return glm_aabb2d_contains(rawAabb, rawOther); +} + +#endif /* cglms_aabb2ds_h */ diff --git a/cglm/include/cglm/struct/affine-mat.h b/cglm/include/cglm/struct/affine-mat.h new file mode 100644 index 0000000..e1d4ff3 --- /dev/null +++ b/cglm/include/cglm/struct/affine-mat.h @@ -0,0 +1,90 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_mul(mat4 m1, mat4 m2); + CGLM_INLINE mat4s glms_mul_rot(mat4 m1, mat4 m2); + CGLM_INLINE mat4s glms_inv_tr(); + */ + +#ifndef cglms_affine_mat_h +#define cglms_affine_mat_h + +#include "../common.h" +#include "../types-struct.h" +#include "../affine-mat.h" +#include "vec3.h" +#include "vec4.h" +#include "mat4.h" + +/*! + * @brief this is similar to glms_mat4_mul but specialized to affine transform + * + * Matrix format should be: + * R R R X + * R R R Y + * R R R Z + * 0 0 0 W + * + * this reduces some multiplications. It should be faster than mat4_mul. + * if you are not sure about matrix format then DON'T use this! use mat4_mul + * + * @param[in] m1 affine matrix 1 + * @param[in] m2 affine matrix 2 + * @returns destination matrix + */ +CGLM_INLINE +mat4s +glms_mul(mat4s m1, mat4s m2){ + mat4s r; + glm_mul(m1.raw, m2.raw, r.raw); + return r; +} + +/*! + * @brief this is similar to glm_mat4_mul but specialized to affine transform + * + * Right Matrix format should be: + * R R R 0 + * R R R 0 + * R R R 0 + * 0 0 0 1 + * + * this reduces some multiplications. It should be faster than mat4_mul. + * if you are not sure about matrix format then DON'T use this! use mat4_mul + * + * @param[in] m1 affine matrix 1 + * @param[in] m2 affine matrix 2 + * @returns destination matrix + */ +CGLM_INLINE +mat4s +glms_mul_rot(mat4s m1, mat4s m2){ + mat4s r; + glm_mul_rot(m1.raw, m2.raw, r.raw); + return r; +} + +/*! + * @brief inverse orthonormal rotation + translation matrix (ridig-body) + * + * @code + * X = | R T | X' = | R' -R'T | + * | 0 1 | | 0 1 | + * @endcode + * + * @param[in] m matrix + * @returns destination matrix + */ +CGLM_INLINE +mat4s +glms_inv_tr(mat4s m){ + glm_inv_tr(m.raw); + return m; +} +#endif /* cglms_affine_mat_h */ diff --git a/cglm/include/cglm/struct/affine-post.h b/cglm/include/cglm/struct/affine-post.h new file mode 100644 index 0000000..e155660 --- /dev/null +++ b/cglm/include/cglm/struct/affine-post.h @@ -0,0 +1,184 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_translated(mat4s m, vec3s v); + CGLM_INLINE mat4s glms_translated_x(mat4s m, float x); + CGLM_INLINE mat4s glms_translated_y(mat4s m, float y); + CGLM_INLINE mat4s glms_translated_z(mat4s m, float z); + CGLM_INLINE mat4s glms_rotated_x(mat4s m, float angle); + CGLM_INLINE mat4s glms_rotated_y(mat4s m, float angle); + CGLM_INLINE mat4s glms_rotated_z(mat4s m, float angle); + CGLM_INLINE mat4s glms_rotated(mat4s m, float angle, vec3s axis); + CGLM_INLINE mat4s glms_rotated_at(mat4s m, vec3s pivot, float angle, vec3s axis); + CGLM_INLINE mat4s glms_spinned(mat4s m, float angle, vec3s axis); + */ + +#ifndef cglms_affines_post_h +#define cglms_affines_post_h + +#include "../common.h" +#include "../types-struct.h" +#include "../affine.h" +#include "vec3.h" +#include "vec4.h" +#include "mat4.h" + +/*! + * @brief translate existing transform matrix by v vector + * and stores result in same matrix + * + * @param[in] m affine transform + * @param[in] v translate vector [x, y, z] + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_translated(mat4s m, vec3s v) { + glm_translated(m.raw, v.raw); + return m; +} + +/*! + * @brief translate existing transform matrix by x factor + * + * @param[in] m affine transform + * @param[in] x x factor + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_translated_x(mat4s m, float x) { + glm_translated_x(m.raw, x); + return m; +} + +/*! + * @brief translate existing transform matrix by y factor + * + * @param[in] m affine transform + * @param[in] y y factor + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_translated_y(mat4s m, float y) { + glm_translated_y(m.raw, y); + return m; +} + +/*! + * @brief translate existing transform matrix by z factor + * + * @param[in] m affine transform + * @param[in] z z factor + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_translated_z(mat4s m, float z) { + glm_translated_z(m.raw, z); + return m; +} + +/*! + * @brief rotate existing transform matrix around X axis by angle + * and store result in dest + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @returns rotated matrix + */ +CGLM_INLINE +mat4s +glms_rotated_x(mat4s m, float angle) { + mat4s r; + glm_rotated_x(m.raw, angle, r.raw); + return r; +} + +/*! + * @brief rotate existing transform matrix around Y axis by angle + * and store result in dest + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @returns rotated matrix + */ +CGLM_INLINE +mat4s +glms_rotated_y(mat4s m, float angle) { + mat4s r; + glm_rotated_y(m.raw, angle, r.raw); + return r; +} + +/*! + * @brief rotate existing transform matrix around Z axis by angle + * and store result in dest + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @returns rotated matrix + */ +CGLM_INLINE +mat4s +glms_rotated_z(mat4s m, float angle) { + mat4s r; + glm_rotated_z(m.raw, angle, r.raw); + return r; +} + +/*! + * @brief rotate existing transform matrix around given axis by angle + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @param[in] axis axis + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_rotated(mat4s m, float angle, vec3s axis) { + glm_rotated(m.raw, angle, axis.raw); + return m; +} + +/*! + * @brief rotate existing transform + * around given axis by angle at given pivot point (rotation center) + * + * @param[in] m affine transform + * @param[in] pivot rotation center + * @param[in] angle angle (radians) + * @param[in] axis axis + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_rotated_at(mat4s m, vec3s pivot, float angle, vec3s axis) { + glm_rotated_at(m.raw, pivot.raw, angle, axis.raw); + return m; +} + +/*! + * @brief rotate existing transform matrix around given axis by angle around self (doesn't affected by position) + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @param[in] axis axis + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_spinned(mat4s m, float angle, vec3s axis) { + glm_spinned(m.raw, angle, axis.raw); + return m; +} + +#endif /* cglms_affines_post_h */ diff --git a/cglm/include/cglm/struct/affine-pre.h b/cglm/include/cglm/struct/affine-pre.h new file mode 100644 index 0000000..e323ffa --- /dev/null +++ b/cglm/include/cglm/struct/affine-pre.h @@ -0,0 +1,184 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_translate(mat4s m, vec3s v); + CGLM_INLINE mat4s glms_translate_x(mat4s m, float x); + CGLM_INLINE mat4s glms_translate_y(mat4s m, float y); + CGLM_INLINE mat4s glms_translate_z(mat4s m, float z); + CGLM_INLINE mat4s glms_rotate_x(mat4s m, float angle); + CGLM_INLINE mat4s glms_rotate_y(mat4s m, float angle); + CGLM_INLINE mat4s glms_rotate_z(mat4s m, float angle); + CGLM_INLINE mat4s glms_rotate(mat4s m, float angle, vec3s axis); + CGLM_INLINE mat4s glms_rotate_at(mat4s m, vec3s pivot, float angle, vec3s axis); + CGLM_INLINE mat4s glms_spin(mat4s m, float angle, vec3s axis); + */ + +#ifndef cglms_affines_pre_h +#define cglms_affines_pre_h + +#include "../common.h" +#include "../types-struct.h" +#include "../affine.h" +#include "vec3.h" +#include "vec4.h" +#include "mat4.h" + +/*! + * @brief translate existing transform matrix by v vector + * and stores result in same matrix + * + * @param[in] m affine transform + * @param[in] v translate vector [x, y, z] + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_translate(mat4s m, vec3s v) { + glm_translate(m.raw, v.raw); + return m; +} + +/*! + * @brief translate existing transform matrix by x factor + * + * @param[in] m affine transform + * @param[in] x x factor + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_translate_x(mat4s m, float x) { + glm_translate_x(m.raw, x); + return m; +} + +/*! + * @brief translate existing transform matrix by y factor + * + * @param[in] m affine transform + * @param[in] y y factor + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_translate_y(mat4s m, float y) { + glm_translate_y(m.raw, y); + return m; +} + +/*! + * @brief translate existing transform matrix by z factor + * + * @param[in] m affine transform + * @param[in] z z factor + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_translate_z(mat4s m, float z) { + glm_translate_z(m.raw, z); + return m; +} + +/*! + * @brief rotate existing transform matrix around X axis by angle + * and store result in dest + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @returns rotated matrix + */ +CGLM_INLINE +mat4s +glms_rotate_x(mat4s m, float angle) { + mat4s r; + glm_rotate_x(m.raw, angle, r.raw); + return r; +} + +/*! + * @brief rotate existing transform matrix around Y axis by angle + * and store result in dest + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @returns rotated matrix + */ +CGLM_INLINE +mat4s +glms_rotate_y(mat4s m, float angle) { + mat4s r; + glm_rotate_y(m.raw, angle, r.raw); + return r; +} + +/*! + * @brief rotate existing transform matrix around Z axis by angle + * and store result in dest + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @returns rotated matrix + */ +CGLM_INLINE +mat4s +glms_rotate_z(mat4s m, float angle) { + mat4s r; + glm_rotate_z(m.raw, angle, r.raw); + return r; +} + +/*! + * @brief rotate existing transform matrix around given axis by angle + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @param[in] axis axis + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_rotate(mat4s m, float angle, vec3s axis) { + glm_rotate(m.raw, angle, axis.raw); + return m; +} + +/*! + * @brief rotate existing transform + * around given axis by angle at given pivot point (rotation center) + * + * @param[in] m affine transform + * @param[in] pivot rotation center + * @param[in] angle angle (radians) + * @param[in] axis axis + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_rotate_at(mat4s m, vec3s pivot, float angle, vec3s axis) { + glm_rotate_at(m.raw, pivot.raw, angle, axis.raw); + return m; +} + +/*! + * @brief rotate existing transform matrix around given axis by angle around self (doesn't affected by position) + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @param[in] axis axis + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_spin(mat4s m, float angle, vec3s axis) { + glm_spin(m.raw, angle, axis.raw); + return m; +} + +#endif /* cglms_affines_pre_h */ diff --git a/cglm/include/cglm/struct/affine.h b/cglm/include/cglm/struct/affine.h new file mode 100644 index 0000000..37f11be --- /dev/null +++ b/cglm/include/cglm/struct/affine.h @@ -0,0 +1,201 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_translate(mat4s m, vec3s v); + CGLM_INLINE mat4s glms_translate_x(mat4s m, float x); + CGLM_INLINE mat4s glms_translate_y(mat4s m, float y); + CGLM_INLINE mat4s glms_translate_z(mat4s m, float z); + CGLM_INLINE mat4s glms_translate_make(vec3s v); + CGLM_INLINE mat4s glms_scale_to(mat4s m, vec3s v); + CGLM_INLINE mat4s glms_scale_make(vec3s v); + CGLM_INLINE mat4s glms_scale(mat4s m, vec3s v); + CGLM_INLINE mat4s glms_scale_uni(mat4s m, float s); + CGLM_INLINE mat4s glms_rotate_x(mat4s m, float angle); + CGLM_INLINE mat4s glms_rotate_y(mat4s m, float angle); + CGLM_INLINE mat4s glms_rotate_z(mat4s m, float angle); + CGLM_INLINE mat4s glms_rotate_make(float angle, vec3s axis); + CGLM_INLINE mat4s glms_rotate(mat4s m, float angle, vec3s axis); + CGLM_INLINE mat4s glms_rotate_at(mat4s m, vec3s pivot, float angle, vec3s axis); + CGLM_INLINE mat4s glms_rotate_atm(vec3s pivot, float angle, vec3s axis); + CGLM_INLINE mat4s glms_spin(mat4s m, float angle, vec3s axis); + CGLM_INLINE vec3s glms_decompose_scalev(mat4s m); + CGLM_INLINE bool glms_uniscaled(mat4s m); + CGLM_INLINE void glms_decompose_rs(mat4s m, mat4s * r, vec3s * s); + CGLM_INLINE void glms_decompose(mat4s m, vec4s t, mat4s * r, vec3s * s); + */ + +#ifndef cglms_affines_h +#define cglms_affines_h + +#include "../common.h" +#include "../types-struct.h" +#include "../affine.h" +#include "vec3.h" +#include "vec4.h" +#include "mat4.h" +#include "affine-mat.h" + +/*! + * @brief creates NEW translate transform matrix by v vector + * + * @param[in] v translate vector [x, y, z] + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_translate_make(vec3s v) { + mat4s m; + glm_translate_make(m.raw, v.raw); + return m; +} + +/*! + * @brief creates NEW scale matrix by v vector + * + * @param[in] v scale vector [x, y, z] + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_scale_make(vec3s v) { + mat4s m; + glm_scale_make(m.raw, v.raw); + return m; +} + +/*! + * @brief scales existing transform matrix by v vector + * and stores result in same matrix + * + * @param[in] m affine transform + * @param[in] v scale vector [x, y, z] + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_scale(mat4s m, vec3s v) { + mat4s r; + glm_scale_to(m.raw, v.raw, r.raw); + return r; +} + +/*! + * @brief applies uniform scale to existing transform matrix v = [s, s, s] + * and stores result in same matrix + * + * @param[in] m affine transform + * @param[in] s scale factor + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_scale_uni(mat4s m, float s) { + glm_scale_uni(m.raw, s); + return m; +} + +/*! + * @brief creates NEW rotation matrix by angle and axis + * + * axis will be normalized so you don't need to normalize it + * + * @param[in] angle angle (radians) + * @param[in] axis axis + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_rotate_make(float angle, vec3s axis) { + mat4s m; + glm_rotate_make(m.raw, angle, axis.raw); + return m; +} + +/*! + * @brief creates NEW rotation matrix by angle and axis at given point + * + * this creates rotation matrix, it assumes you don't have a matrix + * + * this should work faster than glm_rotate_at because it reduces + * one glm_translate. + * + * @param[in] pivot rotation center + * @param[in] angle angle (radians) + * @param[in] axis axis + * @returns affine transform + */ +CGLM_INLINE +mat4s +glms_rotate_atm(vec3s pivot, float angle, vec3s axis) { + mat4s m; + glm_rotate_atm(m.raw, pivot.raw, angle, axis.raw); + return m; +} + +/*! + * @brief decompose scale vector + * + * @param[in] m affine transform + * @returns scale vector (Sx, Sy, Sz) + */ +CGLM_INLINE +vec3s +glms_decompose_scalev(mat4s m) { + vec3s r; + glm_decompose_scalev(m.raw, r.raw); + return r; +} + +/*! + * @brief returns true if matrix is uniform scaled. This is helpful for + * creating normal matrix. + * + * @param[in] m m + * + * @return boolean + */ +CGLM_INLINE +bool +glms_uniscaled(mat4s m) { + return glm_uniscaled(m.raw); +} + +/*! + * @brief decompose rotation matrix (mat4) and scale vector [Sx, Sy, Sz] + * DON'T pass projected matrix here + * + * @param[in] m affine transform + * @param[out] r rotation matrix + * @param[out] s scale matrix + */ +CGLM_INLINE +void +glms_decompose_rs(mat4s m, mat4s * __restrict r, vec3s * __restrict s) { + glm_decompose_rs(m.raw, r->raw, s->raw); +} + +/*! + * @brief decompose affine transform, TODO: extract shear factors. + * DON'T pass projected matrix here + * + * @param[in] m affine transform + * @param[out] t translation vector + * @param[out] r rotation matrix (mat4) + * @param[out] s scaling vector [X, Y, Z] + */ +CGLM_INLINE +void +glms_decompose(mat4s m, vec4s * __restrict t, mat4s * __restrict r, vec3s * __restrict s) { + glm_decompose(m.raw, t->raw, r->raw, s->raw); +} + +#include "affine-pre.h" +#include "affine-post.h" + +#endif /* cglms_affines_h */ diff --git a/cglm/include/cglm/struct/affine2d.h b/cglm/include/cglm/struct/affine2d.h new file mode 100644 index 0000000..ade7c32 --- /dev/null +++ b/cglm/include/cglm/struct/affine2d.h @@ -0,0 +1,177 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat3s glms_translate2d(mat3 m, vec2 v) + CGLM_INLINE mat3s glms_translate2d_x(mat3s m, float x) + CGLM_INLINE mat3s glms_translate2d_y(mat3s m, float y) + CGLM_INLINE mat3s glms_translate2d_make(vec2s v) + CGLM_INLINE mat3s glms_scale2d_make(vec2s v) + CGLM_INLINE mat3s glms_scale2d(mat3s m, vec2s v) + CGLM_INLINE mat3s glms_scale2d_uni(mat3s m, float s) + CGLM_INLINE mat3s glms_rotate2d_make(float angle) + CGLM_INLINE mat3s glms_rotate2d(mat3s m, float angle) + CGLM_INLINE mat3s glms_rotate2d_to(mat3s m, float angle) + */ + +#ifndef cglms_affine2ds_h +#define cglms_affine2ds_h + +#include "../common.h" +#include "../types-struct.h" +#include "../affine2d.h" +#include "vec3.h" +#include "mat3.h" + +/*! + * @brief translate existing 2d transform matrix by v vector + * and stores result in same matrix + * + * @param[in] m affine transform + * @param[in] v translate vector [x, y] + * @returns affine transform + */ +CGLM_INLINE +mat3s +glms_translate2d(mat3s m, vec2s v) { + glm_translate2d(m.raw, v.raw); + return m; +} + +/*! + * @brief translate existing 2d transform matrix by x factor + * + * @param[in] m affine transform + * @param[in] x x factor + * @returns affine transform + */ +CGLM_INLINE +mat3s +glms_translate2d_x(mat3s m, float x) { + glm_translate2d_x(m.raw, x); + return m; +} + +/*! + * @brief translate existing 2d transform matrix by y factor + * + * @param[in] m affine transform + * @param[in] y y factor + * @returns affine transform + */ +CGLM_INLINE +mat3s +glms_translate2d_y(mat3s m, float y) { + glm_translate2d_y(m.raw, y); + return m; +} + +/*! + * @brief creates NEW translate 2d transform matrix by v vector + * + * @param[in] v translate vector [x, y] + * @returns affine transform + */ +CGLM_INLINE +mat3s +glms_translate2d_make(vec2s v) { + mat3s m; + glm_translate2d_make(m.raw, v.raw); + return m; +} + +/*! + * @brief creates NEW 2d scale matrix by v vector + * + * @param[in] v scale vector [x, y] + * @returns affine transform + */ +CGLM_INLINE +mat3s +glms_scale2d_make(vec2s v) { + mat3s m; + glm_scale2d_make(m.raw, v.raw); + return m; +} + +/*! + * @brief scales existing 2d transform matrix by v vector + * and stores result in same matrix + * + * @param[in] m affine transform + * @param[in] v scale vector [x, y, z] + * @returns affine transform + */ +CGLM_INLINE +mat3s +glms_scale2d(mat3s m, vec2s v) { + mat3s r; + glm_scale2d_to(m.raw, v.raw, r.raw); + return r; +} + +/*! + * @brief applies uniform scale to existing 2d transform matrix v = [s, s, s] + * and stores result in same matrix + * + * @param[in] m affine transform + * @param[in] s scale factor + * @returns affine transform + */ +CGLM_INLINE +mat3s +glms_scale2d_uni(mat3s m, float s) { + glm_scale2d_uni(m.raw, s); + return m; +} + +/*! + * @brief creates NEW 2d rotation matrix by angle and axis + * + * axis will be normalized so you don't need to normalize it + * + * @param[in] angle angle (radians) + * @returns affine transform + */ +CGLM_INLINE +mat3s +glms_rotate2d_make(float angle) { + mat3s m; + glm_rotate2d_make(m.raw, angle); + return m; +} + +/*! + * @brief rotate existing 2d transform matrix around given axis by angle + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @returns affine transform + */ +CGLM_INLINE +mat3s +glms_rotate2d(mat3s m, float angle) { + glm_rotate2d(m.raw, angle); + return m; +} + +/*! + * @brief rotate existing 2d transform matrix around given axis by angle + * + * @param[in] m affine transform + * @param[in] angle angle (radians) + * @returns affine transform + */ +CGLM_INLINE +mat3s +glms_rotate2d_to(mat3s m, float angle) { + glm_rotate2d(m.raw, angle); + return m; +} + +#endif /* cglms_affine2ds_h */ diff --git a/cglm/include/cglm/struct/box.h b/cglm/include/cglm/struct/box.h new file mode 100644 index 0000000..ac32328 --- /dev/null +++ b/cglm/include/cglm/struct/box.h @@ -0,0 +1,259 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglms_boxs_h +#define cglms_boxs_h + +#include "../common.h" +#include "../types-struct.h" +#include "../box.h" +#include "vec3.h" +#include "vec4.h" +#include "mat4.h" + +/* api definition */ +#define glms_aabb_(NAME) CGLM_STRUCTAPI(aabb, NAME) + +/*! + * @brief apply transform to Axis-Aligned Bounding Box + * + * @param[in] box bounding box + * @param[in] m transform matrix + * @param[out] dest transformed bounding box + */ +CGLM_INLINE +void +glms_aabb_(transform)(vec3s box[2], mat4s m, vec3s dest[2]) { + vec3 rawBox[2]; + vec3 rawDest[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glm_aabb_transform(rawBox, m.raw, rawDest); + glms_vec3_(pack)(dest, rawDest, 2); +} + +/*! + * @brief merges two AABB bounding box and creates new one + * + * two box must be in same space, if one of box is in different space then + * you should consider to convert it's space by glm_box_space + * + * @param[in] box1 bounding box 1 + * @param[in] box2 bounding box 2 + * @param[out] dest merged bounding box + */ +CGLM_INLINE +void +glms_aabb_(merge)(vec3s box1[2], vec3s box2[2], vec3s dest[2]) { + vec3 rawBox1[2]; + vec3 rawBox2[2]; + vec3 rawDest[2]; + + glms_vec3_(unpack)(rawBox1, box1, 2); + glms_vec3_(unpack)(rawBox2, box2, 2); + glm_aabb_merge(rawBox1, rawBox2, rawDest); + glms_vec3_(pack)(dest, rawDest, 2); +} + +/*! + * @brief crops a bounding box with another one. + * + * this could be useful for getting a bbox which fits with view frustum and + * object bounding boxes. In this case you crop view frustum box with objects + * box + * + * @param[in] box bounding box 1 + * @param[in] cropBox crop box + * @param[out] dest cropped bounding box + */ +CGLM_INLINE +void +glms_aabb_(crop)(vec3s box[2], vec3s cropBox[2], vec3s dest[2]) { + vec3 rawBox[2]; + vec3 rawCropBox[2]; + vec3 rawDest[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glms_vec3_(unpack)(rawCropBox, cropBox, 2); + glm_aabb_crop(rawBox, rawCropBox, rawDest); + glms_vec3_(pack)(dest, rawDest, 2); +} + +/*! + * @brief crops a bounding box with another one. + * + * this could be useful for getting a bbox which fits with view frustum and + * object bounding boxes. In this case you crop view frustum box with objects + * box + * + * @param[in] box bounding box + * @param[in] cropBox crop box + * @param[in] clampBox minimum box + * @param[out] dest cropped bounding box + */ +CGLM_INLINE +void +glms_aabb_(crop_until)(vec3s box[2], + vec3s cropBox[2], + vec3s clampBox[2], + vec3s dest[2]) { + glms_aabb_(crop)(box, cropBox, dest); + glms_aabb_(merge)(clampBox, dest, dest); +} + +/*! + * @brief check if AABB intersects with frustum planes + * + * this could be useful for frustum culling using AABB. + * + * OPTIMIZATION HINT: + * if planes order is similar to LEFT, RIGHT, BOTTOM, TOP, NEAR, FAR + * then this method should run even faster because it would only use two + * planes if object is not inside the two planes + * fortunately cglm extracts planes as this order! just pass what you got! + * + * @param[in] box bounding box + * @param[in] planes frustum planes + */ +CGLM_INLINE +bool +glms_aabb_(frustum)(vec3s box[2], vec4s planes[6]) { + vec3 rawBox[2]; + vec4 rawPlanes[6]; + + glms_vec3_(unpack)(rawBox, box, 2); + glms_vec4_(unpack)(rawPlanes, planes, 6); + return glm_aabb_frustum(rawBox, rawPlanes); +} + +/*! + * @brief invalidate AABB min and max values + * + * @param[in, out] box bounding box + */ +CGLM_INLINE +void +glms_aabb_(invalidate)(vec3s box[2]) { + box[0] = glms_vec3_(broadcast)(FLT_MAX); + box[1] = glms_vec3_(broadcast)(-FLT_MAX); +} + +/*! + * @brief check if AABB is valid or not + * + * @param[in] box bounding box + */ +CGLM_INLINE +bool +glms_aabb_(isvalid)(vec3s box[2]) { + vec3 rawBox[2]; + glms_vec3_(unpack)(rawBox, box, 2); + return glm_aabb_isvalid(rawBox); +} + +/*! + * @brief distance between of min and max + * + * @param[in] box bounding box + */ +CGLM_INLINE +float +glms_aabb_(size)(vec3s box[2]) { + return glm_vec3_distance(box[0].raw, box[1].raw); +} + +/*! + * @brief radius of sphere which surrounds AABB + * + * @param[in] box bounding box + */ +CGLM_INLINE +float +glms_aabb_(radius)(vec3s box[2]) { + return glms_aabb_(size)(box) * 0.5f; +} + +/*! + * @brief computes center point of AABB + * + * @param[in] box bounding box + * @returns center of bounding box + */ +CGLM_INLINE +vec3s +glms_aabb_(center)(vec3s box[2]) { + return glms_vec3_(center)(box[0], box[1]); +} + +/*! + * @brief check if two AABB intersects + * + * @param[in] box bounding box + * @param[in] other other bounding box + */ +CGLM_INLINE +bool +glms_aabb_(aabb)(vec3s box[2], vec3s other[2]) { + vec3 rawBox[2]; + vec3 rawOther[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glms_vec3_(unpack)(rawOther, other, 2); + return glm_aabb_aabb(rawBox, rawOther); +} + +/*! + * @brief check if AABB intersects with sphere + * + * https://github.com/erich666/GraphicsGems/blob/master/gems/BoxSphere.c + * Solid Box - Solid Sphere test. + * + * @param[in] box solid bounding box + * @param[in] s solid sphere + */ +CGLM_INLINE +bool +glms_aabb_(sphere)(vec3s box[2], vec4s s) { + vec3 rawBox[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + return glm_aabb_sphere(rawBox, s.raw); +} + +/*! + * @brief check if point is inside of AABB + * + * @param[in] box bounding box + * @param[in] point point + */ +CGLM_INLINE +bool +glms_aabb_(point)(vec3s box[2], vec3s point) { + vec3 rawBox[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + return glm_aabb_point(rawBox, point.raw); +} + +/*! + * @brief check if AABB contains other AABB + * + * @param[in] box bounding box + * @param[in] other other bounding box + */ +CGLM_INLINE +bool +glms_aabb_(contains)(vec3s box[2], vec3s other[2]) { + vec3 rawBox[2]; + vec3 rawOther[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glms_vec3_(unpack)(rawOther, other, 2); + return glm_aabb_contains(rawBox, rawOther); +} + +#endif /* cglms_boxs_h */ diff --git a/cglm/include/cglm/struct/cam.h b/cglm/include/cglm/struct/cam.h new file mode 100644 index 0000000..ab6cbbb --- /dev/null +++ b/cglm/include/cglm/struct/cam.h @@ -0,0 +1,646 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_frustum(float left, float right, + float bottom, float top, + float nearZ, float farZ) + CGLM_INLINE mat4s glms_ortho(float left, float right, + float bottom, float top, + float nearZ, float farZ) + CGLM_INLINE mat4s glms_ortho_aabb(vec3s box[2]); + CGLM_INLINE mat4s glms_ortho_aabb_p(vec3s box[2], float padding); + CGLM_INLINE mat4s glms_ortho_aabb_pz(vec3s box[2], float padding); + CGLM_INLINE mat4s glms_ortho_default(float aspect) + CGLM_INLINE mat4s glms_ortho_default_s(float aspect, float size) + CGLM_INLINE mat4s glms_perspective(float fovy, + float aspect, + float nearZ, + float farZ) + CGLM_INLINE void glms_persp_move_far(mat4s proj, float deltaFar) + CGLM_INLINE mat4s glms_perspective_default(float aspect) + CGLM_INLINE void glms_perspective_resize(mat4s proj, float aspect) + CGLM_INLINE mat4s glms_lookat(vec3s eye, vec3s center, vec3s up) + CGLM_INLINE mat4s glms_look(vec3s eye, vec3s dir, vec3s up) + CGLM_INLINE mat4s glms_look_anyup(vec3s eye, vec3s dir) + CGLM_INLINE void glms_persp_decomp(mat4s proj, + float *nearv, float *farv, + float *top, float *bottom, + float *left, float *right) + CGLM_INLINE void glms_persp_decompv(mat4s proj, float dest[6]) + CGLM_INLINE void glms_persp_decomp_x(mat4s proj, float *left, float *right) + CGLM_INLINE void glms_persp_decomp_y(mat4s proj, float *top, float *bottom) + CGLM_INLINE void glms_persp_decomp_z(mat4s proj, float *nearv, float *farv) + CGLM_INLINE void glms_persp_decomp_far(mat4s proj, float *farZ) + CGLM_INLINE void glms_persp_decomp_near(mat4s proj, float *nearZ) + CGLM_INLINE float glms_persp_fovy(mat4s proj) + CGLM_INLINE float glms_persp_aspect(mat4s proj) + CGLM_INLINE vec4s glms_persp_sizes(mat4s proj, float fovy) + */ + +#ifndef cglms_cam_h +#define cglms_cam_h + +#include "../common.h" +#include "../types-struct.h" +#include "../plane.h" +#include "../cam.h" + +#ifndef CGLM_CLIPSPACE_INCLUDE_ALL +# if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO +# include "clipspace/ortho_lh_zo.h" +# include "clipspace/persp_lh_zo.h" +# include "clipspace/view_lh_zo.h" +# elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO +# include "clipspace/ortho_lh_no.h" +# include "clipspace/persp_lh_no.h" +# include "clipspace/view_lh_no.h" +# elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO +# include "clipspace/ortho_rh_zo.h" +# include "clipspace/persp_rh_zo.h" +# include "clipspace/view_rh_zo.h" +# elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO +# include "clipspace/ortho_rh_no.h" +# include "clipspace/persp_rh_no.h" +# include "clipspace/view_rh_no.h" +# endif +#else +# include "clipspace/ortho_lh_zo.h" +# include "clipspace/persp_lh_zo.h" +# include "clipspace/ortho_lh_no.h" +# include "clipspace/persp_lh_no.h" +# include "clipspace/ortho_rh_zo.h" +# include "clipspace/persp_rh_zo.h" +# include "clipspace/ortho_rh_no.h" +# include "clipspace/persp_rh_no.h" +# include "clipspace/view_lh_zo.h" +# include "clipspace/view_lh_no.h" +# include "clipspace/view_rh_zo.h" +# include "clipspace/view_rh_no.h" +#endif + +/*! + * @brief set up perspective peprojection matrix + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_frustum(float left, float right, + float bottom, float top, + float nearZ, float farZ) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_frustum_lh_zo(left, right, bottom, top, nearZ, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_frustum_lh_no(left, right, bottom, top, nearZ, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_frustum_rh_zo(left, right, bottom, top, nearZ, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_frustum_rh_no(left, right, bottom, top, nearZ, farZ); +#endif +} + +/*! + * @brief set up orthographic projection matrix + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho(float left, float right, + float bottom, float top, + float nearZ, float farZ) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_ortho_lh_zo(left, right, bottom, top, nearZ, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_ortho_lh_no(left, right, bottom, top, nearZ, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_ortho_rh_zo(left, right, bottom, top, nearZ, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_ortho_rh_no(left, right, bottom, top, nearZ, farZ); +#endif +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_aabb(vec3s box[2]) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_ortho_aabb_lh_zo(box); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_ortho_aabb_lh_no(box); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_ortho_aabb_rh_zo(box); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_ortho_aabb_rh_no(box); +#endif +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_aabb_p(vec3s box[2], float padding) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_ortho_aabb_p_lh_zo(box, padding); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_ortho_aabb_p_lh_no(box, padding); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_ortho_aabb_p_rh_zo(box, padding); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_ortho_aabb_p_rh_no(box, padding); +#endif +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding for near and far + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_aabb_pz(vec3s box[2], float padding) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_ortho_aabb_pz_lh_zo(box, padding); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_ortho_aabb_pz_lh_no(box, padding); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_ortho_aabb_pz_rh_zo(box, padding); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_ortho_aabb_pz_rh_no(box, padding); +#endif +} + +/*! + * @brief set up unit orthographic projection matrix + * + * @param[in] aspect aspect ration ( width / height ) + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_default(float aspect) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_ortho_default_lh_zo(aspect); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_ortho_default_lh_no(aspect); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_ortho_default_rh_zo(aspect); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_ortho_default_rh_no(aspect); +#endif +} + +/*! + * @brief set up orthographic projection matrix with given CUBE size + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] size cube size + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_default_s(float aspect, float size) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_ortho_default_s_lh_zo(aspect, size); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_ortho_default_s_lh_no(aspect, size); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_ortho_default_s_rh_zo(aspect, size); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_ortho_default_s_rh_no(aspect, size); +#endif +} + +/*! + * @brief set up perspective projection matrix + * + * @param[in] fovy field of view angle + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping planes + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_perspective(float fovy, float aspect, float nearZ, float farZ) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_perspective_lh_zo(fovy, aspect, nearZ, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_perspective_lh_no(fovy, aspect, nearZ, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_perspective_rh_zo(fovy, aspect, nearZ, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_perspective_rh_no(fovy, aspect, nearZ, farZ); +#endif +} + +/*! + * @brief extend perspective projection matrix's far distance + * + * NOTE: if you dodn't want to create new matrix then use array api on struct.raw + * like glm_persp_move_far(prooj.raw, deltaFar) to avoid create new mat4 + * each time + * + * this function does not guarantee far >= near, be aware of that! + * + * @param[in, out] proj projection matrix to extend + * @param[in] deltaFar distance from existing far (negative to shink) + */ +CGLM_INLINE +mat4s +glms_persp_move_far(mat4s proj, float deltaFar) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_persp_move_far_lh_zo(proj, deltaFar); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_persp_move_far_lh_no(proj, deltaFar); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_persp_move_far_rh_zo(proj, deltaFar); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_persp_move_far_rh_no(proj, deltaFar); +#endif +} + +/*! + * @brief set up perspective projection matrix with default near/far + * and angle values + * + * @param[in] aspect aspect ratio ( width / height ) + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_perspective_default(float aspect) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_perspective_default_lh_zo(aspect); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_perspective_default_lh_no(aspect); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_perspective_default_rh_zo(aspect); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_perspective_default_rh_no(aspect); +#endif +} + +/*! + * @brief resize perspective matrix by aspect ratio ( width / height ) + * this makes very easy to resize proj matrix when window /viewport + * reized + * + * NOTE: if you dodn't want to create new matrix then use array api on struct.raw + * like glms_perspective_resize(proj.raw, aspect) to avoid create new mat4 + * each time + * + * @param[in, out] proj perspective projection matrix + * @param[in] aspect aspect ratio ( width / height ) + */ +CGLM_INLINE +mat4s +glms_perspective_resize(mat4s proj, float aspect) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_perspective_resize_lh_zo(proj, aspect); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_perspective_resize_lh_no(proj, aspect); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_perspective_resize_rh_zo(proj, aspect); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_perspective_resize_rh_no(proj, aspect); +#endif +} + +/*! + * @brief set up view matrix + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] center center vector + * @param[in] up up vector + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_lookat(vec3s eye, vec3s center, vec3s up) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_lookat_lh_zo(eye, center, up); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_lookat_lh_no(eye, center, up); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_lookat_rh_zo(eye, center, up); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_lookat_rh_no(eye, center, up); +#endif +} + +/*! + * @brief set up view matrix + * + * convenient wrapper for lookat: if you only have direction not target self + * then this might be useful. Because you need to get target from direction. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[in] up up vector + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_look(vec3s eye, vec3s dir, vec3s up) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_look_lh_zo(eye, dir, up); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_look_lh_no(eye, dir, up); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_look_rh_zo(eye, dir, up); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_look_rh_no(eye, dir, up); +#endif +} + +/*! + * @brief set up view matrix + * + * convenient wrapper for look: if you only have direction and if you don't + * care what UP vector is then this might be useful to create view matrix + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_look_anyup(vec3s eye, vec3s dir) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_look_anyup_lh_zo(eye, dir); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_look_anyup_lh_no(eye, dir); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_look_anyup_rh_zo(eye, dir); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_look_anyup_rh_no(eye, dir); +#endif +} + +/*! + * @brief decomposes frustum values of perspective projection. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + * @param[out] top top + * @param[out] bottom bottom + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glms_persp_decomp(mat4s proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glms_persp_decomp_lh_zo(proj, nearZ, farZ, top, bottom, left, right); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glms_persp_decomp_lh_no(proj, nearZ, farZ, top, bottom, left, right); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glms_persp_decomp_rh_zo(proj, nearZ, farZ, top, bottom, left, right); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glms_persp_decomp_rh_no(proj, nearZ, farZ, top, bottom, left, right); +#endif +} + +/*! + * @brief decomposes frustum values of perspective projection. + * this makes easy to get all values at once + * + * @param[in] proj perspective projection matrix + * @param[out] dest array + */ +CGLM_INLINE +void +glms_persp_decompv(mat4s proj, float dest[6]) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glms_persp_decompv_lh_zo(proj, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glms_persp_decompv_lh_no(proj, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glms_persp_decompv_rh_zo(proj, dest); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glms_persp_decompv_rh_no(proj, dest); +#endif +} + +/*! + * @brief decomposes left and right values of perspective projection. + * x stands for x axis (left / right axis) + * + * @param[in] proj perspective projection matrix + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glms_persp_decomp_x(mat4s proj, + float * __restrict left, + float * __restrict right) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glms_persp_decomp_x_lh_zo(proj, left, right); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glms_persp_decomp_x_lh_no(proj, left, right); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glms_persp_decomp_x_rh_zo(proj, left, right); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glms_persp_decomp_x_rh_no(proj, left, right); +#endif +} + +/*! + * @brief decomposes top and bottom values of perspective projection. + * y stands for y axis (top / bottom axis) + * + * @param[in] proj perspective projection matrix + * @param[out] top top + * @param[out] bottom bottom + */ +CGLM_INLINE +void +glms_persp_decomp_y(mat4s proj, + float * __restrict top, + float * __restrict bottom) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glms_persp_decomp_y_lh_zo(proj, top, bottom); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glms_persp_decomp_y_lh_no(proj, top, bottom); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glms_persp_decomp_y_rh_zo(proj, top, bottom); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glms_persp_decomp_y_rh_no(proj, top, bottom); +#endif +} + +/*! + * @brief decomposes near and far values of perspective projection. + * z stands for z axis (near / far axis) + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + */ +CGLM_INLINE +void +glms_persp_decomp_z(mat4s proj, + float * __restrict nearZ, + float * __restrict farZ) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glms_persp_decomp_z_lh_zo(proj, nearZ, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glms_persp_decomp_z_lh_no(proj, nearZ, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glms_persp_decomp_z_rh_zo(proj, nearZ, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glms_persp_decomp_z_rh_no(proj, nearZ, farZ); +#endif +} + +/*! + * @brief decomposes far value of perspective projection. + * + * @param[in] proj perspective projection matrix + * @param[out] farZ far + */ +CGLM_INLINE +void +glms_persp_decomp_far(mat4s proj, float * __restrict farZ) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glms_persp_decomp_far_lh_zo(proj, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glms_persp_decomp_far_lh_no(proj, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glms_persp_decomp_far_rh_zo(proj, farZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glms_persp_decomp_far_rh_no(proj, farZ); +#endif +} + +/*! + * @brief decomposes near value of perspective projection. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + */ +CGLM_INLINE +void +glms_persp_decomp_near(mat4s proj, float * __restrict nearZ) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + glms_persp_decomp_near_lh_zo(proj, nearZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + glms_persp_decomp_near_lh_no(proj, nearZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + glms_persp_decomp_near_rh_zo(proj, nearZ); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + glms_persp_decomp_near_rh_no(proj, nearZ); +#endif +} + +/*! + * @brief returns field of view angle along the Y-axis (in radians) + * + * if you need to degrees, use glm_deg to convert it or use this: + * fovy_deg = glm_deg(glm_persp_fovy(projMatrix)) + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glms_persp_fovy(mat4s proj) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_persp_fovy_lh_zo(proj); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_persp_fovy_lh_no(proj); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_persp_fovy_rh_zo(proj); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_persp_fovy_rh_no(proj); +#endif +} + +/*! + * @brief returns aspect ratio of perspective projection + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glms_persp_aspect(mat4s proj) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_persp_aspect_lh_zo(proj); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_persp_aspect_lh_no(proj); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_persp_aspect_rh_zo(proj); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_persp_aspect_rh_no(proj); +#endif +} + +/*! + * @brief returns sizes of near and far planes of perspective projection + * + * @param[in] proj perspective projection matrix + * @param[in] fovy fovy (see brief) + * @returns sizes as vector, sizes order: [Wnear, Hnear, Wfar, Hfar] + */ +CGLM_INLINE +vec4s +glms_persp_sizes(mat4s proj, float fovy) { +#if CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_ZO + return glms_persp_sizes_lh_zo(proj, fovy); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_LH_NO + return glms_persp_sizes_lh_no(proj, fovy); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_ZO + return glms_persp_sizes_rh_zo(proj, fovy); +#elif CGLM_CONFIG_CLIP_CONTROL == CGLM_CLIP_CONTROL_RH_NO + return glms_persp_sizes_rh_no(proj, fovy); +#endif +} + +#endif /* cglms_cam_h */ diff --git a/cglm/include/cglm/struct/clipspace/ortho_lh_no.h b/cglm/include/cglm/struct/clipspace/ortho_lh_no.h new file mode 100644 index 0000000..a743fdf --- /dev/null +++ b/cglm/include/cglm/struct/clipspace/ortho_lh_no.h @@ -0,0 +1,154 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_ortho_lh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ) + CGLM_INLINE mat4s glms_ortho_aabb_lh_no(vec3s box[2]); + CGLM_INLINE mat4s glms_ortho_aabb_p_lh_no(vec3s box[2], float padding); + CGLM_INLINE mat4s glms_ortho_aabb_pz_lh_no(vec3s box[2], float padding); + CGLM_INLINE mat4s glms_ortho_default_lh_no(float aspect) + CGLM_INLINE mat4s glms_ortho_default_s_lh_no(float aspect, float size) + */ + +#ifndef cglms_ortho_lh_no_h +#define cglms_ortho_lh_no_h + +#include "../../common.h" +#include "../../types-struct.h" +#include "../../plane.h" +#include "../../cam.h" +#include "../vec3.h" +#include "../../clipspace/ortho_lh_no.h" + +/*! + * @brief set up orthographic projection matrix + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_lh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ) { + mat4s dest; + glm_ortho_lh_no(left, right, bottom, top, nearZ, farZ, dest.raw); + return dest; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_aabb_lh_no(vec3s box[2]) { + mat4s dest; + vec3 rawBox[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glm_ortho_aabb_lh_no(rawBox, dest.raw); + + return dest; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_aabb_p_lh_no(vec3s box[2], float padding) { + mat4s dest; + vec3 rawBox[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glm_ortho_aabb_p_lh_no(rawBox, padding, dest.raw); + + return dest; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding for near and far + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_aabb_pz_lh_no(vec3s box[2], float padding) { + mat4s dest; + vec3 rawBox[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glm_ortho_aabb_pz_lh_no(rawBox, padding, dest.raw); + + return dest; +} + +/*! + * @brief set up unit orthographic projection matrix + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] aspect aspect ration ( width / height ) + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_default_lh_no(float aspect) { + mat4s dest; + glm_ortho_default_lh_no(aspect, dest.raw); + return dest; +} + +/*! + * @brief set up orthographic projection matrix with given CUBE size + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] size cube size + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_default_s_lh_no(float aspect, float size) { + mat4s dest; + glm_ortho_default_s_lh_no(aspect, size, dest.raw); + return dest; +} + +#endif /* cglms_ortho_lh_no_h */ diff --git a/cglm/include/cglm/struct/clipspace/ortho_lh_zo.h b/cglm/include/cglm/struct/clipspace/ortho_lh_zo.h new file mode 100644 index 0000000..4f15656 --- /dev/null +++ b/cglm/include/cglm/struct/clipspace/ortho_lh_zo.h @@ -0,0 +1,154 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_ortho_lh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ) + CGLM_INLINE mat4s glms_ortho_aabb_lh_zo(vec3s box[2]); + CGLM_INLINE mat4s glms_ortho_aabb_p_lh_zo(vec3s box[2], float padding); + CGLM_INLINE mat4s glms_ortho_aabb_pz_lh_zo(vec3s box[2], float padding); + CGLM_INLINE mat4s glms_ortho_default_lh_zo(float aspect) + CGLM_INLINE mat4s glms_ortho_default_s_lh_zo(float aspect, float size) + */ + +#ifndef cglms_ortho_lh_zo_h +#define cglms_ortho_lh_zo_h + +#include "../../common.h" +#include "../../types-struct.h" +#include "../../plane.h" +#include "../../cam.h" +#include "../vec3.h" +#include "../../clipspace/ortho_lh_zo.h" + +/*! + * @brief set up orthographic projection matrix + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_lh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ) { + mat4s dest; + glm_ortho_lh_zo(left, right, bottom, top, nearZ, farZ, dest.raw); + return dest; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_aabb_lh_zo(vec3s box[2]) { + mat4s dest; + vec3 rawBox[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glm_ortho_aabb_lh_zo(rawBox, dest.raw); + + return dest; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_aabb_p_lh_zo(vec3s box[2], float padding) { + mat4s dest; + vec3 rawBox[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glm_ortho_aabb_p_lh_zo(rawBox, padding, dest.raw); + + return dest; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding for near and far + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_aabb_pz_lh_zo(vec3s box[2], float padding) { + mat4s dest; + vec3 rawBox[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glm_ortho_aabb_pz_lh_zo(rawBox, padding, dest.raw); + + return dest; +} + +/*! + * @brief set up unit orthographic projection matrix + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] aspect aspect ration ( width / height ) + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_default_lh_zo(float aspect) { + mat4s dest; + glm_ortho_default_lh_zo(aspect, dest.raw); + return dest; +} + +/*! + * @brief set up orthographic projection matrix with given CUBE size + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] size cube size + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_default_s_lh_zo(float aspect, float size) { + mat4s dest; + glm_ortho_default_s_lh_zo(aspect, size, dest.raw); + return dest; +} + +#endif /* cglms_ortho_lh_zo_h */ diff --git a/cglm/include/cglm/struct/clipspace/ortho_rh_no.h b/cglm/include/cglm/struct/clipspace/ortho_rh_no.h new file mode 100644 index 0000000..ecb4d32 --- /dev/null +++ b/cglm/include/cglm/struct/clipspace/ortho_rh_no.h @@ -0,0 +1,154 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_ortho_rh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ) + CGLM_INLINE mat4s glms_ortho_aabb_rh_no(vec3s box[2]); + CGLM_INLINE mat4s glms_ortho_aabb_p_rh_no(vec3s box[2], float padding); + CGLM_INLINE mat4s glms_ortho_aabb_pz_rh_no(vec3s box[2], float padding); + CGLM_INLINE mat4s glms_ortho_default_rh_no(float aspect) + CGLM_INLINE mat4s glms_ortho_default_s_rh_no(float aspect, float size) + */ + +#ifndef cglms_ortho_rh_no_h +#define cglms_ortho_rh_no_h + +#include "../../common.h" +#include "../../types-struct.h" +#include "../../plane.h" +#include "../../cam.h" +#include "../vec3.h" +#include "../../clipspace/ortho_rh_no.h" + +/*! + * @brief set up orthographic projection matrix + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_rh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ) { + mat4s dest; + glm_ortho_rh_no(left, right, bottom, top, nearZ, farZ, dest.raw); + return dest; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_aabb_rh_no(vec3s box[2]) { + mat4s dest; + vec3 rawBox[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glm_ortho_aabb_rh_no(rawBox, dest.raw); + + return dest; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_aabb_p_rh_no(vec3s box[2], float padding) { + mat4s dest; + vec3 rawBox[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glm_ortho_aabb_p_rh_no(rawBox, padding, dest.raw); + + return dest; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding for near and far + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_aabb_pz_rh_no(vec3s box[2], float padding) { + mat4s dest; + vec3 rawBox[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glm_ortho_aabb_pz_rh_no(rawBox, padding, dest.raw); + + return dest; +} + +/*! + * @brief set up unit orthographic projection matrix + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] aspect aspect ration ( width / height ) + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_default_rh_no(float aspect) { + mat4s dest; + glm_ortho_default_rh_no(aspect, dest.raw); + return dest; +} + +/*! + * @brief set up orthographic projection matrix with given CUBE size + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] size cube size + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_default_s_rh_no(float aspect, float size) { + mat4s dest; + glm_ortho_default_s_rh_no(aspect, size, dest.raw); + return dest; +} + +#endif /* cglms_ortho_rh_no_h */ diff --git a/cglm/include/cglm/struct/clipspace/ortho_rh_zo.h b/cglm/include/cglm/struct/clipspace/ortho_rh_zo.h new file mode 100644 index 0000000..2d50ee1 --- /dev/null +++ b/cglm/include/cglm/struct/clipspace/ortho_rh_zo.h @@ -0,0 +1,154 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_ortho_rh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ) + CGLM_INLINE mat4s glms_ortho_aabb_rh_zo(vec3s box[2]); + CGLM_INLINE mat4s glms_ortho_aabb_p_rh_zo(vec3s box[2], float padding); + CGLM_INLINE mat4s glms_ortho_aabb_pz_rh_zo(vec3s box[2], float padding); + CGLM_INLINE mat4s glms_ortho_default_rh_zo(float aspect) + CGLM_INLINE mat4s glms_ortho_default_s_rh_zo(float aspect, float size) + */ + +#ifndef cglms_ortho_rh_zo_h +#define cglms_ortho_rh_zo_h + +#include "../../common.h" +#include "../../types-struct.h" +#include "../../plane.h" +#include "../../cam.h" +#include "../vec3.h" +#include "../../clipspace/ortho_rh_zo.h" + +/*! + * @brief set up orthographic projection matrix + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_rh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ) { + mat4s dest; + glm_ortho_rh_zo(left, right, bottom, top, nearZ, farZ, dest.raw); + return dest; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_aabb_rh_zo(vec3s box[2]) { + mat4s dest; + vec3 rawBox[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glm_ortho_aabb_rh_zo(rawBox, dest.raw); + + return dest; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_aabb_p_rh_zo(vec3s box[2], float padding) { + mat4s dest; + vec3 rawBox[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glm_ortho_aabb_p_rh_zo(rawBox, padding, dest.raw); + + return dest; +} + +/*! + * @brief set up orthographic projection matrix using bounding box + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * bounding box (AABB) must be in view space + * + * @param[in] box AABB + * @param[in] padding padding for near and far + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_aabb_pz_rh_zo(vec3s box[2], float padding) { + mat4s dest; + vec3 rawBox[2]; + + glms_vec3_(unpack)(rawBox, box, 2); + glm_ortho_aabb_pz_rh_zo(rawBox, padding, dest.raw); + + return dest; +} + +/*! + * @brief set up unit orthographic projection matrix + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] aspect aspect ration ( width / height ) + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_default_rh_zo(float aspect) { + mat4s dest; + glm_ortho_default_rh_zo(aspect, dest.raw); + return dest; +} + +/*! + * @brief set up orthographic projection matrix with given CUBE size + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] size cube size + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_ortho_default_s_rh_zo(float aspect, float size) { + mat4s dest; + glm_ortho_default_s_rh_zo(aspect, size, dest.raw); + return dest; +} + +#endif /* cglms_ortho_rh_zo_h */ diff --git a/cglm/include/cglm/struct/clipspace/persp_lh_no.h b/cglm/include/cglm/struct/clipspace/persp_lh_no.h new file mode 100644 index 0000000..bc35ca0 --- /dev/null +++ b/cglm/include/cglm/struct/clipspace/persp_lh_no.h @@ -0,0 +1,312 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_frustum_lh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ) + CGLM_INLINE mat4s glms_perspective_lh_no(float fovy, + float aspect, + float nearZ, + float farZ) + CGLM_INLINE void glms_persp_move_far_lh_no(mat4s proj, float deltaFar) + CGLM_INLINE mat4s glms_perspective_default_lh_no(float aspect) + CGLM_INLINE void glms_perspective_resize_lh_no(mat4s proj, float aspect) + CGLM_INLINE void glms_persp_decomp_lh_no(mat4s proj, + float *nearv, float *farv, + float *top, float *bottom, + float *left, float *right) + CGLM_INLINE void glms_persp_decompv_lh_no(mat4s proj, float dest[6]) + CGLM_INLINE void glms_persp_decomp_x_lh_no(mat4s proj, float *left, float *right) + CGLM_INLINE void glms_persp_decomp_y_lh_no(mat4s proj, float *top, float *bottom) + CGLM_INLINE void glms_persp_decomp_z_lh_no(mat4s proj, float *nearv, float *farv) + CGLM_INLINE void glms_persp_decomp_far_lh_no(mat4s proj, float *farZ) + CGLM_INLINE void glms_persp_decomp_near_lh_no(mat4s proj, float *nearZ) + CGLM_INLINE float glms_persp_fovy_lh_no(mat4s proj) + CGLM_INLINE float glms_persp_aspect_lh_no(mat4s proj) + CGLM_INLINE vec4s glms_persp_sizes_lh_no(mat4s proj, float fovy) + */ + +#ifndef cglms_persp_lh_no_h +#define cglms_persp_lh_no_h + +#include "../../common.h" +#include "../../types-struct.h" +#include "../../plane.h" +#include "../../cam.h" +#include "../../clipspace/persp_lh_no.h" + +/*! + * @brief set up perspective peprojection matrix + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_frustum_lh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ) { + mat4s dest; + glm_frustum_lh_no(left, right, bottom, top, nearZ, farZ, dest.raw); + return dest; +} + +/*! + * @brief set up perspective projection matrix + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] fovy field of view angle + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping planes + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_perspective_lh_no(float fovy, float aspect, float nearZ, float farZ) { + mat4s dest; + glm_perspective_lh_no(fovy, aspect, nearZ, farZ, dest.raw); + return dest; +} + +/*! + * @brief extend perspective projection matrix's far distance + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * NOTE: if you dodn't want to create new matrix then use array api on struct.raw + * like glms_persp_move_far_lh_no(prooj.raw, deltaFar) to avoid create new mat4 + * each time + * + * this function does not guarantee far >= near, be aware of that! + * + * @param[in, out] proj projection matrix to extend + * @param[in] deltaFar distance from existing far (negative to shink) + */ +CGLM_INLINE +mat4s +glms_persp_move_far_lh_no(mat4s proj, float deltaFar) { + mat4s dest; + dest = proj; + glm_persp_move_far_lh_no(dest.raw, deltaFar); + return dest; +} + +/*! + * @brief set up perspective projection matrix with default near/far + * and angle values with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_perspective_default_lh_no(float aspect) { + mat4s dest; + glm_perspective_default_lh_no(aspect, dest.raw); + return dest; +} + +/*! + * @brief resize perspective matrix by aspect ratio ( width / height ) + * this makes very easy to resize proj matrix when window /viewport + * reized with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * NOTE: if you dodn't want to create new matrix then use array api on struct.raw + * like glm_perspective_resize_lh_no(proj.raw, aspect) to avoid create new mat4 + * each time + * + * @param[in, out] proj perspective projection matrix + * @param[in] aspect aspect ratio ( width / height ) + */ +CGLM_INLINE +mat4s +glms_perspective_resize_lh_no(mat4s proj, float aspect) { + mat4s dest; + dest = proj; + glm_perspective_resize_lh_no(aspect, dest.raw); + return dest; +} + +/*! + * @brief decomposes frustum values of perspective projection. + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + * @param[out] top top + * @param[out] bottom bottom + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glms_persp_decomp_lh_no(mat4s proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right) { + glm_persp_decomp_lh_no(proj.raw, nearZ, farZ, top, bottom, left, right); +} + +/*! + * @brief decomposes frustum values of perspective projection. + * this makes easy to get all values at once + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] dest array + */ +CGLM_INLINE +void +glms_persp_decompv_lh_no(mat4s proj, float dest[6]) { + glm_persp_decompv_lh_no(proj.raw, dest); +} + +/*! + * @brief decomposes left and right values of perspective projection + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * x stands for x axis (left / right axis) + * + * @param[in] proj perspective projection matrix + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glms_persp_decomp_x_lh_no(mat4s proj, + float * __restrict left, + float * __restrict right) { + glm_persp_decomp_x_lh_no(proj.raw, left, right); +} + +/*! + * @brief decomposes top and bottom values of perspective projection + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * y stands for y axis (top / bottom axis) + * + * @param[in] proj perspective projection matrix + * @param[out] top top + * @param[out] bottom bottom + */ +CGLM_INLINE +void +glms_persp_decomp_y_lh_no(mat4s proj, + float * __restrict top, + float * __restrict bottom) { + glm_persp_decomp_y_lh_no(proj.raw, top, bottom); +} + +/*! + * @brief decomposes near and far values of perspective projection + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * z stands for z axis (near / far axis) + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + */ +CGLM_INLINE +void +glms_persp_decomp_z_lh_no(mat4s proj, + float * __restrict nearZ, + float * __restrict farZ) { + glm_persp_decomp_z_lh_no(proj.raw, nearZ, farZ); +} + +/*! + * @brief decomposes far value of perspective projection + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] farZ far + */ +CGLM_INLINE +void +glms_persp_decomp_far_lh_no(mat4s proj, float * __restrict farZ) { + glm_persp_decomp_far_lh_no(proj.raw, farZ); +} + +/*! + * @brief decomposes near value of perspective projection + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + */ +CGLM_INLINE +void +glms_persp_decomp_near_lh_no(mat4s proj, float * __restrict nearZ) { + glm_persp_decomp_near_lh_no(proj.raw, nearZ); +} + +/*! + * @brief returns field of view angle along the Y-axis (in radians) + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * if you need to degrees, use glm_deg to convert it or use this: + * fovy_deg = glm_deg(glm_persp_fovy(projMatrix)) + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glms_persp_fovy_lh_no(mat4s proj) { + return glm_persp_fovy_lh_no(proj.raw); +} + +/*! + * @brief returns aspect ratio of perspective projection + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glms_persp_aspect_lh_no(mat4s proj) { + return glm_persp_aspect_lh_no(proj.raw); +} + +/*! + * @brief returns sizes of near and far planes of perspective projection + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[in] fovy fovy (see brief) + * @returns sizes as vector, sizes order: [Wnear, Hnear, Wfar, Hfar] + */ +CGLM_INLINE +vec4s +glms_persp_sizes_lh_no(mat4s proj, float fovy) { + vec4s dest; + glm_persp_sizes_lh_no(proj.raw, fovy, dest.raw); + return dest; +} + +#endif /* cglms_persp_lh_no_h */ diff --git a/cglm/include/cglm/struct/clipspace/persp_lh_zo.h b/cglm/include/cglm/struct/clipspace/persp_lh_zo.h new file mode 100644 index 0000000..29af065 --- /dev/null +++ b/cglm/include/cglm/struct/clipspace/persp_lh_zo.h @@ -0,0 +1,312 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_frustum_lh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ) + CGLM_INLINE mat4s glms_perspective_lh_zo(float fovy, + float aspect, + float nearZ, + float farZ) + CGLM_INLINE void glms_persp_move_far_lh_zo(mat4s proj, float deltaFar) + CGLM_INLINE mat4s glms_perspective_default_lh_zo(float aspect) + CGLM_INLINE void glms_perspective_resize_lh_zo(mat4s proj, float aspect) + CGLM_INLINE void glms_persp_decomp_lh_zo(mat4s proj, + float *nearv, float *farv, + float *top, float *bottom, + float *left, float *right) + CGLM_INLINE void glms_persp_decompv_lh_zo(mat4s proj, float dest[6]) + CGLM_INLINE void glms_persp_decomp_x_lh_zo(mat4s proj, float *left, float *right) + CGLM_INLINE void glms_persp_decomp_y_lh_zo(mat4s proj, float *top, float *bottom) + CGLM_INLINE void glms_persp_decomp_z_lh_zo(mat4s proj, float *nearv, float *farv) + CGLM_INLINE void glms_persp_decomp_far_lh_zo(mat4s proj, float *farZ) + CGLM_INLINE void glms_persp_decomp_near_lh_zo(mat4s proj, float *nearZ) + CGLM_INLINE float glms_persp_fovy_lh_zo(mat4s proj) + CGLM_INLINE float glms_persp_aspect_lh_zo(mat4s proj) + CGLM_INLINE vec4s glms_persp_sizes_lh_zo(mat4s proj, float fovy) + */ + +#ifndef cglms_persp_lh_zo_h +#define cglms_persp_lh_zo_h + +#include "../../common.h" +#include "../../types-struct.h" +#include "../../plane.h" +#include "../../cam.h" +#include "../../clipspace/persp_lh_zo.h" + +/*! + * @brief set up perspective peprojection matrix + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_frustum_lh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ) { + mat4s dest; + glm_frustum_lh_zo(left, right, bottom, top, nearZ, farZ, dest.raw); + return dest; +} + +/*! + * @brief set up perspective projection matrix + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] fovy field of view angle + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping planes + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_perspective_lh_zo(float fovy, float aspect, float nearZ, float farZ) { + mat4s dest; + glm_perspective_lh_zo(fovy, aspect, nearZ, farZ, dest.raw); + return dest; +} + +/*! + * @brief extend perspective projection matrix's far distance + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * NOTE: if you dodn't want to create new matrix then use array api on struct.raw + * like glms_persp_move_far_lh_zo(prooj.raw, deltaFar) to avoid create new mat4 + * each time + * + * this function does not guarantee far >= near, be aware of that! + * + * @param[in, out] proj projection matrix to extend + * @param[in] deltaFar distance from existing far (negative to shink) + */ +CGLM_INLINE +mat4s +glms_persp_move_far_lh_zo(mat4s proj, float deltaFar) { + mat4s dest; + dest = proj; + glm_persp_move_far_lh_zo(dest.raw, deltaFar); + return dest; +} + +/*! + * @brief set up perspective projection matrix with default near/far + * and angle values with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_perspective_default_lh_zo(float aspect) { + mat4s dest; + glm_perspective_default_lh_zo(aspect, dest.raw); + return dest; +} + +/*! + * @brief resize perspective matrix by aspect ratio ( width / height ) + * this makes very easy to resize proj matrix when window /viewport + * reized with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * NOTE: if you dodn't want to create new matrix then use array api on struct.raw + * like glms_perspective_resize_lh_zo(proj.raw, aspect) to avoid create new mat4 + * each time + * + * @param[in, out] proj perspective projection matrix + * @param[in] aspect aspect ratio ( width / height ) + */ +CGLM_INLINE +mat4s +glms_perspective_resize_lh_zo(mat4s proj, float aspect) { + mat4s dest; + dest = proj; + glm_perspective_resize_lh_zo(aspect, dest.raw); + return dest; +} + +/*! + * @brief decomposes frustum values of perspective projection. + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + * @param[out] top top + * @param[out] bottom bottom + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glms_persp_decomp_lh_zo(mat4s proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right) { + glm_persp_decomp_lh_zo(proj.raw, nearZ, farZ, top, bottom, left, right); +} + +/*! + * @brief decomposes frustum values of perspective projection. + * this makes easy to get all values at once + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] dest array + */ +CGLM_INLINE +void +glms_persp_decompv_lh_zo(mat4s proj, float dest[6]) { + glm_persp_decompv_lh_zo(proj.raw, dest); +} + +/*! + * @brief decomposes left and right values of perspective projection + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * x stands for x axis (left / right axis) + * + * @param[in] proj perspective projection matrix + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glms_persp_decomp_x_lh_zo(mat4s proj, + float * __restrict left, + float * __restrict right) { + glm_persp_decomp_x_lh_zo(proj.raw, left, right); +} + +/*! + * @brief decomposes top and bottom values of perspective projection + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * y stands for y axis (top / bottom axis) + * + * @param[in] proj perspective projection matrix + * @param[out] top top + * @param[out] bottom bottom + */ +CGLM_INLINE +void +glms_persp_decomp_y_lh_zo(mat4s proj, + float * __restrict top, + float * __restrict bottom) { + glm_persp_decomp_y_lh_zo(proj.raw, top, bottom); +} + +/*! + * @brief decomposes near and far values of perspective projection + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * z stands for z axis (near / far axis) + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + */ +CGLM_INLINE +void +glms_persp_decomp_z_lh_zo(mat4s proj, + float * __restrict nearZ, + float * __restrict farZ) { + glm_persp_decomp_z_lh_zo(proj.raw, nearZ, farZ); +} + +/*! + * @brief decomposes far value of perspective projection + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] farZ far + */ +CGLM_INLINE +void +glms_persp_decomp_far_lh_zo(mat4s proj, float * __restrict farZ) { + glm_persp_decomp_far_lh_zo(proj.raw, farZ); +} + +/*! + * @brief decomposes near value of perspective projection + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + */ +CGLM_INLINE +void +glms_persp_decomp_near_lh_zo(mat4s proj, float * __restrict nearZ) { + glm_persp_decomp_near_lh_zo(proj.raw, nearZ); +} + +/*! + * @brief returns field of view angle along the Y-axis (in radians) + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * if you need to degrees, use glm_deg to convert it or use this: + * fovy_deg = glm_deg(glm_persp_fovy(projMatrix)) + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glms_persp_fovy_lh_zo(mat4s proj) { + return glm_persp_fovy_lh_zo(proj.raw); +} + +/*! + * @brief returns aspect ratio of perspective projection + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glms_persp_aspect_lh_zo(mat4s proj) { + return glm_persp_aspect_lh_zo(proj.raw); +} + +/*! + * @brief returns sizes of near and far planes of perspective projection + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[in] fovy fovy (see brief) + * @returns sizes as vector, sizes order: [Wnear, Hnear, Wfar, Hfar] + */ +CGLM_INLINE +vec4s +glms_persp_sizes_lh_zo(mat4s proj, float fovy) { + vec4s dest; + glm_persp_sizes_lh_zo(proj.raw, fovy, dest.raw); + return dest; +} + +#endif /* cglms_persp_lh_zo_h */ diff --git a/cglm/include/cglm/struct/clipspace/persp_rh_no.h b/cglm/include/cglm/struct/clipspace/persp_rh_no.h new file mode 100644 index 0000000..7120fdf --- /dev/null +++ b/cglm/include/cglm/struct/clipspace/persp_rh_no.h @@ -0,0 +1,312 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_frustum_rh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ) + CGLM_INLINE mat4s glms_perspective_rh_no(float fovy, + float aspect, + float nearZ, + float farZ) + CGLM_INLINE void glms_persp_move_far_rh_no(mat4s proj, float deltaFar) + CGLM_INLINE mat4s glms_perspective_default_rh_no(float aspect) + CGLM_INLINE void glms_perspective_resize_rh_no(mat4s proj, float aspect) + CGLM_INLINE void glms_persp_decomp_rh_no(mat4s proj, + float *nearv, float *farv, + float *top, float *bottom, + float *left, float *right) + CGLM_INLINE void glms_persp_decompv_rh_no(mat4s proj, float dest[6]) + CGLM_INLINE void glms_persp_decomp_x_rh_no(mat4s proj, float *left, float *right) + CGLM_INLINE void glms_persp_decomp_y_rh_no(mat4s proj, float *top, float *bottom) + CGLM_INLINE void glms_persp_decomp_z_rh_no(mat4s proj, float *nearv, float *farv) + CGLM_INLINE void glms_persp_decomp_far_rh_no(mat4s proj, float *farZ) + CGLM_INLINE void glms_persp_decomp_near_rh_no(mat4s proj, float *nearZ) + CGLM_INLINE float glms_persp_fovy_rh_no(mat4s proj) + CGLM_INLINE float glms_persp_aspect_rh_no(mat4s proj) + CGLM_INLINE vec4s glms_persp_sizes_rh_no(mat4s proj, float fovy) + */ + +#ifndef cglms_persp_rh_no_h +#define cglms_persp_rh_no_h + +#include "../../common.h" +#include "../../types-struct.h" +#include "../../plane.h" +#include "../../cam.h" +#include "../../clipspace/persp_rh_no.h" + +/*! + * @brief set up perspective peprojection matrix + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_frustum_rh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ) { + mat4s dest; + glm_frustum_rh_no(left, right, bottom, top, nearZ, farZ, dest.raw); + return dest; +} + +/*! + * @brief set up perspective projection matrix + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] fovy field of view angle + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping planes + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_perspective_rh_no(float fovy, float aspect, float nearZ, float farZ) { + mat4s dest; + glm_perspective_rh_no(fovy, aspect, nearZ, farZ, dest.raw); + return dest; +} + +/*! + * @brief extend perspective projection matrix's far distance + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * NOTE: if you dodn't want to create new matrix then use array api on struct.raw + * like glms_persp_move_far_rh_no(prooj.raw, deltaFar) to avoid create new mat4 + * each time + * s + * this function does not guarantee far >= near, be aware of that! + * + * @param[in, out] proj projection matrix to extend + * @param[in] deltaFar distance from existing far (negative to shink) + */ +CGLM_INLINE +mat4s +glms_persp_move_far_rh_no(mat4s proj, float deltaFar) { + mat4s dest; + dest = proj; + glm_persp_move_far_rh_no(dest.raw, deltaFar); + return dest; +} + +/*! + * @brief set up perspective projection matrix with default near/far + * and angle values with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_perspective_default_rh_no(float aspect) { + mat4s dest; + glm_perspective_default_rh_no(aspect, dest.raw); + return dest; +} + +/*! + * @brief resize perspective matrix by aspect ratio ( width / height ) + * this makes very easy to resize proj matrix when window /viewport + * reized with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * NOTE: if you dodn't want to create new matrix then use array api on struct.raw + * like glm_perspective_resize_rh_no(proj.raw, aspect) to avoid create new mat4 + * each time + * + * @param[in, out] proj perspective projection matrix + * @param[in] aspect aspect ratio ( width / height ) + */ +CGLM_INLINE +mat4s +glms_perspective_resize_rh_no(mat4s proj, float aspect) { + mat4s dest; + dest = proj; + glm_perspective_resize_rh_no(aspect, dest.raw); + return dest; +} + +/*! + * @brief decomposes frustum values of perspective projection. + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + * @param[out] top top + * @param[out] bottom bottom + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glms_persp_decomp_rh_no(mat4s proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right) { + glm_persp_decomp_rh_no(proj.raw, nearZ, farZ, top, bottom, left, right); +} + +/*! + * @brief decomposes frustum values of perspective projection. + * this makes easy to get all values at once + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] dest array + */ +CGLM_INLINE +void +glms_persp_decompv_rh_no(mat4s proj, float dest[6]) { + glm_persp_decompv_rh_no(proj.raw, dest); +} + +/*! + * @brief decomposes left and right values of perspective projection + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * x stands for x axis (left / right axis) + * + * @param[in] proj perspective projection matrix + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glms_persp_decomp_x_rh_no(mat4s proj, + float * __restrict left, + float * __restrict right) { + glm_persp_decomp_x_rh_no(proj.raw, left, right); +} + +/*! + * @brief decomposes top and bottom values of perspective projection + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * y stands for y axis (top / bottom axis) + * + * @param[in] proj perspective projection matrix + * @param[out] top top + * @param[out] bottom bottom + */ +CGLM_INLINE +void +glms_persp_decomp_y_rh_no(mat4s proj, + float * __restrict top, + float * __restrict bottom) { + glm_persp_decomp_y_rh_no(proj.raw, top, bottom); +} + +/*! + * @brief decomposes near and far values of perspective projection + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * z stands for z axis (near / far axis) + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + */ +CGLM_INLINE +void +glms_persp_decomp_z_rh_no(mat4s proj, + float * __restrict nearZ, + float * __restrict farZ) { + glm_persp_decomp_z_rh_no(proj.raw, nearZ, farZ); +} + +/*! + * @brief decomposes far value of perspective projection + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] farZ far + */ +CGLM_INLINE +void +glms_persp_decomp_far_rh_no(mat4s proj, float * __restrict farZ) { + glm_persp_decomp_far_rh_no(proj.raw, farZ); +} + +/*! + * @brief decomposes near value of perspective projection + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + */ +CGLM_INLINE +void +glms_persp_decomp_near_rh_no(mat4s proj, float * __restrict nearZ) { + glm_persp_decomp_near_rh_no(proj.raw, nearZ); +} + +/*! + * @brief returns field of view angle along the Y-axis (in radians) + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * if you need to degrees, use glm_deg to convert it or use this: + * fovy_deg = glm_deg(glm_persp_fovy(projMatrix)) + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glms_persp_fovy_rh_no(mat4s proj) { + return glm_persp_fovy_rh_no(proj.raw); +} + +/*! + * @brief returns aspect ratio of perspective projection + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glms_persp_aspect_rh_no(mat4s proj) { + return glm_persp_aspect_rh_no(proj.raw); +} + +/*! + * @brief returns sizes of near and far planes of perspective projection + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * @param[in] proj perspective projection matrix + * @param[in] fovy fovy (see brief) + * @returns sizes as vector, sizes order: [Wnear, Hnear, Wfar, Hfar] + */ +CGLM_INLINE +vec4s +glms_persp_sizes_rh_no(mat4s proj, float fovy) { + vec4s dest; + glm_persp_sizes_rh_no(proj.raw, fovy, dest.raw); + return dest; +} + +#endif /* cglms_persp_rh_no_h */ diff --git a/cglm/include/cglm/struct/clipspace/persp_rh_zo.h b/cglm/include/cglm/struct/clipspace/persp_rh_zo.h new file mode 100644 index 0000000..e3585a2 --- /dev/null +++ b/cglm/include/cglm/struct/clipspace/persp_rh_zo.h @@ -0,0 +1,312 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_frustum_rh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ) + CGLM_INLINE mat4s glms_perspective_rh_zo(float fovy, + float aspect, + float nearZ, + float farZ) + CGLM_INLINE void glms_persp_move_far_rh_zo(mat4s proj, float deltaFar) + CGLM_INLINE mat4s glms_perspective_default_rh_zo(float aspect) + CGLM_INLINE void glms_perspective_resize_rh_zo(mat4s proj, float aspect) + CGLM_INLINE void glms_persp_decomp_rh_zo(mat4s proj, + float *nearv, float *farv, + float *top, float *bottom, + float *left, float *right) + CGLM_INLINE void glms_persp_decompv_rh_zo(mat4s proj, float dest[6]) + CGLM_INLINE void glms_persp_decomp_x_rh_zo(mat4s proj, float *left, float *right) + CGLM_INLINE void glms_persp_decomp_y_rh_zo(mat4s proj, float *top, float *bottom) + CGLM_INLINE void glms_persp_decomp_z_rh_zo(mat4s proj, float *nearv, float *farv) + CGLM_INLINE void glms_persp_decomp_far_rh_zo(mat4s proj, float *farZ) + CGLM_INLINE void glms_persp_decomp_near_rh_zo(mat4s proj, float *nearZ) + CGLM_INLINE float glms_persp_fovy_rh_zo(mat4s proj) + CGLM_INLINE float glms_persp_aspect_rh_zo(mat4s proj) + CGLM_INLINE vec4s glms_persp_sizes_rh_zo(mat4s proj, float fovy) + */ + +#ifndef cglms_persp_rh_zo_h +#define cglms_persp_rh_zo_h + +#include "../../common.h" +#include "../../types-struct.h" +#include "../../plane.h" +#include "../../cam.h" +#include "../../clipspace/persp_rh_zo.h" + +/*! + * @brief set up perspective peprojection matrix + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] left viewport.left + * @param[in] right viewport.right + * @param[in] bottom viewport.bottom + * @param[in] top viewport.top + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping plane + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_frustum_rh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ) { + mat4s dest; + glm_frustum_rh_zo(left, right, bottom, top, nearZ, farZ, dest.raw); + return dest; +} + +/*! + * @brief set up perspective projection matrix + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] fovy field of view angle + * @param[in] aspect aspect ratio ( width / height ) + * @param[in] nearZ near clipping plane + * @param[in] farZ far clipping planes + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_perspective_rh_zo(float fovy, float aspect, float nearZ, float farZ) { + mat4s dest; + glm_perspective_rh_zo(fovy, aspect, nearZ, farZ, dest.raw); + return dest; +} + +/*! + * @brief extend perspective projection matrix's far distance + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * NOTE: if you dodn't want to create new matrix then use array api on struct.raw + * like glms_persp_move_far_rh_zo(prooj.raw, deltaFar) to avoid create new mat4 + * each time + * + * this function does not guarantee far >= near, be aware of that! + * + * @param[in, out] proj projection matrix to extend + * @param[in] deltaFar distance from existing far (negative to shink) + */ +CGLM_INLINE +mat4s +glms_persp_move_far_rh_zo(mat4s proj, float deltaFar) { + mat4s dest; + dest = proj; + glm_persp_move_far_rh_zo(dest.raw, deltaFar); + return dest; +} + +/*! + * @brief set up perspective projection matrix with default near/far + * and angle values with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] aspect aspect ratio ( width / height ) + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_perspective_default_rh_zo(float aspect) { + mat4s dest; + glm_perspective_default_rh_zo(aspect, dest.raw); + return dest; +} + +/*! + * @brief resize perspective matrix by aspect ratio ( width / height ) + * this makes very easy to resize proj matrix when window /viewport + * reized with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * NOTE: if you dodn't want to create new matrix then use array api on struct.raw + * like glm_perspective_resize_rh_zo(proj.raw, aspect) to avoid create new mat4 + * each time + * + * @param[in, out] proj perspective projection matrix + * @param[in] aspect aspect ratio ( width / height ) + */ +CGLM_INLINE +mat4s +glms_perspective_resize_rh_zo(mat4s proj, float aspect) { + mat4s dest; + dest = proj; + glm_perspective_resize_rh_zo(aspect, dest.raw); + return dest; +} + +/*! + * @brief decomposes frustum values of perspective projection. + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + * @param[out] top top + * @param[out] bottom bottom + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glms_persp_decomp_rh_zo(mat4s proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right) { + glm_persp_decomp_rh_zo(proj.raw, nearZ, farZ, top, bottom, left, right); +} + +/*! + * @brief decomposes frustum values of perspective projection. + * this makes easy to get all values at once + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] dest array + */ +CGLM_INLINE +void +glms_persp_decompv_rh_zo(mat4s proj, float dest[6]) { + glm_persp_decompv_rh_zo(proj.raw, dest); +} + +/*! + * @brief decomposes left and right values of perspective projection + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * x stands for x axis (left / right axis) + * + * @param[in] proj perspective projection matrix + * @param[out] left left + * @param[out] right right + */ +CGLM_INLINE +void +glms_persp_decomp_x_rh_zo(mat4s proj, + float * __restrict left, + float * __restrict right) { + glm_persp_decomp_x_rh_zo(proj.raw, left, right); +} + +/*! + * @brief decomposes top and bottom values of perspective projection + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * y stands for y axis (top / bottom axis) + * + * @param[in] proj perspective projection matrix + * @param[out] top top + * @param[out] bottom bottom + */ +CGLM_INLINE +void +glms_persp_decomp_y_rh_zo(mat4s proj, + float * __restrict top, + float * __restrict bottom) { + glm_persp_decomp_y_rh_zo(proj.raw, top, bottom); +} + +/*! + * @brief decomposes near and far values of perspective projection + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * z stands for z axis (near / far axis) + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + * @param[out] farZ far + */ +CGLM_INLINE +void +glms_persp_decomp_z_rh_zo(mat4s proj, + float * __restrict nearZ, + float * __restrict farZ) { + glm_persp_decomp_z_rh_zo(proj.raw, nearZ, farZ); +} + +/*! + * @brief decomposes far value of perspective projection + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] farZ far + */ +CGLM_INLINE +void +glms_persp_decomp_far_rh_zo(mat4s proj, float * __restrict farZ) { + glm_persp_decomp_far_rh_zo(proj.raw, farZ); +} + +/*! + * @brief decomposes near value of perspective projection + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[out] nearZ near + */ +CGLM_INLINE +void +glms_persp_decomp_near_rh_zo(mat4s proj, float * __restrict nearZ) { + glm_persp_decomp_near_rh_zo(proj.raw, nearZ); +} + +/*! + * @brief returns field of view angle along the Y-axis (in radians) + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * if you need to degrees, use glm_deg to convert it or use this: + * fovy_deg = glm_deg(glm_persp_fovy(projMatrix)) + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glms_persp_fovy_rh_zo(mat4s proj) { + return glm_persp_fovy_rh_zo(proj.raw); +} + +/*! + * @brief returns aspect ratio of perspective projection + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + */ +CGLM_INLINE +float +glms_persp_aspect_rh_zo(mat4s proj) { + return glm_persp_aspect_rh_zo(proj.raw); +} + +/*! + * @brief returns sizes of near and far planes of perspective projection + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * @param[in] proj perspective projection matrix + * @param[in] fovy fovy (see brief) + * @returns sizes as vector, sizes order: [Wnear, Hnear, Wfar, Hfar] + */ +CGLM_INLINE +vec4s +glms_persp_sizes_rh_zo(mat4s proj, float fovy) { + vec4s dest; + glm_persp_sizes_rh_zo(proj.raw, fovy, dest.raw); + return dest; +} + +#endif /* cglms_persp_rh_zo_h */ diff --git a/cglm/include/cglm/struct/clipspace/project_no.h b/cglm/include/cglm/struct/clipspace/project_no.h new file mode 100644 index 0000000..1a28d47 --- /dev/null +++ b/cglm/include/cglm/struct/clipspace/project_no.h @@ -0,0 +1,98 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE vec3s glms_unprojecti_no(vec3s pos, mat4s invMat, vec4s vp) + CGLM_INLINE vec3s glms_project_no(vec3s pos, mat4s m, vec4s vp) + CGLM_INLINE float glms_project_z_no(vec3s v, mat4s m) + */ + +#ifndef cglms_project_no_h +#define cglms_project_no_h + +#include "../../common.h" +#include "../../types-struct.h" +#include "../../plane.h" +#include "../../cam.h" +#include "../../clipspace/project_no.h" + +/*! + * @brief maps the specified viewport coordinates into specified space [1] + * the matrix should contain projection matrix. + * + * if you don't have ( and don't want to have ) an inverse matrix then use + * glm_unproject version. You may use existing inverse of matrix in somewhere + * else, this is why glm_unprojecti exists to save save inversion cost + * + * [1] space: + * 1- if m = invProj: View Space + * 2- if m = invViewProj: World Space + * 3- if m = invMVP: Object Space + * + * You probably want to map the coordinates into object space + * so use invMVP as m + * + * Computing viewProj: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * glm_mat4_inv(viewProj, invMVP); + * + * @param[in] pos point/position in viewport coordinates + * @param[in] invMat matrix (see brief) + * @param[in] vp viewport as [x, y, width, height] + * + * @returns unprojected coordinates + */ +CGLM_INLINE +vec3s +glms_unprojecti_no(vec3s pos, mat4s invMat, vec4s vp) { + vec3s dest; + glm_unprojecti_no(pos.raw, invMat.raw, vp.raw, dest.raw); + return dest; +} + +/*! + * @brief map object coordinates to window coordinates + * + * Computing MVP: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * + * @param[in] pos object coordinates + * @param[in] m MVP matrix + * @param[in] vp viewport as [x, y, width, height] + * + * @returns projected coordinates + */ +CGLM_INLINE +vec3s +glms_project_no(vec3s pos, mat4s m, vec4s vp) { + vec3s dest; + glm_project_no(pos.raw, m.raw, vp.raw, dest.raw); + return dest; +} + +/*! + * @brief map object's z coordinate to window coordinates + * + * Computing MVP: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * + * @param[in] v object coordinates + * @param[in] m MVP matrix + * + * @returns projected z coordinate + */ +CGLM_INLINE +float +glms_project_z_no(vec3s v, mat4s m) { + return glm_project_z_no(v.raw, m.raw); +} + +#endif /* cglms_project_rh_no_h */ diff --git a/cglm/include/cglm/struct/clipspace/project_zo.h b/cglm/include/cglm/struct/clipspace/project_zo.h new file mode 100644 index 0000000..13065f1 --- /dev/null +++ b/cglm/include/cglm/struct/clipspace/project_zo.h @@ -0,0 +1,98 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE vec3s glms_unprojecti_no(vec3s pos, mat4s invMat, vec4s vp) + CGLM_INLINE vec3s glms_project_no(vec3s pos, mat4s m, vec4s vp) + CGLM_INLINE float glms_project_z_zo(vec3s v, mat4s m) + */ + +#ifndef cglms_project_zo_h +#define cglms_project_zo_h + +#include "../../common.h" +#include "../../types-struct.h" +#include "../../plane.h" +#include "../../cam.h" +#include "../../clipspace/project_zo.h" + +/*! + * @brief maps the specified viewport coordinates into specified space [1] + * the matrix should contain projection matrix. + * + * if you don't have ( and don't want to have ) an inverse matrix then use + * glm_unproject version. You may use existing inverse of matrix in somewhere + * else, this is why glm_unprojecti exists to save save inversion cost + * + * [1] space: + * 1- if m = invProj: View Space + * 2- if m = invViewProj: World Space + * 3- if m = invMVP: Object Space + * + * You probably want to map the coordinates into object space + * so use invMVP as m + * + * Computing viewProj: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * glm_mat4_inv(viewProj, invMVP); + * + * @param[in] pos point/position in viewport coordinates + * @param[in] invMat matrix (see brief) + * @param[in] vp viewport as [x, y, width, height] + * + * @returns unprojected coordinates + */ +CGLM_INLINE +vec3s +glms_unprojecti_zo(vec3s pos, mat4s invMat, vec4s vp) { + vec3s dest; + glm_unprojecti_zo(pos.raw, invMat.raw, vp.raw, dest.raw); + return dest; +} + +/*! + * @brief map object coordinates to window coordinates + * + * Computing MVP: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * + * @param[in] pos object coordinates + * @param[in] m MVP matrix + * @param[in] vp viewport as [x, y, width, height] + * + * @returns projected coordinates + */ +CGLM_INLINE +vec3s +glms_project_zo(vec3s pos, mat4s m, vec4s vp) { + vec3s dest; + glm_project_zo(pos.raw, m.raw, vp.raw, dest.raw); + return dest; +} + +/*! + * @brief map object's z coordinate to window coordinates + * + * Computing MVP: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * + * @param[in] v object coordinates + * @param[in] m MVP matrix + * + * @returns projected z coordinate + */ +CGLM_INLINE +float +glms_project_z_zo(vec3s v, mat4s m) { + return glm_project_z_zo(v.raw, m.raw); +} + +#endif /* cglm_project_zo_h */ diff --git a/cglm/include/cglm/struct/clipspace/view_lh_no.h b/cglm/include/cglm/struct/clipspace/view_lh_no.h new file mode 100644 index 0000000..e4ca5ba --- /dev/null +++ b/cglm/include/cglm/struct/clipspace/view_lh_no.h @@ -0,0 +1,89 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_lookat_lh_no(vec3s eye, vec3s center, vec3s up) + CGLM_INLINE mat4s glms_look_lh_no(vec3s eye, vec3s dir, vec3s up) + CGLM_INLINE mat4s glms_look_anyup_lh_no(vec3s eye, vec3s dir) + */ + +#ifndef cglms_view_lh_no_h +#define cglms_view_lh_no_h + +#include "../../common.h" +#include "../../types-struct.h" +#include "../../plane.h" +#include "../../cam.h" +#include "../../clipspace/view_lh_no.h" + +/*! + * @brief set up view matrix + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] center center vector + * @param[in] up up vector + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_lookat_lh_no(vec3s eye, vec3s center, vec3s up) { + mat4s dest; + glm_lookat_lh_no(eye.raw, center.raw, up.raw, dest.raw); + return dest; +} + +/*! + * @brief set up view matrix + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * convenient wrapper for lookat: if you only have direction not target self + * then this might be useful. Because you need to get target from direction. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[in] up up vector + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_look_lh_no(vec3s eye, vec3s dir, vec3s up) { + mat4s dest; + glm_look_lh_no(eye.raw, dir.raw, up.raw, dest.raw); + return dest; +} + +/*! + * @brief set up view matrix + * with a left-hand coordinate system and a + * clip-space of [-1, 1]. + * + * convenient wrapper for look: if you only have direction and if you don't + * care what UP vector is then this might be useful to create view matrix + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_look_anyup_lh_no(vec3s eye, vec3s dir) { + mat4s dest; + glm_look_anyup_lh_no(eye.raw, dir.raw, dest.raw); + return dest; +} + +#endif /* cglms_view_lh_no_h */ diff --git a/cglm/include/cglm/struct/clipspace/view_lh_zo.h b/cglm/include/cglm/struct/clipspace/view_lh_zo.h new file mode 100644 index 0000000..ac1ada9 --- /dev/null +++ b/cglm/include/cglm/struct/clipspace/view_lh_zo.h @@ -0,0 +1,89 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_lookat_lh_zo(vec3s eye, vec3s center, vec3s up) + CGLM_INLINE mat4s glms_look_lh_zo(vec3s eye, vec3s dir, vec3s up) + CGLM_INLINE mat4s glms_look_anyup_lh_zo(vec3s eye, vec3s dir) + */ + +#ifndef cglms_view_lh_zo_h +#define cglms_view_lh_zo_h + +#include "../../common.h" +#include "../../types-struct.h" +#include "../../plane.h" +#include "../../cam.h" +#include "../../clipspace/view_lh_zo.h" + +/*! + * @brief set up view matrix + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] center center vector + * @param[in] up up vector + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_lookat_lh_zo(vec3s eye, vec3s center, vec3s up) { + mat4s dest; + glm_lookat_lh_zo(eye.raw, center.raw, up.raw, dest.raw); + return dest; +} + +/*! + * @brief set up view matrix + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * convenient wrapper for lookat: if you only have direction not target self + * then this might be useful. Because you need to get target from direction. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[in] up up vector + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_look_lh_zo(vec3s eye, vec3s dir, vec3s up) { + mat4s dest; + glm_look_lh_zo(eye.raw, dir.raw, up.raw, dest.raw); + return dest; +} + +/*! + * @brief set up view matrix + * with a left-hand coordinate system and a + * clip-space of [0, 1]. + * + * convenient wrapper for look: if you only have direction and if you don't + * care what UP vector is then this might be useful to create view matrix + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_look_anyup_lh_zo(vec3s eye, vec3s dir) { + mat4s dest; + glm_look_anyup_lh_zo(eye.raw, dir.raw, dest.raw); + return dest; +} + +#endif /* cglms_view_lh_zo_h */ diff --git a/cglm/include/cglm/struct/clipspace/view_rh_no.h b/cglm/include/cglm/struct/clipspace/view_rh_no.h new file mode 100644 index 0000000..99b03c3 --- /dev/null +++ b/cglm/include/cglm/struct/clipspace/view_rh_no.h @@ -0,0 +1,89 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_lookat_rh_no(vec3s eye, vec3s center, vec3s up) + CGLM_INLINE mat4s glms_look_rh_no(vec3s eye, vec3s dir, vec3s up) + CGLM_INLINE mat4s glms_look_anyup_rh_no(vec3s eye, vec3s dir) + */ + +#ifndef cglms_view_rh_no_h +#define cglms_view_rh_no_h + +#include "../../common.h" +#include "../../types-struct.h" +#include "../../plane.h" +#include "../../cam.h" +#include "../../clipspace/view_rh_no.h" + +/*! + * @brief set up view matrix + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] center center vector + * @param[in] up up vector + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_lookat_rh_no(vec3s eye, vec3s center, vec3s up) { + mat4s dest; + glm_lookat_rh_no(eye.raw, center.raw, up.raw, dest.raw); + return dest; +} + +/*! + * @brief set up view matrix + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * convenient wrapper for lookat: if you only have direction not target self + * then this might be useful. Because you need to get target from direction. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[in] up up vector + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_look_rh_no(vec3s eye, vec3s dir, vec3s up) { + mat4s dest; + glm_look_rh_no(eye.raw, dir.raw, up.raw, dest.raw); + return dest; +} + +/*! + * @brief set up view matrix + * with a right-hand coordinate system and a + * clip-space of [-1, 1]. + * + * convenient wrapper for look: if you only have direction and if you don't + * care what UP vector is then this might be useful to create view matrix + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_look_anyup_rh_no(vec3s eye, vec3s dir) { + mat4s dest; + glm_look_anyup_rh_no(eye.raw, dir.raw, dest.raw); + return dest; +} + +#endif /* cglms_view_rh_no_h */ diff --git a/cglm/include/cglm/struct/clipspace/view_rh_zo.h b/cglm/include/cglm/struct/clipspace/view_rh_zo.h new file mode 100644 index 0000000..14ffe32 --- /dev/null +++ b/cglm/include/cglm/struct/clipspace/view_rh_zo.h @@ -0,0 +1,89 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE mat4s glms_lookat_rh_zo(vec3s eye, vec3s center, vec3s up) + CGLM_INLINE mat4s glms_look_rh_zo(vec3s eye, vec3s dir, vec3s up) + CGLM_INLINE mat4s glms_look_anyup_rh_zo(vec3s eye, vec3s dir) + */ + +#ifndef cglms_view_rh_zo_h +#define cglms_view_rh_zo_h + +#include "../../common.h" +#include "../../types-struct.h" +#include "../../plane.h" +#include "../../cam.h" +#include "../../clipspace/view_rh_zo.h" + +/*! + * @brief set up view matrix + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] center center vector + * @param[in] up up vector + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_lookat_rh_zo(vec3s eye, vec3s center, vec3s up) { + mat4s dest; + glm_lookat_rh_zo(eye.raw, center.raw, up.raw, dest.raw); + return dest; +} + +/*! + * @brief set up view matrix + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * convenient wrapper for lookat: if you only have direction not target self + * then this might be useful. Because you need to get target from direction. + * + * NOTE: The UP vector must not be parallel to the line of sight from + * the eye point to the reference point + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @param[in] up up vector + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_look_rh_zo(vec3s eye, vec3s dir, vec3s up) { + mat4s dest; + glm_look_rh_zo(eye.raw, dir.raw, up.raw, dest.raw); + return dest; +} + +/*! + * @brief set up view matrix + * with a right-hand coordinate system and a + * clip-space of [0, 1]. + * + * convenient wrapper for look: if you only have direction and if you don't + * care what UP vector is then this might be useful to create view matrix + * + * @param[in] eye eye vector + * @param[in] dir direction vector + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_look_anyup_rh_zo(vec3s eye, vec3s dir) { + mat4s dest; + glm_look_anyup_rh_zo(eye.raw, dir.raw, dest.raw); + return dest; +} + +#endif /* cglms_view_rh_zo_h */ diff --git a/cglm/include/cglm/struct/color.h b/cglm/include/cglm/struct/color.h new file mode 100644 index 0000000..3ce78da --- /dev/null +++ b/cglm/include/cglm/struct/color.h @@ -0,0 +1,27 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglms_colors_h +#define cglms_colors_h + +#include "../common.h" +#include "../types-struct.h" +#include "../color.h" +#include "vec3.h" + +/*! + * @brief averages the color channels into one value + * + * @param[in] rgb RGB color + */ +CGLM_INLINE +float +glms_luminance(vec3s rgb) { + return glm_luminance(rgb.raw); +} + +#endif /* cglms_colors_h */ diff --git a/cglm/include/cglm/struct/curve.h b/cglm/include/cglm/struct/curve.h new file mode 100644 index 0000000..53ea359 --- /dev/null +++ b/cglm/include/cglm/struct/curve.h @@ -0,0 +1,40 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglms_curves_h +#define cglms_curves_h + +#include "../common.h" +#include "../types-struct.h" +#include "../curve.h" +#include "vec4.h" +#include "mat4.h" + +/*! + * @brief helper function to calculate S*M*C multiplication for curves + * + * This function does not encourage you to use SMC, + * instead it is a helper if you use SMC. + * + * if you want to specify S as vector then use more generic glm_mat4_rmc() func. + * + * Example usage: + * B(s) = glm_smc(s, GLM_BEZIER_MAT, (vec4){p0, c0, c1, p1}) + * + * @param[in] s parameter between 0 and 1 (this will be [s3, s2, s, 1]) + * @param[in] m basis matrix + * @param[in] c position/control vector + * + * @return B(s) + */ +CGLM_INLINE +float +glms_smc(float s, mat4s m, vec4s c) { + return glm_smc(s, m.raw, c.raw); +} + +#endif /* cglms_curves_h */ diff --git a/cglm/include/cglm/struct/euler.h b/cglm/include/cglm/struct/euler.h new file mode 100644 index 0000000..19697f7 --- /dev/null +++ b/cglm/include/cglm/struct/euler.h @@ -0,0 +1,249 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + NOTE: + angles must be passed as [X-Angle, Y-Angle, Z-angle] order + For instance you don't pass angles as [Z-Angle, X-Angle, Y-angle] to + glm_euler_zxy function, All RELATED functions accept angles same order + which is [X, Y, Z]. + */ + +/* + Types: + enum glm_euler_seq + + Functions: + CGLM_INLINE vec3s glms_euler_angles(mat4s m) + CGLM_INLINE mat4s glms_euler_xyz(vec3s angles) + CGLM_INLINE mat4s glms_euler_xzy(vec3s angles) + CGLM_INLINE mat4s glms_euler_yxz(vec3s angles) + CGLM_INLINE mat4s glms_euler_yzx(vec3s angles) + CGLM_INLINE mat4s glms_euler_zxy(vec3s angles) + CGLM_INLINE mat4s glms_euler_zyx(vec3s angles) + CGLM_INLINE mat4s glms_euler_by_order(vec3s angles, glm_euler_seq ord) + CGLM_INLINE versors glms_euler_xyz_quat(vec3s angles) + CGLM_INLINE versors glms_euler_xzy_quat(vec3s angles) + CGLM_INLINE versors glms_euler_yxz_quat(vec3s angles) + CGLM_INLINE versors glms_euler_yzx_quat(vec3s angles) + CGLM_INLINE versors glms_euler_zxy_quat(vec3s angles) + CGLM_INLINE versors glms_euler_zyx_quat(vec3s angles) + */ + +#ifndef cglms_euler_h +#define cglms_euler_h + +#include "../common.h" +#include "../types-struct.h" +#include "../euler.h" + +/*! + * @brief extract euler angles (in radians) using xyz order + * + * @param[in] m affine transform + * @returns angles vector [x, y, z] + */ +CGLM_INLINE +vec3s +glms_euler_angles(mat4s m) { + vec3s dest; + glm_euler_angles(m.raw, dest.raw); + return dest; +} + +/*! + * @brief build rotation matrix from euler angles + * + * @param[in] angles angles as vector [Xangle, Yangle, Zangle] + * @returns rotation matrix + */ +CGLM_INLINE +mat4s +glms_euler_xyz(vec3s angles) { + mat4s dest; + glm_euler_xyz(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief build rotation matrix from euler angles + * + * @param[in] angles angles as vector [Xangle, Yangle, Zangle] + * @returns rotation matrix + */ +CGLM_INLINE +mat4s +glms_euler_xzy(vec3s angles) { + mat4s dest; + glm_euler_xzy(angles.raw, dest.raw); + return dest; +} + + +/*! + * @brief build rotation matrix from euler angles + * + * @param[in] angles angles as vector [Xangle, Yangle, Zangle] + * @returns rotation matrix + */ +CGLM_INLINE +mat4s +glms_euler_yxz(vec3s angles) { + mat4s dest; + glm_euler_yxz(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief build rotation matrix from euler angles + * + * @param[in] angles angles as vector [Xangle, Yangle, Zangle] + * @returns rotation matrix + */ +CGLM_INLINE +mat4s +glms_euler_yzx(vec3s angles) { + mat4s dest; + glm_euler_yzx(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief build rotation matrix from euler angles + * + * @param[in] angles angles as vector [Xangle, Yangle, Zangle] + * @returns rotation matrix + */ +CGLM_INLINE +mat4s +glms_euler_zxy(vec3s angles) { + mat4s dest; + glm_euler_zxy(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief build rotation matrix from euler angles + * + * @param[in] angles angles as vector [Xangle, Yangle, Zangle] + * @returns rotation matrix + */ +CGLM_INLINE +mat4s +glms_euler_zyx(vec3s angles) { + mat4s dest; + glm_euler_zyx(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief build rotation matrix from euler angles + * + * @param[in] angles angles as vector [Xangle, Yangle, Zangle] + * @param[in] ord euler order + * @returns rotation matrix + */ +CGLM_INLINE +mat4s +glms_euler_by_order(vec3s angles, glm_euler_seq ord) { + mat4s dest; + glm_euler_by_order(angles.raw, ord, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in x y z order (roll pitch yaw) + * + * @param[in] angles angles x y z (radians) + * @returns quaternion + */ +CGLM_INLINE +versors +glms_euler_xyz_quat(vec3s angles) { + versors dest; + glm_euler_xyz_quat(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in x z y order (roll yaw pitch) + * + * @param[in] angles angles x y z (radians) + * @returns quaternion + */ +CGLM_INLINE +versors +glms_euler_xzy_quat(vec3s angles) { + versors dest; + glm_euler_xzy_quat(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in y x z order (pitch roll yaw) + * + * @param[in] angles angles x y z (radians) + * @returns quaternion + */ +CGLM_INLINE +versors +glms_euler_yxz_quat(vec3s angles) { + versors dest; + glm_euler_yxz_quat(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in y z x order (pitch yaw roll) + * + * @param[in] angles angles x y z (radians) + * @returns quaternion + */ +CGLM_INLINE +versors +glms_euler_yzx_quat(vec3s angles) { + versors dest; + glm_euler_yzx_quat(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in z x y order (yaw roll pitch) + * + * @param[in] angles angles x y z (radians) + * @returns quaternion + */ +CGLM_INLINE +versors +glms_euler_zxy_quat(vec3s angles) { + versors dest; + glm_euler_zxy_quat(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in z y x order (yaw pitch roll) + * + * @param[in] angles angles x y z (radians) + * @returns quaternion + */ +CGLM_INLINE +versors +glms_euler_zyx_quat(vec3s angles) { + versors dest; + glm_euler_zyx_quat(angles.raw, dest.raw); + return dest; +} + + +#endif /* cglms_euler_h */ diff --git a/cglm/include/cglm/struct/frustum.h b/cglm/include/cglm/struct/frustum.h new file mode 100644 index 0000000..81b5b7b --- /dev/null +++ b/cglm/include/cglm/struct/frustum.h @@ -0,0 +1,155 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglms_frustums_h +#define cglms_frustums_h + +#include "../common.h" +#include "../types-struct.h" +#include "../frustum.h" +#include "plane.h" +#include "vec3.h" +#include "vec4.h" +#include "mat4.h" + +/* you can override clip space coords + but you have to provide all with same name + e.g.: define GLM_CSCOORD_LBN {0.0f, 0.0f, 1.0f, 1.0f} */ +#ifndef GLM_CUSTOM_CLIPSPACE + +/* near */ +#define GLMS_CSCOORD_LBN {-1.0f, -1.0f, -1.0f, 1.0f} +#define GLMS_CSCOORD_LTN {-1.0f, 1.0f, -1.0f, 1.0f} +#define GLMS_CSCOORD_RTN { 1.0f, 1.0f, -1.0f, 1.0f} +#define GLMS_CSCOORD_RBN { 1.0f, -1.0f, -1.0f, 1.0f} + +/* far */ +#define GLMS_CSCOORD_LBF {-1.0f, -1.0f, 1.0f, 1.0f} +#define GLMS_CSCOORD_LTF {-1.0f, 1.0f, 1.0f, 1.0f} +#define GLMS_CSCOORD_RTF { 1.0f, 1.0f, 1.0f, 1.0f} +#define GLMS_CSCOORD_RBF { 1.0f, -1.0f, 1.0f, 1.0f} + +#endif + +/*! + * @brief extracts view frustum planes + * + * planes' space: + * 1- if m = proj: View Space + * 2- if m = viewProj: World Space + * 3- if m = MVP: Object Space + * + * You probably want to extract planes in world space so use viewProj as m + * Computing viewProj: + * glm_mat4_mul(proj, view, viewProj); + * + * Exracted planes order: [left, right, bottom, top, near, far] + * + * @param[in] m matrix (see brief) + * @param[out] dest extracted view frustum planes (see brief) + */ +CGLM_INLINE +void +glms_frustum_planes(mat4s m, vec4s dest[6]) { + vec4 rawDest[6]; + glm_frustum_planes(m.raw, rawDest); + glms_vec4_(pack)(dest, rawDest, 6); +} + +/*! + * @brief extracts view frustum corners using clip-space coordinates + * + * corners' space: + * 1- if m = invViewProj: World Space + * 2- if m = invMVP: Object Space + * + * You probably want to extract corners in world space so use invViewProj + * Computing invViewProj: + * glm_mat4_mul(proj, view, viewProj); + * ... + * glm_mat4_inv(viewProj, invViewProj); + * + * if you have a near coord at i index, you can get it's far coord by i + 4 + * + * Find center coordinates: + * for (j = 0; j < 4; j++) { + * glm_vec3_center(corners[i], corners[i + 4], centerCorners[i]); + * } + * + * @param[in] invMat matrix (see brief) + * @param[out] dest exracted view frustum corners (see brief) + */ +CGLM_INLINE +void +glms_frustum_corners(mat4s invMat, vec4s dest[8]) { + vec4 rawDest[8]; + glm_frustum_corners(invMat.raw, rawDest); + glms_vec4_(pack)(dest, rawDest, 8); +} + +/*! + * @brief finds center of view frustum + * + * @param[in] corners view frustum corners + * @returns view frustum center + */ +CGLM_INLINE +vec4s +glms_frustum_center(vec4s corners[8]) { + vec4 rawCorners[8]; + vec4s r; + + glms_vec4_(unpack)(rawCorners, corners, 8); + glm_frustum_center(rawCorners, r.raw); + return r; +} + +/*! + * @brief finds bounding box of frustum relative to given matrix e.g. view mat + * + * @param[in] corners view frustum corners + * @param[in] m matrix to convert existing conners + * @param[out] box bounding box as array [min, max] + */ +CGLM_INLINE +void +glms_frustum_box(vec4s corners[8], mat4s m, vec3s box[2]) { + vec4 rawCorners[8]; + vec3 rawBox[2]; + + glms_vec4_(unpack)(rawCorners, corners, 8); + glm_frustum_box(rawCorners, m.raw, rawBox); + glms_vec3_(pack)(box, rawBox, 2); +} + +/*! + * @brief finds planes corners which is between near and far planes (parallel) + * + * this will be helpful if you want to split a frustum e.g. CSM/PSSM. This will + * find planes' corners but you will need to one more plane. + * Actually you have it, it is near, far or created previously with this func ;) + * + * @param[in] corners view frustum corners + * @param[in] splitDist split distance + * @param[in] farDist far distance (zFar) + * @param[out] planeCorners plane corners [LB, LT, RT, RB] + */ +CGLM_INLINE +void +glms_frustum_corners_at(vec4s corners[8], + float splitDist, + float farDist, + vec4s planeCorners[4]) { + vec4 rawCorners[8]; + vec4 rawPlaneCorners[4]; + + glms_vec4_(unpack)(rawCorners, corners, 8); + glm_frustum_corners_at(rawCorners, splitDist, farDist, rawPlaneCorners); + glms_vec4_(pack)(planeCorners, rawPlaneCorners, 8); +} + +#endif /* cglms_frustums_h */ diff --git a/cglm/include/cglm/struct/handed/euler_to_quat_lh.h b/cglm/include/cglm/struct/handed/euler_to_quat_lh.h new file mode 100644 index 0000000..3964e51 --- /dev/null +++ b/cglm/include/cglm/struct/handed/euler_to_quat_lh.h @@ -0,0 +1,115 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glms_euler_xyz_quat_lh(vec3 angles, versor dest); + CGLM_INLINE void glms_euler_xzy_quat_lh(vec3 angles, versor dest); + CGLM_INLINE void glms_euler_yxz_quat_lh(vec3 angles, versor dest); + CGLM_INLINE void glms_euler_yzx_quat_lh(vec3 angles, versor dest); + CGLM_INLINE void glms_euler_zxy_quat_lh(vec3 angles, versor dest); + CGLM_INLINE void glms_euler_zyx_quat_lh(vec3 angles, versor dest); + */ + +#ifndef cglms_euler_to_quat_lh_h +#define cglms_euler_to_quat_lh_h + +#include "../common.h" + + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in x y z order in left hand (roll pitch yaw) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +versors +glms_euler_xyz_quat_lh(vec3s angles) { + versors dest; + glm_euler_xyz_quat_lh(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in x z y order in left hand (roll yaw pitch) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +versors +glms_euler_xzy_quat_lh(vec3s angles) { + versors dest; + glm_euler_xzy_quat_lh(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in y x z order in left hand (pitch roll yaw) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +versors +glms_euler_yxz_quat_lh(vec3s angles) { + versors dest; + glm_euler_yxz_quat_lh(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in y z x order in left hand (pitch yaw roll) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +versors +glms_euler_yzx_quat_lh(vec3s angles) { + versors dest; + glm_euler_yzx_quat_lh(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in z x y order in left hand (yaw roll pitch) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +versors +glms_euler_zxy_quat_lh(vec3s angles) { + versors dest; + glm_euler_zxy_quat_lh(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in z y x order in left hand (yaw pitch roll) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +versors +glms_euler_zyx_quat_lh(vec3s angles) { + versors dest; + glm_euler_zyx_quat_lh(angles.raw, dest.raw); + return dest; +} + + +#endif /* cglms_euler_to_quat_lh_h */ diff --git a/cglm/include/cglm/struct/handed/euler_to_quat_rh.h b/cglm/include/cglm/struct/handed/euler_to_quat_rh.h new file mode 100644 index 0000000..6c7f400 --- /dev/null +++ b/cglm/include/cglm/struct/handed/euler_to_quat_rh.h @@ -0,0 +1,115 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glms_euler_xyz_quat_rh(vec3 angles, versor dest); + CGLM_INLINE void glms_euler_xzy_quat_rh(vec3 angles, versor dest); + CGLM_INLINE void glms_euler_yxz_quat_rh(vec3 angles, versor dest); + CGLM_INLINE void glms_euler_yzx_quat_rh(vec3 angles, versor dest); + CGLM_INLINE void glms_euler_zxy_quat_rh(vec3 angles, versor dest); + CGLM_INLINE void glms_euler_zyx_quat_rh(vec3 angles, versor dest); + */ + +#ifndef cglms_euler_to_quat_rh_h +#define cglms_euler_to_quat_rh_h + +#include "../common.h" + + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in x y z order in right hand (roll pitch yaw) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +versors +glms_euler_xyz_quat_rh(vec3s angles) { + versors dest; + glm_euler_xyz_quat_rh(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in x z y order in right hand (roll yaw pitch) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +versors +glms_euler_xzy_quat_rh(vec3s angles) { + versors dest; + glm_euler_xzy_quat_rh(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in y x z order in right hand (pitch roll yaw) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +versors +glms_euler_yxz_quat_rh(vec3s angles) { + versors dest; + glm_euler_yxz_quat_rh(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in y z x order in right hand (pitch yaw roll) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +versors +glms_euler_yzx_quat_rh(vec3s angles) { + versors dest; + glm_euler_yzx_quat_rh(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in z x y order in right hand (yaw roll pitch) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +versors +glms_euler_zxy_quat_rh(vec3s angles) { + versors dest; + glm_euler_zxy_quat_rh(angles.raw, dest.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion using rotation angles and does + * rotations in z y x order in right hand (yaw pitch roll) + * + * @param[in] angles angles x y z (radians) + * @param[out] dest quaternion + */ +CGLM_INLINE +versors +glms_euler_zyx_quat_rh(vec3s angles) { + versors dest; + glm_euler_zyx_quat_rh(angles.raw, dest.raw); + return dest; +} + + +#endif /* cglms_euler_to_quat_rh_h */ diff --git a/cglm/include/cglm/struct/io.h b/cglm/include/cglm/struct/io.h new file mode 100644 index 0000000..900c2a8 --- /dev/null +++ b/cglm/include/cglm/struct/io.h @@ -0,0 +1,107 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glms_mat4_print(mat4s matrix, FILE *ostream); + CGLM_INLINE void glms_mat3_print(mat3s matrix, FILE *ostream); + CGLM_INLINE void glms_vec4_print(vec4s vec, FILE *ostream); + CGLM_INLINE void glms_ivec4_print(ivec3s vec, FILE *ostream); + CGLM_INLINE void glms_vec3_print(vec3s vec, FILE *ostream); + CGLM_INLINE void glms_ivec3_print(ivec3s vec, FILE *ostream); + CGLM_INLINE void glms_vec2_print(vec2s vec, FILE *ostream); + CGLM_INLINE void glms_ivec2_print(ivec3s vec, FILE *ostream); + CGLM_INLINE void glms_versor_print(versor vec, FILE *ostream); + CGLM_INLINE void glms_aabb_print(vec3s bbox[2], const char *tag, FILE *ostream); + */ + +#ifndef cglms_ios_h +#define cglms_ios_h + +#include "../common.h" +#include "../io.h" +#include "mat4.h" + +#include +#include + +CGLM_INLINE +void +glms_mat4_print(mat4s matrix, + FILE * __restrict ostream) { + + glm_mat4_print(matrix.raw, ostream); +} + +CGLM_INLINE +void +glms_mat3_print(mat3s matrix, + FILE * __restrict ostream) { + glm_mat3_print(matrix.raw, ostream); +} + +CGLM_INLINE +void +glms_vec4_print(vec4s vec, + FILE * __restrict ostream) { + glm_vec4_print(vec.raw, ostream); +} + +CGLM_INLINE +void +glms_ivec4_print(ivec4s vec, + FILE * __restrict ostream) { + glm_ivec4_print(vec.raw, ostream); +} + +CGLM_INLINE +void +glms_vec3_print(vec3s vec, + FILE * __restrict ostream) { + glm_vec3_print(vec.raw, ostream); +} + +CGLM_INLINE +void +glms_ivec3_print(ivec3s vec, + FILE * __restrict ostream) { + glm_ivec3_print(vec.raw, ostream); +} + +CGLM_INLINE +void +glms_vec2_print(vec2s vec, + FILE * __restrict ostream) { + glm_vec2_print(vec.raw, ostream); +} + +CGLM_INLINE +void +glms_ivec2_print(ivec2s vec, + FILE * __restrict ostream) { + glm_ivec2_print(vec.raw, ostream); +} + +CGLM_INLINE +void +glms_versor_print(versors vec, + FILE * __restrict ostream) { + glm_versor_print(vec.raw, ostream); +} + +CGLM_INLINE +void +glms_aabb_print(vec3s bbox[2], + const char * __restrict tag, + FILE * __restrict ostream) { + vec3 rawBbox[2]; + + glms_vec3_(unpack)(rawBbox, bbox, 2); + glm_aabb_print(rawBbox, tag, ostream); +} + +#endif /* cglms_ios_h */ diff --git a/cglm/include/cglm/struct/ivec2.h b/cglm/include/cglm/struct/ivec2.h new file mode 100644 index 0000000..d53c9f6 --- /dev/null +++ b/cglm/include/cglm/struct/ivec2.h @@ -0,0 +1,708 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLMS_IVEC2_ONE_INIT + GLMS_IVEC2_ZERO_INIT + GLMS_IVEC2_ONE + GLMS_IVEC2_ZERO + + Functions: + CGLM_INLINE ivec2s glms_ivec2(int * __restrict v) + CGLM_INLINE void glms_ivec2_pack(ivec2s dst[], ivec2s src[], size_t len) + CGLM_INLINE void glms_ivec2_unpack(ivec2 dst[], ivec2 src[], size_t len) + CGLM_INLINE ivec2s glms_ivec2_zero(ivec2s v) + CGLM_INLINE ivec2s glms_ivec2_one(ivec2s v) + CGLM_INLINE int glms_ivec2_dot(ivec2s a, ivec2s b) + CGLM_INLINE int glms_ivec2_cross(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_add(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_adds(ivec2s v, int s) + CGLM_INLINE ivec2s glms_ivec2_sub(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_subs(ivec2s v, int s) + CGLM_INLINE ivec2s glms_ivec2_mul(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_scale(ivec2s v, int s) + CGLM_INLINE ivec2s glms_ivec2_div(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_divs(ivec2s v, int s) + CGLM_INLINE ivec2s glms_ivec2_mod(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_addadd(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_addadds(ivec2s a, int s) + CGLM_INLINE ivec2s glms_ivec2_subadd(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_subadds(ivec2s a, int s) + CGLM_INLINE ivec2s glms_ivec2_muladd(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_muladds(ivec2s a, int s) + CGLM_INLINE ivec2s glms_ivec2_maxadd(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_minadd(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_subsub(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_subsubs(ivec2s a, int s) + CGLM_INLINE ivec2s glms_ivec2_addsub(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_addsubs(ivec2s a, int s) + CGLM_INLINE ivec2s glms_ivec2_mulsub(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_mulsubs(ivec2s a, int s) + CGLM_INLINE ivec2s glms_ivec2_maxsub(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_minsub(ivec2s a, ivec2s b) + CGLM_INLINE int glms_ivec2_distance2(ivec2s a, ivec2s b) + CGLM_INLINE float glms_ivec2_distance(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_fill(int val) + CGLM_INLINE bool glms_ivec2_eq(ivec2s v, int val); + CGLM_INLINE bool glms_ivec2_eqv(ivec2s a, ivec2s b); + CGLM_INLINE ivec2s glms_ivec2_maxv(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_minv(ivec2s a, ivec2s b) + CGLM_INLINE ivec2s glms_ivec2_clamp(ivec2s v, int minVal, int maxVal) + CGLM_INLINE ivec2s glms_ivec2_abs(ivec2s v) + */ + +#ifndef cglms_ivec2_h +#define cglms_ivec2_h + +#include "../common.h" +#include "../types-struct.h" +#include "../ivec2.h" + +#define glms_ivec2_(NAME) CGLM_STRUCTAPI(ivec2, NAME) + +#define GLMS_IVEC2_ONE_INIT {GLM_IVEC2_ONE_INIT} +#define GLMS_IVEC2_ZERO_INIT {GLM_IVEC2_ZERO_INIT} + +#define GLMS_IVEC2_ONE ((ivec2s)GLMS_IVEC2_ONE_INIT) +#define GLMS_IVEC2_ZERO ((ivec2s)GLMS_IVEC2_ZERO_INIT) + +/*! + * @brief init ivec2 using ivec3 or ivec4 + * + * @param[in] v vector + * @returns destination + */ +CGLM_INLINE +ivec2s +glms_ivec2(int * __restrict v) { + ivec2s r; + glm_ivec2(v, r.raw); + return r; +} + +/*! + * @brief pack an array of ivec2 into an array of ivec2s + * + * @param[out] dst array of ivec2s + * @param[in] src array of ivec2 + * @param[in] len number of elements + */ +CGLM_INLINE +void +glms_ivec2_(pack)(ivec2s dst[], ivec2 src[], size_t len) { + size_t i; + + for (i = 0; i < len; i++) { + glm_ivec2_copy(src[i], dst[i].raw); + } +} + +/*! + * @brief unpack an array of ivec2s into an array of ivec2 + * + * @param[out] dst array of ivec2 + * @param[in] src array of ivec2s + * @param[in] len number of elements + */ +CGLM_INLINE +void +glms_ivec2_(unpack)(ivec2 dst[], ivec2s src[], size_t len) { + size_t i; + + for (i = 0; i < len; i++) { + glm_ivec2_copy(src[i].raw, dst[i]); + } +} + +/*! + * @brief set all members of [v] to zero + * + * @returns vector + */ +CGLM_INLINE +ivec2s +glms_ivec2_(zero)(void) { + ivec2s r; + glm_ivec2_zero(r.raw); + return r; +} + +/*! + * @brief set all members of [v] to one + * + * @returns vector + */ +CGLM_INLINE +ivec2s +glms_ivec2_(one)(void) { + ivec2s r; + glm_ivec2_one(r.raw); + return r; +} + +/*! + * @brief ivec2 dot product + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return dot product + */ +CGLM_INLINE +int +glms_ivec2_(dot)(ivec2s a, ivec2s b) { + return glm_ivec2_dot(a.raw, b.raw); +} + +/*! + * @brief ivec2 cross product + * + * REF: http://allenchou.net/2013/07/cross-product-of-2d-vectors/ + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return Z component of cross product + */ +CGLM_INLINE +int +glms_ivec2_(cross)(ivec2s a, ivec2s b) { + return glm_ivec2_cross(a.raw, b.raw); +} + +/*! + * @brief add vector [a] to vector [b] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @returns destination + */ +CGLM_INLINE +ivec2s +glms_ivec2_(add)(ivec2s a, ivec2s b) { + ivec2s r; + glm_ivec2_add(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief add scalar s to vector [v] and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination + */ +CGLM_INLINE +ivec2s +glms_ivec2_(adds)(ivec2s v, int s) { + ivec2s r; + glm_ivec2_adds(v.raw, s, r.raw); + return r; +} + +/*! + * @brief subtract vector [b] from vector [a] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @returns destination + */ +CGLM_INLINE +ivec2s +glms_ivec2_(sub)(ivec2s a, ivec2s b) { + ivec2s r; + glm_ivec2_sub(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief subtract scalar s from vector [v] and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination + */ +CGLM_INLINE +ivec2s +glms_ivec2_(subs)(ivec2s v, int s) { + ivec2s r; + glm_ivec2_subs(v.raw, s, r.raw); + return r; +} + +/*! + * @brief multiply vector [a] with vector [b] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @returns destination + */ +CGLM_INLINE +ivec2s +glms_ivec2_(mul)(ivec2s a, ivec2s b) { + ivec2s r; + glm_ivec2_mul(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief multiply vector [a] with scalar s and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination + */ +CGLM_INLINE +ivec2s +glms_ivec2_(scale)(ivec2s v, int s) { + ivec2s r; + glm_ivec2_scale(v.raw, s, r.raw); + return r; +} + +/*! + * @brief div vector with another component-wise division: d = a / b + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns result = (a[0]/b[0], a[1]/b[1]) + */ +CGLM_INLINE +ivec2s +glms_ivec2_(div)(ivec2s a, ivec2s b) { + ivec2s r; + glm_ivec2_div(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief div vector with scalar: d = v / s + * + * @param[in] v vector + * @param[in] s scalar + * @returns result = (a[0]/s, a[1]/s) + */ +CGLM_INLINE +ivec2s +glms_ivec2_(divs)(ivec2s v, int s) { + ivec2s r; + glm_ivec2_divs(v.raw, s, r.raw); + return r; +} + +/*! + * @brief mod vector with another component-wise modulo: d = a % b + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns result = (a[0]%b[0], a[1]%b[1]) + */ +CGLM_INLINE +ivec2s +glms_ivec2_(mod)(ivec2s a, ivec2s b) { + ivec2s r; + glm_ivec2_mod(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief add vector [a] with vector [b] and add result to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest += (a + b) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(addadd)(ivec2s a, ivec2s b, ivec2s dest) { + glm_ivec2_addadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief add scalar [s] onto vector [a] and add result to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[in] dest dest += (a + s) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(addadds)(ivec2s a, int s, ivec2s dest) { + glm_ivec2_addadds(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief subtract vector [a] from vector [b] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest += (a - b) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(subadd)(ivec2s a, ivec2s b, ivec2s dest) { + glm_ivec2_subadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief subtract scalar [s] from vector [a] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first + * @param[in] s scalar + * @param[in] dest dest += (a - s) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(subadds)(ivec2s a, int s, ivec2s dest) { + glm_ivec2_subadds(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief multiply vector [a] with vector [b] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest += (a * b) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(muladd)(ivec2s a, ivec2s b, ivec2s dest) { + glm_ivec2_muladd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief multiply vector [a] with scalar [s] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[in] dest dest += (a * s) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(muladds)(ivec2s a, int s, ivec2s dest) { + glm_ivec2_muladds(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief add maximum of vector [a] and vector [b] to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest += max(a, b) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(maxadd)(ivec2s a, ivec2s b, ivec2s dest) { + glm_ivec2_maxadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief add minimum of vector [a] and vector [b] to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest += min(a, b) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(minadd)(ivec2s a, ivec2s b, ivec2s dest) { + glm_ivec2_minadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief subtract vector [a] from vector [b] and subtract result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest -= (a - b) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(subsub)(ivec2s a, ivec2s b, ivec2s dest) { + glm_ivec2_subsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief subtract scalar [s] from vector [a] and subtract result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[in] dest dest -= (a - s) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(subsubs)(ivec2s a, int s, ivec2s dest) { + glm_ivec2_subsubs(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief add vector [a] to vector [b] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] b scalar + * @param[in] dest dest -= (a + b) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(addsub)(ivec2s a, ivec2s b, ivec2s dest) { + glm_ivec2_addsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief add scalar [s] to vector [a] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[in] dest dest -= (a + b) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(addsubs)(ivec2s a, int s, ivec2s dest) { + glm_ivec2_addsubs(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief multiply vector [a] and vector [b] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] b scalar + * @param[in] dest dest -= (a * b) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(mulsub)(ivec2s a, ivec2s b, ivec2s dest) { + glm_ivec2_mulsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief multiply vector [a] with scalar [s] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[in] dest dest -= (a * s) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(mulsubs)(ivec2s a, int s, ivec2s dest) { + glm_ivec2_mulsubs(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief subtract maximum of vector [a] and vector [b] from vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest -= max(a, b) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(maxsub)(ivec2s a, ivec2s b, ivec2s dest) { + glm_ivec2_maxsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief subtract minimum of vector [a] and vector [b] from vector [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest -= min(a, b) + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(minsub)(ivec2s a, ivec2s b, ivec2s dest) { + glm_ivec2_minsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief squared distance between two vectors + * + * @param[in] a first vector + * @param[in] b second vector + * @return returns squared distance (distance * distance) + */ +CGLM_INLINE +int +glms_ivec2_(distance2)(ivec2s a, ivec2s b) { + return glm_ivec2_distance2(a.raw, b.raw); +} + +/*! + * @brief distance between two vectors + * + * @param[in] a first vector + * @param[in] b second vector + * @return returns distance + */ +CGLM_INLINE +float +glms_ivec2_(distance)(ivec2s a, ivec2s b) { + return glm_ivec2_distance(a.raw, b.raw); +} + +/*! + * @brief fill a vector with specified value + * + * @param[in] val value + * @returns dest + */ +CGLM_INLINE +ivec2s +glms_ivec2_(fill)(int val) { + ivec2s r; + glm_ivec2_fill(r.raw, val); + return r; +} + +/*! + * @brief check if vector is equal to value + * + * @param[in] v vector + * @param[in] val value + */ +CGLM_INLINE +bool +glms_ivec2_(eq)(ivec2s v, int val) { + return glm_ivec2_eq(v.raw, val); +} + +/*! + * @brief check if vector is equal to another + * + * @param[in] a vector + * @param[in] b vector + */ +CGLM_INLINE +bool +glms_ivec2_(eqv)(ivec2s a, ivec2s b) { + return glm_ivec2_eqv(a.raw, b.raw); +} + +/*! + * @brief set each member of dest to greater of vector a and b + * + * @param[in] a first vector + * @param[in] b second vector + * @returns destination + */ +CGLM_INLINE +ivec2s +glms_ivec2_(maxv)(ivec2s a, ivec2s b) { + ivec2s r; + glm_ivec2_maxv(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief set each member of dest to lesser of vector a and b + * + * @param[in] a first vector + * @param[in] b second vector + * @returns destination + */ +CGLM_INLINE +ivec2s +glms_ivec2_(minv)(ivec2s a, ivec2s b) { + ivec2s r; + glm_ivec2_minv(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief clamp each member of [v] between minVal and maxVal (inclusive) + * + * @param[in] v vector + * @param[in] minVal minimum value + * @param[in] maxVal maximum value + * @returns clamped vector + */ +CGLM_INLINE +ivec2s +glms_ivec2_(clamp)(ivec2s v, int minVal, int maxVal) { + glm_ivec2_clamp(v.raw, minVal, maxVal); + return v; +} + +/*! + * @brief absolute value of v + * + * @param[in] v vector + * @returns destination + */ +CGLM_INLINE +ivec2s +glms_ivec2_(abs)(ivec2s v) { + ivec2s r; + glm_ivec2_abs(v.raw, r.raw); + return r; +} + +#endif /* cglms_ivec2_h */ diff --git a/cglm/include/cglm/struct/ivec3.h b/cglm/include/cglm/struct/ivec3.h new file mode 100644 index 0000000..c2c5f3b --- /dev/null +++ b/cglm/include/cglm/struct/ivec3.h @@ -0,0 +1,725 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLMS_IVEC3_ONE_INIT + GLMS_IVEC3_ZERO_INIT + GLMS_IVEC3_ONE + GLMS_IVEC3_ZERO + + Functions: + CGLM_INLINE ivec3s glms_ivec3(ivec4s v4) + CGLM_INLINE void glms_ivec3_pack(ivec3s dst[], ivec3 src[], size_t len) + CGLM_INLINE void glms_ivec3_unpack(ivec3 dst[], ivec3s src[], size_t len) + CGLM_INLINE ivec3s glms_ivec3_zero(void) + CGLM_INLINE ivec3s glms_ivec3_one(void) + CGLM_INLINE int glms_ivec3_dot(ivec3s a, ivec3s b) + CGLM_INLINE int glms_ivec3_norm2(ivec3s v) + CGLM_INLINE int glms_ivec3_norm(ivec3s v) + CGLM_INLINE ivec3s glms_ivec3_add(ivec3s a, ivec3s b) + CGLM_INLINE ivec3s glms_ivec3_adds(ivec3s v, int s) + CGLM_INLINE ivec3s glms_ivec3_sub(ivec3s a, ivec3s b) + CGLM_INLINE ivec3s glms_ivec3_subs(ivec3s v, int s) + CGLM_INLINE ivec3s glms_ivec3_mul(ivec3s a, ivec3s b) + CGLM_INLINE ivec3s glms_ivec3_scale(ivec3s v, int s) + CGLM_INLINE ivec3s glms_ivec3_div(ivec3s a, ivec3s b) + CGLM_INLINE ivec3s glms_ivec3_divs(ivec3s v, int s) + CGLM_INLINE ivec3s glms_ivec3_mod(ivec3s a, ivec3s b) + CGLM_INLINE ivec3s glms_ivec3_addadd(ivec3s a, ivec3s b, ivec3s dest) + CGLM_INLINE ivec3s glms_ivec3_addadds(ivec3s a, int s, ivec3s dest) + CGLM_INLINE ivec3s glms_ivec3_subadd(ivec3s a, ivec3s b, ivec3s dest) + CGLM_INLINE ivec3s glms_ivec3_subadds(ivec3s a, int s, ivec3s dest) + CGLM_INLINE ivec3s glms_ivec3_muladd(ivec3s a, ivec3s b, ivec3s dest) + CGLM_INLINE ivec3s glms_ivec3_muladds(ivec3s a, int s, ivec3s dest) + CGLM_INLINE ivec3s glms_ivec3_minadd(ivec3s a, ivec3s b, ivec3s dest) + CGLM_INLINE ivec3s glms_ivec3_subsub(ivec3s a, ivec3s b, ivec3s dest) + CGLM_INLINE ivec3s glms_ivec3_subsubs(ivec3s a, int s, ivec3s dest) + CGLM_INLINE ivec3s glms_ivec3_addsub(ivec3s a, ivec3s b, ivec3s dest) + CGLM_INLINE ivec3s glms_ivec3_addsubs(ivec3s a, int s, ivec3s dest) + CGLM_INLINE ivec3s glms_ivec3_mulsub(ivec3s a, ivec3s b, ivec3s dest) + CGLM_INLINE ivec3s glms_ivec3_mulsubs(ivec3s a, int s, ivec3s dest) + CGLM_INLINE ivec3s glms_ivec3_maxsub(ivec3s a, ivec3s b, ivec3s dest) + CGLM_INLINE ivec3s glms_ivec3_minsub(ivec3s a, ivec3s b, ivec3s dest) + CGLM_INLINE int glms_ivec3_distance2(ivec3s a, ivec3s b) + CGLM_INLINE float glms_ivec3_distance(ivec3s a, ivec3s b) + CGLM_INLINE ivec3s glms_ivec3_fill(int val) + CGLM_INLINE bool glms_ivec3_eq(ivec3s v, int val) + CGLM_INLINE bool glms_ivec3_eqv(ivec3s a, ivec3s b) + CGLM_INLINE ivec3s glms_ivec3_maxv(ivec3s a, ivec3s b) + CGLM_INLINE ivec3s glms_ivec3_minv(ivec3s a, ivec3s b) + CGLM_INLINE ivec3s glms_ivec3_clamp(ivec3s v, int minVal, int maxVal) + CGLM_INLINE ivec3s glms_ivec3_abs(ivec3s v) + */ + +#ifndef cglms_ivec3_h +#define cglms_ivec3_h + +#include "../common.h" +#include "../types-struct.h" +#include "../ivec3.h" + +#define glms_ivec3_(NAME) CGLM_STRUCTAPI(ivec3, NAME) + +#define GLMS_IVEC3_ONE_INIT {GLM_IVEC3_ONE_INIT} +#define GLMS_IVEC3_ZERO_INIT {GLM_IVEC3_ZERO_INIT} + +#define GLMS_IVEC3_ONE ((ivec3s)GLMS_IVEC3_ONE_INIT) +#define GLMS_IVEC3_ZERO ((ivec3s)GLMS_IVEC3_ZERO_INIT) + +/*! + * @brief init ivec3 using ivec4 + * + * @param[in] v4 vector4 + * @returns destination + */ +CGLM_INLINE +ivec3s +glms_ivec3(ivec4s v4) { + ivec3s r; + glm_ivec3(v4.raw, r.raw); + return r; +} + +/*! + * @brief pack an array of ivec3 into an array of ivec3s + * + * @param[out] dst array of ivec3s + * @param[in] src array of ivec3 + * @param[in] len number of elements + */ +CGLM_INLINE +void +glms_ivec3_(pack)(ivec3s dst[], ivec3 src[], size_t len) { + size_t i; + + for (i = 0; i < len; i++) { + glm_ivec3_copy(src[i], dst[i].raw); + } +} + +/*! + * @brief unpack an array of ivec3s into an array of ivec3 + * + * @param[out] dst array of ivec3 + * @param[in] src array of ivec3s + * @param[in] len number of elements + */ +CGLM_INLINE +void +glms_ivec3_(unpack)(ivec3 dst[], ivec3s src[], size_t len) { + size_t i; + + for (i = 0; i < len; i++) { + glm_ivec3_copy(src[i].raw, dst[i]); + } +} + +/*! + * @brief set all members of [v] to zero + * + * @returns vector + */ +CGLM_INLINE +ivec3s +glms_ivec3_(zero)(void) { + ivec3s r; + glm_ivec3_zero(r.raw); + return r; +} + +/*! + * @brief set all members of [v] to one + * + * @returns vector + */ +CGLM_INLINE +ivec3s +glms_ivec3_(one)(void) { + ivec3s r; + glm_ivec3_one(r.raw); + return r; +} + +/*! + * @brief ivec3 dot product + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return dot product + */ +CGLM_INLINE +int +glms_ivec3_(dot)(ivec3s a, ivec3s b) { + return glm_ivec3_dot(a.raw, b.raw); +} + +/*! + * @brief norm * norm (magnitude) of vec + * + * we can use this func instead of calling norm * norm, because it would call + * sqrtf function twice but with this func we can avoid func call, maybe this is + * not good name for this func + * + * @param[in] v vector + * + * @return norm * norm + */ +CGLM_INLINE +int +glms_ivec3_(norm2)(ivec3s v) { + return glm_ivec3_norm2(v.raw); +} + +/*! + * @brief euclidean norm (magnitude), also called L2 norm + * this will give magnitude of vector in euclidean space + * + * @param[in] v vector + * + * @return norm + */ +CGLM_INLINE +int +glms_ivec3_(norm)(ivec3s v) { + return glm_ivec3_norm(v.raw); +} + +/*! + * @brief add vector [a] to vector [b] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @returns destination + */ +CGLM_INLINE +ivec3s +glms_ivec3_(add)(ivec3s a, ivec3s b) { + ivec3s r; + glm_ivec3_add(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief add scalar s to vector [v] and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination + */ +CGLM_INLINE +ivec3s +glms_ivec3_(adds)(ivec3s v, int s) { + ivec3s r; + glm_ivec3_adds(v.raw, s, r.raw); + return r; +} + +/*! + * @brief subtract vector [b] from vector [a] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @returns destination + */ +CGLM_INLINE +ivec3s +glms_ivec3_(sub)(ivec3s a, ivec3s b) { + ivec3s r; + glm_ivec3_sub(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief subtract scalar s from vector [v] and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination + */ +CGLM_INLINE +ivec3s +glms_ivec3_(subs)(ivec3s v, int s) { + ivec3s r; + glm_ivec3_subs(v.raw, s, r.raw); + return r; +} + +/*! + * @brief multiply vector [a] with vector [b] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @returns destination + */ +CGLM_INLINE +ivec3s +glms_ivec3_(mul)(ivec3s a, ivec3s b) { + ivec3s r; + glm_ivec3_mul(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief multiply vector [a] with scalar s and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination + */ +CGLM_INLINE +ivec3s +glms_ivec3_(scale)(ivec3s v, int s) { + ivec3s r; + glm_ivec3_scale(v.raw, s, r.raw); + return r; +} + +/*! + * @brief div vector with another component-wise division: d = a / b + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns result = (a[0]/b[0], a[1]/b[1], a[2]/b[2]) + */ +CGLM_INLINE +ivec3s +glms_ivec3_(div)(ivec3s a, ivec3s b) { + ivec3s r; + glm_ivec3_div(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief div vector with scalar: d = v / s + * + * @param[in] v vector + * @param[in] s scalar + * @returns result = (a[0]/s, a[1]/s, a[2]/s) + */ +CGLM_INLINE +ivec3s +glms_ivec3_(divs)(ivec3s v, int s) { + ivec3s r; + glm_ivec3_divs(v.raw, s, r.raw); + return r; +} + +/*! + * @brief Element-wise modulo operation on ivec3 vectors: dest = a % b + * + * Performs element-wise modulo on each component of vectors `a` and `b`. + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns result = (a[0]%b[0], a[1]%b[1], a[2]%b[2]) + */ +CGLM_INLINE +ivec3s +glms_ivec3_(mod)(ivec3s a, ivec3s b) { + ivec3s r; + glm_ivec3_mod(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief add vector [a] with vector [b] and add result to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest += (a + b) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(addadd)(ivec3s a, ivec3s b, ivec3s dest) { + glm_ivec3_addadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief add scalar [s] onto vector [a] and add result to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[in] dest dest += (a + s) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(addadds)(ivec3s a, int s, ivec3s dest) { + glm_ivec3_addadds(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief subtract vector [a] from vector [b] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest += (a - b) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(subadd)(ivec3s a, ivec3s b, ivec3s dest) { + glm_ivec3_subadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief subtract scalar [s] from vector [a] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first + * @param[in] s scalar + * @param[in] dest dest += (a - s) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(subadds)(ivec3s a, int s, ivec3s dest) { + glm_ivec3_subadds(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief multiply vector [a] with vector [b] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest += (a * b) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(muladd)(ivec3s a, ivec3s b, ivec3s dest) { + glm_ivec3_muladd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief multiply vector [a] with scalar [s] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[in] dest dest += (a * s) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(muladds)(ivec3s a, int s, ivec3s dest) { + glm_ivec3_muladds(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief add maximum of vector [a] and vector [b] to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest += max(a, b) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(maxadd)(ivec3s a, ivec3s b, ivec3s dest) { + glm_ivec3_maxadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief add minimum of vector [a] and vector [b] to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest += min(a, b) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(minadd)(ivec3s a, ivec3s b, ivec3s dest) { + glm_ivec3_minadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief subtract vector [a] from vector [b] and subtract result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest -= (a - b) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(subsub)(ivec3s a, ivec3s b, ivec3s dest) { + glm_ivec3_subsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief subtract scalar [s] from vector [a] and subtract result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[in] dest dest -= (a - s) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(subsubs)(ivec3s a, int s, ivec3s dest) { + glm_ivec3_subsubs(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief add vector [a] to vector [b] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] b scalar + * @param[in] dest dest -= (a + b) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(addsub)(ivec3s a, ivec3s b, ivec3s dest) { + glm_ivec3_addsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief add scalar [s] to vector [a] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[in] dest dest -= (a + b) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(addsubs)(ivec3s a, int s, ivec3s dest) { + glm_ivec3_addsubs(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief multiply vector [a] and vector [b] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] b scalar + * @param[in] dest dest -= (a * b) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(mulsub)(ivec3s a, ivec3s b, ivec3s dest) { + glm_ivec3_mulsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief multiply vector [a] with scalar [s] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[in] dest dest -= (a * s) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(mulsubs)(ivec3s a, int s, ivec3s dest) { + glm_ivec3_mulsubs(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief subtract maximum of vector [a] and vector [b] from vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest -= max(a, b) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(maxsub)(ivec3s a, ivec3s b, ivec3s dest) { + glm_ivec3_maxsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief subtract minimum of vector [a] and vector [b] from vector [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest -= min(a, b) + * @returns dest + */ +CGLM_INLINE +ivec3s +glms_ivec3_(minsub)(ivec3s a, ivec3s b, ivec3s dest) { + glm_ivec3_minsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief squared distance between two vectors + * + * @param[in] a first vector + * @param[in] b second vector + * @return returns squared distance (distance * distance) + */ +CGLM_INLINE +int +glms_ivec3_(distance2)(ivec3s a, ivec3s b) { + return glm_ivec3_distance2(a.raw, b.raw); +} + +/*! + * @brief distance between two vectors + * + * @param[in] a first vector + * @param[in] b second vector + * @return returns distance + */ +CGLM_INLINE +float +glms_ivec3_(distance)(ivec3s a, ivec3s b) { + return glm_ivec3_distance(a.raw, b.raw); +} + +/*! + * @brief fill a vector with specified value + * + * @param[in] val value + * @returns destination + */ +CGLM_INLINE +ivec3s +glms_ivec3_(fill)(int val) { + ivec3s r; + glm_ivec3_fill(r.raw, val); + return r; +} + +/*! + * @brief check if vector is equal to value + * + * @param[in] v vector + * @param[in] val value + */ +CGLM_INLINE +bool +glms_ivec3_(eq)(ivec3s v, int val) { + return glm_ivec3_eq(v.raw, val); +} + +/*! + * @brief check if vector is equal to another + * + * @param[in] a vector + * @param[in] b vector + */ +CGLM_INLINE +bool +glms_ivec3_(eqv)(ivec3s a, ivec3s b) { + return glm_ivec3_eqv(a.raw, b.raw); +} + +/*! + * @brief set each member of dest to greater of vector a and b + * + * @param[in] a first vector + * @param[in] b second vector + * @returns destination + */ +CGLM_INLINE +ivec3s +glms_ivec3_(maxv)(ivec3s a, ivec3s b) { + ivec3s r; + glm_ivec3_maxv(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief set each member of dest to lesser of vector a and b + * + * @param[in] a first vector + * @param[in] b second vector + * @returns destination + */ +CGLM_INLINE +ivec3s +glms_ivec3_(minv)(ivec3s a, ivec3s b) { + ivec3s r; + glm_ivec3_minv(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief clamp each member of [v] between minVal and maxVal (inclusive) + * + * @param[in] v vector + * @param[in] minVal minimum value + * @param[in] maxVal maximum value + * @returns clamped vector + */ +CGLM_INLINE +ivec3s +glms_ivec3_(clamp)(ivec3s v, int minVal, int maxVal) { + glm_ivec3_clamp(v.raw, minVal, maxVal); + return v; +} + +/*! + * @brief absolute value of v + * + * @param[in] v vector + * @returns destination + */ +CGLM_INLINE +ivec3s +glms_ivec3_(abs)(ivec3s v) { + ivec3s r; + glm_ivec3_abs(v.raw, r.raw); + return r; +} + +#endif /* cglms_ivec3_h */ diff --git a/cglm/include/cglm/struct/ivec4.h b/cglm/include/cglm/struct/ivec4.h new file mode 100644 index 0000000..103e887 --- /dev/null +++ b/cglm/include/cglm/struct/ivec4.h @@ -0,0 +1,588 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLMS_IVEC4_ONE_INIT + GLMS_IVEC4_ZERO_INIT + GLMS_IVEC4_ONE + GLMS_IVEC4_ZERO + + Functions: + CGLM_INLINE ivec4s glms_ivec4(ivec3s v3, int last) + CGLM_INLINE void glms_ivec4_pack(ivec4s dst[], ivec4 src[], size_t len) + CGLM_INLINE void glms_ivec4_unpack(ivec4 dst[], ivec4s src[], size_t len) + CGLM_INLINE ivec4s glms_ivec4_zero(void) + CGLM_INLINE ivec4s glms_ivec4_one(void) + CGLM_INLINE ivec4s glms_ivec4_add(ivec4s a, ivec4s b) + CGLM_INLINE ivec4s glms_ivec4_adds(ivec4s v, int s) + CGLM_INLINE ivec4s glms_ivec4_sub(ivec4s a, ivec4s b) + CGLM_INLINE ivec4s glms_ivec4_subs(ivec4s v, int s) + CGLM_INLINE ivec4s glms_ivec4_mul(ivec4s a, ivec4s b) + CGLM_INLINE ivec4s glms_ivec4_scale(ivec4s v, int s) + CGLM_INLINE ivec4s glms_ivec4_addadd(ivec4s a, ivec4s b, ivec4s dest) + CGLM_INLINE ivec4s glms_ivec4_addadds(ivec4s a, int s, ivec4s dest) + CGLM_INLINE ivec4s glms_ivec4_subadd(ivec4s a, ivec4s b, ivec4s dest) + CGLM_INLINE ivec4s glms_ivec4_subadds(ivec4s a, int s, ivec4s dest) + CGLM_INLINE ivec4s glms_ivec4_muladd(ivec4s a, ivec4s b, ivec4s dest) + CGLM_INLINE ivec4s glms_ivec4_muladds(ivec4s a, int s, ivec4s dest) + CGLM_INLINE ivec4s glms_ivec4_maxadd(ivec4s a, ivec4s b, ivec4s dest) + CGLM_INLINE ivec4s glms_ivec4_minadd(ivec4s a, ivec4s b, ivec4s dest) + CGLM_INLINE ivec4s glms_ivec4_subsub(ivec4s a, ivec4s b, ivec4s dest) + CGLM_INLINE ivec4s glms_ivec4_subsubs(ivec4s a, int s, ivec4s dest) + CGLM_INLINE ivec4s glms_ivec4_addsub(ivec4s a, ivec4s b, ivec4s dest) + CGLM_INLINE ivec4s glms_ivec4_addsubs(ivec4s a, int s, ivec4s dest) + CGLM_INLINE ivec4s glms_ivec4_mulsub(ivec4s a, ivec4s b, ivec4s dest) + CGLM_INLINE ivec4s glms_ivec4_mulsubs(ivec4s a, int s, ivec4s dest) + CGLM_INLINE ivec4s glms_ivec4_maxsub(ivec4s a, ivec4s b, ivec4s dest) + CGLM_INLINE ivec4s glms_ivec4_minsub(ivec4s a, ivec4s b, ivec4s dest) + CGLM_INLINE int glms_ivec4_distance2(ivec4s a, ivec4s b) + CGLM_INLINE float glms_ivec4_distance(ivec4s a, ivec4s b) + CGLM_INLINE ivec4s glms_ivec4_maxv(ivec4s a, ivec4s b) + CGLM_INLINE ivec4s glms_ivec4_minv(ivec4s a, ivec4s b) + CGLM_INLINE ivec4s glms_ivec4_clamp(ivec4s v, int minVal, int maxVal) + CGLM_INLINE ivec4s glms_ivec4_abs(ivec4s v) + */ + +#ifndef cglms_ivec4_h +#define cglms_ivec4_h + +#include "../common.h" +#include "../types-struct.h" +#include "../ivec4.h" + +#define glms_ivec4_(NAME) CGLM_STRUCTAPI(ivec4, NAME) + +#define GLMS_IVEC4_ONE_INIT {GLM_IVEC4_ONE_INIT} +#define GLMS_IVEC4_ZERO_INIT {GLM_IVEC4_ZERO_INIT} + +#define GLMS_IVEC4_ONE ((ivec4s)GLMS_IVEC4_ONE_INIT) +#define GLMS_IVEC4_ZERO ((ivec4s)GLMS_IVEC4_ZERO_INIT) + +/*! + * @brief init ivec4 using ivec3 + * + * @param[in] v3 vector3 + * @param[in] last last item + * @returns destination + */ +CGLM_INLINE +ivec4s +glms_ivec4(ivec3s v3, int last) { + ivec4s r; + glm_ivec4(v3.raw, last, r.raw); + return r; +} + +/*! + * @brief pack an array of ivec4 into an array of ivec4s + * + * @param[out] dst array of ivec4s + * @param[in] src array of ivec4 + * @param[in] len number of elements + */ +CGLM_INLINE +void +glms_ivec4_(pack)(ivec4s dst[], ivec4 src[], size_t len) { + size_t i; + + for (i = 0; i < len; i++) { + glm_ivec4_copy(src[i], dst[i].raw); + } +} + +/*! + * @brief unpack an array of ivec4s into an array of ivec4 + * + * @param[out] dst array of ivec4 + * @param[in] src array of ivec4s + * @param[in] len number of elements + */ +CGLM_INLINE +void +glms_ivec4_(unpack)(ivec4 dst[], ivec4s src[], size_t len) { + size_t i; + + for (i = 0; i < len; i++) { + glm_ivec4_copy(src[i].raw, dst[i]); + } +} + +/*! + * @brief set all members of [v] to zero + * + * @returns vector + */ +CGLM_INLINE +ivec4s +glms_ivec4_(zero)(void) { + ivec4s r; + glm_ivec4_zero(r.raw); + return r; +} + +/*! + * @brief set all members of [v] to one + * + * @returns vector + */ +CGLM_INLINE +ivec4s +glms_ivec4_(one)(void) { + ivec4s r; + glm_ivec4_one(r.raw); + return r; +} + +/*! + * @brief add vector [a] to vector [b] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @returns destination + */ +CGLM_INLINE +ivec4s +glms_ivec4_(add)(ivec4s a, ivec4s b) { + ivec4s r; + glm_ivec4_add(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief add scalar s to vector [v] and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination + */ +CGLM_INLINE +ivec4s +glms_ivec4_(adds)(ivec4s v, int s) { + ivec4s r; + glm_ivec4_adds(v.raw, s, r.raw); + return r; +} + +/*! + * @brief subtract vector [b] from vector [a] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @returns destination + */ +CGLM_INLINE +ivec4s +glms_ivec4_(sub)(ivec4s a, ivec4s b) { + ivec4s r; + glm_ivec4_sub(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief subtract scalar s from vector [v] and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination + */ +CGLM_INLINE +ivec4s +glms_ivec4_(subs)(ivec4s v, int s) { + ivec4s r; + glm_ivec4_subs(v.raw, s, r.raw); + return r; +} + +/*! + * @brief multiply vector [a] with vector [b] and store result in [dest] + * + * @param[in] a first vector + * @param[in] b second vector + * @returns destination + */ +CGLM_INLINE +ivec4s +glms_ivec4_(mul)(ivec4s a, ivec4s b) { + ivec4s r; + glm_ivec4_mul(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief multiply vector [a] with scalar s and store result in [dest] + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination + */ +CGLM_INLINE +ivec4s +glms_ivec4_(scale)(ivec4s v, int s) { + ivec4s r; + glm_ivec4_scale(v.raw, s, r.raw); + return r; +} + +/*! + * @brief add vector [a] with vector [b] and add result to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest += (a + b) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(addadd)(ivec4s a, ivec4s b, ivec4s dest) { + glm_ivec4_addadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief add scalar [s] onto vector [a] and add result to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[in] dest dest += (a + s) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(addadds)(ivec4s a, int s, ivec4s dest) { + glm_ivec4_addadds(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief subtract vector [a] from vector [b] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest += (a - b) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(subadd)(ivec4s a, ivec4s b, ivec4s dest) { + glm_ivec4_subadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief subtract scalar [s] from vector [a] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first + * @param[in] s scalar + * @param[in] dest dest += (a - s) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(subadds)(ivec4s a, int s, ivec4s dest) { + glm_ivec4_subadds(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief multiply vector [a] with vector [b] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest += (a * b) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(muladd)(ivec4s a, ivec4s b, ivec4s dest) { + glm_ivec4_muladd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief multiply vector [a] with scalar [s] and add result to [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[in] dest dest += (a * s) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(muladds)(ivec4s a, int s, ivec4s dest) { + glm_ivec4_muladds(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief add maximum of vector [a] and vector [b] to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest += max(a, b) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(maxadd)(ivec4s a, ivec4s b, ivec4s dest) { + glm_ivec4_maxadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief add minimum of vector [a] and vector [b] to vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest += min(a, b) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(minadd)(ivec4s a, ivec4s b, ivec4s dest) { + glm_ivec4_minadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief subtract vector [a] from vector [b] and subtract result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest -= (a - b) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(subsub)(ivec4s a, ivec4s b, ivec4s dest) { + glm_ivec4_subsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief subtract scalar [s] from vector [a] and subtract result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[in] dest dest -= (a - s) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(subsubs)(ivec4s a, int s, ivec4s dest) { + glm_ivec4_subsubs(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief add vector [a] to vector [b] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] b scalar + * @param[in] dest dest -= (a + b) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(addsub)(ivec4s a, ivec4s b, ivec4s dest) { + glm_ivec4_addsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief add scalar [s] to vector [a] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[in] dest dest -= (a + b) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(addsubs)(ivec4s a, int s, ivec4s dest) { + glm_ivec4_addsubs(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief multiply vector [a] and vector [b] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] b scalar + * @param[in] dest dest -= (a * b) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(mulsub)(ivec4s a, ivec4s b, ivec4s dest) { + glm_ivec4_mulsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief multiply vector [a] with scalar [s] and subtract the result from [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[in] dest dest -= (a * s) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(mulsubs)(ivec4s a, int s, ivec4s dest) { + glm_ivec4_mulsubs(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief subtract maximum of vector [a] and vector [b] from vector [dest] + * + * applies += operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest -= max(a, b) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(maxsub)(ivec4s a, ivec4s b, ivec4s dest) { + glm_ivec4_maxsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief subtract minimum of vector [a] and vector [b] from vector [dest] + * + * applies -= operator so dest must be initialized + * + * @param[in] a first vector + * @param[in] b second vector + * @param[in] dest dest -= min(a, b) + * @returns dest + */ +CGLM_INLINE +ivec4s +glms_ivec4_(minsub)(ivec4s a, ivec4s b, ivec4s dest) { + glm_ivec4_minsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief squared distance between two vectors + * + * @param[in] a first vector + * @param[in] b second vector + * @return returns squared distance (distance * distance) + */ +CGLM_INLINE +int +glms_ivec4_(distance2)(ivec4s a, ivec4s b) { + return glm_ivec4_distance2(a.raw, b.raw); +} + +/*! + * @brief distance between two vectors + * + * @param[in] a first vector + * @param[in] b second vector + * @return returns distance + */ +CGLM_INLINE +float +glms_ivec4_(distance)(ivec4s a, ivec4s b) { + return glm_ivec4_distance(a.raw, b.raw); +} + +/*! + * @brief set each member of dest to greater of vector a and b + * + * @param[in] a first vector + * @param[in] b second vector + * @returns destination + */ +CGLM_INLINE +ivec4s +glms_ivec4_(maxv)(ivec4s a, ivec4s b) { + ivec4s r; + glm_ivec4_maxv(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief set each member of dest to lesser of vector a and b + * + * @param[in] a first vector + * @param[in] b second vector + * @returns destination + */ +CGLM_INLINE +ivec4s +glms_ivec4_(minv)(ivec4s a, ivec4s b) { + ivec4s r; + glm_ivec4_minv(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief clamp each member of [v] between minVal and maxVal (inclusive) + * + * @param[in] v vector + * @param[in] minVal minimum value + * @param[in] maxVal maximum value + * @returns clamped vector + */ +CGLM_INLINE +ivec4s +glms_ivec4_(clamp)(ivec4s v, int minVal, int maxVal) { + glm_ivec4_clamp(v.raw, minVal, maxVal); + return v; +} + +/*! + * @brief absolute value of v + * + * @param[in] v vector + * @returns destination + */ +CGLM_INLINE +ivec4s +glms_ivec4_(abs)(ivec4s v) { + ivec4s r; + glm_ivec4_abs(v.raw, r.raw); + return r; +} + +#endif /* cglms_ivec4_h */ diff --git a/cglm/include/cglm/struct/mat2.h b/cglm/include/cglm/struct/mat2.h new file mode 100644 index 0000000..5942ced --- /dev/null +++ b/cglm/include/cglm/struct/mat2.h @@ -0,0 +1,276 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_MAT2_IDENTITY_INIT + GLM_MAT2_ZERO_INIT + GLM_MAT2_IDENTITY + GLM_MAT2_ZERO + + Functions: + CGLM_INLINE void glms_mat2_identity(mat2 mat) + CGLM_INLINE void glms_mat2_identity_array(mat2 * restrict mat, size_t count) + CGLM_INLINE void glms_mat2_zero(mat2 mat) + CGLM_INLINE void glms_mat2_mul(mat2 m1, mat2 m2, mat2 dest) + CGLM_INLINE void glms_mat2_transpose_to(mat2 m, mat2 dest) + CGLM_INLINE void glms_mat2_transpose(mat2 m) + CGLM_INLINE void glms_mat2_mulv(mat2 m, vec2 v, vec2 dest) + CGLM_INLINE float glms_mat2_trace(mat2 m) + CGLM_INLINE void glms_mat2_scale(mat2 m, float s) + CGLM_INLINE float glms_mat2_det(mat2 mat) + CGLM_INLINE void glms_mat2_inv(mat2 mat, mat2 dest) + CGLM_INLINE void glms_mat2_swap_col(mat2 mat, int col1, int col2) + CGLM_INLINE void glms_mat2_swap_row(mat2 mat, int row1, int row2) + CGLM_INLINE float glms_mat2_rmc(vec2 r, mat2 m, vec2 c) + CGLM_INLINE mat2s glms_mat2_make(const float * __restrict src); + */ + +#ifndef cglms_mat2_h +#define cglms_mat2_h + +#include "../common.h" +#include "../types-struct.h" +#include "../mat2.h" + +/* api definition */ +#define glms_mat2_(NAME) CGLM_STRUCTAPI(mat2, NAME) + +#define GLMS_MAT2_IDENTITY_INIT {GLM_MAT2_IDENTITY_INIT} +#define GLMS_MAT2_ZERO_INIT {GLM_MAT2_ZERO_INIT} + +/* for C only */ +#define GLMS_MAT2_IDENTITY ((mat2s)GLMS_MAT2_IDENTITY_INIT) +#define GLMS_MAT2_ZERO ((mat2s)GLMS_MAT2_ZERO_INIT) + +/*! + * @brief make given matrix identity. It is identical with below, + * but it is more easy to do that with this func especially for members + * e.g. glm_mat2_identity(aStruct->aMatrix); + * + * @code + * glm_mat2_copy(GLM_MAT2_IDENTITY, mat); // C only + * + * // or + * mat2 mat = GLM_MAT2_IDENTITY_INIT; + * @endcode + * + * @returns identity matrix + */ +CGLM_INLINE +mat2s +glms_mat2_(identity)(void) { + mat2s r; + glm_mat2_identity(r.raw); + return r; +} + +/*! + * @brief make given matrix array's each element identity matrix + * + * @param[in, out] mat matrix array (must be aligned (16) + * if alignment is not disabled) + * + * @param[in] count count of matrices + */ +CGLM_INLINE +void +glms_mat2_(identity_array)(mat2s * __restrict mat, size_t count) { + CGLM_ALIGN_MAT mat2s t = GLMS_MAT2_IDENTITY_INIT; + size_t i; + + for (i = 0; i < count; i++) { + glm_mat2_copy(t.raw, mat[i].raw); + } +} + +/*! + * @brief make given matrix zero. + * + * @returns matrix + */ +CGLM_INLINE +mat2s +glms_mat2_(zero)(void) { + mat2s r; + glm_mat2_zero(r.raw); + return r; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * m1, m2 and dest matrices can be same matrix, it is possible to write this: + * + * @code + * mat2 m = GLM_MAT2_IDENTITY_INIT; + * r = glms_mat2_mul(m, m); + * @endcode + * + * @param[in] m1 left matrix + * @param[in] m2 right matrix + * + * @returns destination matrix + */ +CGLM_INLINE +mat2s +glms_mat2_(mul)(mat2s m1, mat2s m2) { + mat2s r; + glm_mat2_mul(m1.raw, m2.raw, r.raw); + return r; +} + +/*! + * @brief transpose mat2 + * + * @param[in] m matrix to transpose + * + * @returns transposed matrix + */ +CGLM_INLINE +mat2s +glms_mat2_(transpose)(mat2s m) { + glm_mat2_transpose(m.raw); + return m; +} + +/*! + * @brief multiply mat2 with vec2 (column vector) and store in dest vector + * + * @param[in] m mat2 (left) + * @param[in] v vec2 (right, column vector) + * @returns vec2 (result, column vector) + */ +CGLM_INLINE +vec2s +glms_mat2_(mulv)(mat2s m, vec2s v) { + vec2s r; + glm_mat2_mulv(m.raw, v.raw, r.raw); + return r; +} + +/*! + * @brief trace of matrix + * + * sum of the elements on the main diagonal from upper left to the lower right + * + * @param[in] m matrix + */ +CGLM_INLINE +float +glms_mat2_(trace)(mat2s m) { + return glm_mat2_trace(m.raw); +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + * @returns matrix + */ +CGLM_INLINE +mat2s +glms_mat2_(scale)(mat2s m, float s) { + glm_mat2_scale(m.raw, s); + return m; +} + +/*! + * @brief mat2 determinant + * + * @param[in] mat matrix + * + * @return determinant + */ +CGLM_INLINE +float +glms_mat2_(det)(mat2s mat) { + return glm_mat2_det(mat.raw); +} + +/*! + * @brief inverse mat2 and store in dest + * + * @param[in] mat matrix + * @returns matrix + */ +CGLM_INLINE +mat2s +glms_mat2_(inv)(mat2s mat) { + mat2s r; + glm_mat2_inv(mat.raw, r.raw); + return r; +} + +/*! + * @brief swap two matrix columns + * + * @param[in] mat matrix + * @param[in] col1 col1 + * @param[in] col2 col2 + * @returns matrix + */ +CGLM_INLINE +mat2s +glms_mat2_(swap_col)(mat2s mat, int col1, int col2) { + glm_mat2_swap_col(mat.raw, col1, col2); + return mat; +} + +/*! + * @brief swap two matrix rows + * + * @param[in] mat matrix + * @param[in] row1 row1 + * @param[in] row2 row2 + * @returns matrix + */ +CGLM_INLINE +mat2s +glms_mat2_(swap_row)(mat2s mat, int row1, int row2) { + glm_mat2_swap_row(mat.raw, row1, row2); + return mat; +} + +/*! + * @brief helper for R (row vector) * M (matrix) * C (column vector) + * + * rmc stands for Row * Matrix * Column + * + * the result is scalar because R * M = Matrix1x2 (row vector), + * then Matrix1x2 * Vec2 (column vector) = Matrix1x1 (Scalar) + * + * @param[in] r row vector or matrix1x2 + * @param[in] m matrix2x2 + * @param[in] c column vector or matrix2x1 + * + * @return scalar value e.g. Matrix1x1 + */ +CGLM_INLINE +float +glms_mat2_(rmc)(vec2s r, mat2s m, vec2s c) { + return glm_mat2_rmc(r.raw, m.raw, c.raw); +} + +/*! + * @brief Create mat2 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @return constructed matrix from raw pointer + */ +CGLM_INLINE +mat2s +glms_mat2_(make)(const float * __restrict src) { + mat2s r; + glm_mat2_make(src, r.raw); + return r; +} + +#endif /* cglms_mat2_h */ diff --git a/cglm/include/cglm/struct/mat2x3.h b/cglm/include/cglm/struct/mat2x3.h new file mode 100644 index 0000000..3798893 --- /dev/null +++ b/cglm/include/cglm/struct/mat2x3.h @@ -0,0 +1,127 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLMS_MAT2X3_ZERO_INIT + GLMS_MAT2X3_ZERO + + Functions: + CGLM_INLINE mat2x3s glms_mat2x3_zero(void); + CGLM_INLINE mat2x3s glms_mat2x3_make(const float * __restrict src); + CGLM_INLINE mat2s glms_mat2x3_mul(mat2x3s m1, mat3x2s m2); + CGLM_INLINE vec3s glms_mat2x3_mulv(mat2x3s m, vec2s v); + CGLM_INLINE mat3x2s glms_mat2x3_transpose(mat2x3s m); + CGLM_INLINE mat2x3s glms_mat2x3_scale(mat2x3s m, float s); + */ + +#ifndef cglms_mat2x3_h +#define cglms_mat2x3_h + +#include "../common.h" +#include "../types-struct.h" +#include "../mat2x3.h" + +/* api definition */ +#define glms_mat2x3_(NAME) CGLM_STRUCTAPI(mat2x3, NAME) + +#define GLMS_MAT2X3_ZERO_INIT {GLM_MAT2X3_ZERO_INIT} + +/* for C only */ +#define GLMS_MAT2X3_ZERO ((mat2x3s)GLMS_MAT2X3_ZERO_INIT) + +/*! + * @brief make given matrix zero. + * + * @param[in, out] mat matrix + */ +CGLM_INLINE +mat2x3s +glms_mat2x3_(zero)(void) { + mat2x3s r; + glm_mat2x3_zero(r.raw); + return r; +} + +/*! + * @brief Create mat2x3 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @return constructed matrix from raw pointer + */ +CGLM_INLINE +mat2x3s +glms_mat2x3_(make)(const float * __restrict src) { + mat2x3s r; + glm_mat2x3_make(src, r.raw); + return r; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * @code + * r = glms_mat2x3_mul(mat2x3s, mat3x2s); + * @endcode + * + * @param[in] m1 left matrix (mat2x3s) + * @param[in] m2 right matrix (mat3x2s) + * @returns destination matrix (mat2s) + */ +CGLM_INLINE +mat3s +glms_mat2x3_(mul)(mat2x3s m1, mat3x2s m2) { + mat3s r; + glm_mat2x3_mul(m1.raw, m2.raw, r.raw); + return r; +} + +/*! + * @brief multiply matrix with column vector and store in dest vector + * + * @param[in] m matrix (left) + * @param[in] v vector (right, column vector) + * @returns destination vector (vec3s) + */ +CGLM_INLINE +vec3s +glms_mat2x3_(mulv)(mat2x3s m, vec2s v) { + vec3s r; + glm_mat2x3_mulv(m.raw, v.raw, r.raw); + return r; +} + +/*! + * @brief transpose matrix and store in dest + * + * @param[in] m matrix + * @param[out] dest result + */ +CGLM_INLINE +mat3x2s +glms_mat2x3_(transpose)(mat2x3s m) { + mat3x2s r; + glm_mat2x3_transpose(m.raw, r.raw); + return r; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +mat2x3s +glms_mat2x3_(scale)(mat2x3s m, float s) { + glm_mat2x3_scale(m.raw, s); + return m; +} + +#endif /* cglms_mat2x3_h */ diff --git a/cglm/include/cglm/struct/mat2x4.h b/cglm/include/cglm/struct/mat2x4.h new file mode 100644 index 0000000..afeedc1 --- /dev/null +++ b/cglm/include/cglm/struct/mat2x4.h @@ -0,0 +1,127 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLMS_MAT2X4_ZERO_INIT + GLMS_MAT2X4_ZERO + + Functions: + CGLM_INLINE mat2x4s glms_mat2x4_zero(void); + CGLM_INLINE mat2x4s glms_mat2x4_make(const float * __restrict src); + CGLM_INLINE mat2s glms_mat2x4_mul(mat2x4s m1, mat4x2s m2); + CGLM_INLINE vec4s glms_mat2x4_mulv(mat2x4s m, vec2s v); + CGLM_INLINE mat4x2s glms_mat2x4_transpose(mat2x4s m); + CGLM_INLINE mat2x4s glms_mat2x4_scale(mat2x4s m, float s); + */ + +#ifndef cglms_mat2x4_h +#define cglms_mat2x4_h + +#include "../common.h" +#include "../types-struct.h" +#include "../mat2x4.h" + +/* api definition */ +#define glms_mat2x4_(NAME) CGLM_STRUCTAPI(mat2x4, NAME) + +#define GLMS_MAT2X4_ZERO_INIT {GLM_MAT2X4_ZERO_INIT} + +/* for C only */ +#define GLMS_MAT2X4_ZERO ((mat2x4s)GLMS_MAT2X4_ZERO_INIT) + +/*! + * @brief make given matrix zero. + * + * @param[in, out] mat matrix + */ +CGLM_INLINE +mat2x4s +glms_mat2x4_(zero)(void) { + mat2x4s r; + glm_mat2x4_zero(r.raw); + return r; +} + +/*! + * @brief Create mat2x4 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @return constructed matrix from raw pointer + */ +CGLM_INLINE +mat2x4s +glms_mat2x4_(make)(const float * __restrict src) { + mat2x4s r; + glm_mat2x4_make(src, r.raw); + return r; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * @code + * r = glms_mat2x4_mul(mat2x4s, mat4x2s); + * @endcode + * + * @param[in] m1 left matrix (mat2x4s) + * @param[in] m2 right matrix (mat4x2s) + * @returns destination matrix (mat2s) + */ +CGLM_INLINE +mat4s +glms_mat2x4_(mul)(mat2x4s m1, mat4x2s m2) { + mat4s r; + glm_mat2x4_mul(m1.raw, m2.raw, r.raw); + return r; +} + +/*! + * @brief multiply matrix with column vector and store in dest column vector + * + * @param[in] m matrix (left) + * @param[in] v vector (right, column vector) + * @param[out] dest result vector + */ +CGLM_INLINE +vec4s +glms_mat2x4_(mulv)(mat2x4s m, vec2s v) { + vec4s r; + glm_mat2x4_mulv(m.raw, v.raw, r.raw); + return r; +} + +/*! + * @brief transpose matrix and store in dest + * + * @param[in] m matrix + * @param[out] dest result + */ +CGLM_INLINE +mat4x2s +glms_mat2x4_(transpose)(mat2x4s m) { + mat4x2s r; + glm_mat2x4_transpose(m.raw, r.raw); + return r; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +mat2x4s +glms_mat2x4_(scale)(mat2x4s m, float s) { + glm_mat2x4_scale(m.raw, s); + return m; +} + +#endif /* cglms_mat2x4_h */ diff --git a/cglm/include/cglm/struct/mat3.h b/cglm/include/cglm/struct/mat3.h new file mode 100644 index 0000000..4bfdde8 --- /dev/null +++ b/cglm/include/cglm/struct/mat3.h @@ -0,0 +1,303 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLMS_MAT3_IDENTITY_INIT + GLMS_MAT3_ZERO_INIT + GLMS_MAT3_IDENTITY + GLMS_MAT3_ZERO + + Functions: + CGLM_INLINE mat3s glms_mat3_copy(mat3s mat); + CGLM_INLINE mat3s glms_mat3_identity(void); + CGLM_INLINE void glms_mat3_identity_array(mat3s * __restrict mat, size_t count); + CGLM_INLINE mat3s glms_mat3_zero(void); + CGLM_INLINE mat3s glms_mat3_mul(mat3s m1, mat3s m2); + CGLM_INLINE ma3s glms_mat3_transpose(mat3s m); + CGLM_INLINE vec3s glms_mat3_mulv(mat3s m, vec3s v); + CGLM_INLINE float glms_mat3_trace(mat3s m); + CGLM_INLINE versor glms_mat3_quat(mat3s m); + CGLM_INLINE mat3s glms_mat3_scale(mat3s m, float s); + CGLM_INLINE float glms_mat3_det(mat3s mat); + CGLM_INLINE mat3s glms_mat3_inv(mat3s mat); + CGLM_INLINE mat3s glms_mat3_swap_col(mat3s mat, int col1, int col2); + CGLM_INLINE mat3s glms_mat3_swap_row(mat3s mat, int row1, int row2); + CGLM_INLINE float glms_mat3_rmc(vec3s r, mat3s m, vec3s c); + CGLM_INLINE mat3s glms_mat3_make(const float * __restrict src); + */ + +#ifndef cglms_mat3s_h +#define cglms_mat3s_h + +#include "../common.h" +#include "../types-struct.h" +#include "../mat3.h" +#include "vec3.h" + +/* api definition */ +#define glms_mat3_(NAME) CGLM_STRUCTAPI(mat3, NAME) + +#define GLMS_MAT3_IDENTITY_INIT {GLM_MAT3_IDENTITY_INIT} +#define GLMS_MAT3_ZERO_INIT {GLM_MAT3_ZERO_INIT} + +/* for C only */ +#define GLMS_MAT3_IDENTITY ((mat3s)GLMS_MAT3_IDENTITY_INIT) +#define GLMS_MAT3_ZERO ((mat3s)GLMS_MAT3_ZERO_INIT) + +/*! + * @brief copy all members of [mat] to [dest] + * + * @param[in] mat source + * @returns destination + */ +CGLM_INLINE +mat3s +glms_mat3_(copy)(mat3s mat) { + mat3s r; + glm_mat3_copy(mat.raw, r.raw); + return r; +} + +/*! + * @brief make given matrix identity. It is identical with below, + * but it is more easy to do that with this func especially for members + * e.g. glm_mat3_identity(aStruct->aMatrix); + * + * @code + * glm_mat3_copy(GLM_MAT3_IDENTITY, mat); // C only + * + * // or + * mat3 mat = GLM_MAT3_IDENTITY_INIT; + * @endcode + * + * @returns destination + */ +CGLM_INLINE +mat3s +glms_mat3_(identity)(void) { + mat3s r; + glm_mat3_identity(r.raw); + return r; +} + +/*! + * @brief make given matrix array's each element identity matrix + * + * @param[in, out] mat matrix array (must be aligned (16/32) + * if alignment is not disabled) + * + * @param[in] count count of matrices + */ +CGLM_INLINE +void +glms_mat3_(identity_array)(mat3s * __restrict mat, size_t count) { + CGLM_ALIGN_MAT mat3s t = GLMS_MAT3_IDENTITY_INIT; + size_t i; + + for (i = 0; i < count; i++) { + glm_mat3_copy(t.raw, mat[i].raw); + } +} + +/*! + * @brief make given matrix zero. + * + * @returns matrix + */ +CGLM_INLINE +mat3s +glms_mat3_(zero)(void) { + mat3s r; + glm_mat3_zero(r.raw); + return r; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * m1, m2 and dest matrices can be same matrix, it is possible to write this: + * + * @code + * mat3 m = GLM_MAT3_IDENTITY_INIT; + * r = glms_mat3_mul(m, m); + * @endcode + * + * @param[in] m1 left matrix + * @param[in] m2 right matrix + * @returns destination matrix + */ +CGLM_INLINE +mat3s +glms_mat3_(mul)(mat3s m1, mat3s m2) { + mat3s r; + glm_mat3_mul(m1.raw, m2.raw, r.raw); + return r; +} + +/*! + * @brief transpose mat3 and store result in same matrix + * + * @param[in, out] m source and dest + */ +CGLM_INLINE +mat3s +glms_mat3_(transpose)(mat3s m) { + glm_mat3_transpose(m.raw); + return m; +} + +/*! + * @brief multiply mat3 with vec3 (column vector) and store in dest vector + * + * @param[in] m mat3 (left) + * @param[in] v vec3 (right, column vector) + * @returns vec3 (result, column vector) + */ +CGLM_INLINE +vec3s +glms_mat3_(mulv)(mat3s m, vec3s v) { + vec3s r; + glm_mat3_mulv(m.raw, v.raw, r.raw); + return r; +} + +/*! + * @brief trace of matrix + * + * sum of the elements on the main diagonal from upper left to the lower right + * + * @param[in] m matrix + */ +CGLM_INLINE +float +glms_mat3_(trace)(mat3s m) { + return glm_mat3_trace(m.raw); +} + +/*! + * @brief convert mat3 to quaternion + * + * @param[in] m rotation matrix + * @returns destination quaternion + */ +CGLM_INLINE +versors +glms_mat3_(quat)(mat3s m) { + versors r; + glm_mat3_quat(m.raw, r.raw); + return r; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in] m matrix + * @param[in] s scalar + * @returns scaled matrix + */ +CGLM_INLINE +mat3s +glms_mat3_(scale)(mat3s m, float s) { + glm_mat3_scale(m.raw, s); + return m; +} + +/*! + * @brief mat3 determinant + * + * @param[in] mat matrix + * + * @return determinant + */ +CGLM_INLINE +float +glms_mat3_(det)(mat3s mat) { + return glm_mat3_det(mat.raw); +} + +/*! + * @brief inverse mat3 and store in dest + * + * @param[in] mat matrix + * @returns inverse matrix + */ +CGLM_INLINE +mat3s +glms_mat3_(inv)(mat3s mat) { + mat3s r; + glm_mat3_inv(mat.raw, r.raw); + return r; +} + +/*! + * @brief swap two matrix columns + * + * @param[in] mat matrix + * @param[in] col1 col1 + * @param[in] col2 col2 + * @returns matrix + */ +CGLM_INLINE +mat3s +glms_mat3_(swap_col)(mat3s mat, int col1, int col2) { + glm_mat3_swap_col(mat.raw, col1, col2); + return mat; +} + +/*! + * @brief swap two matrix rows + * + * @param[in] mat matrix + * @param[in] row1 row1 + * @param[in] row2 row2 + * @returns matrix + */ +CGLM_INLINE +mat3s +glms_mat3_(swap_row)(mat3s mat, int row1, int row2) { + glm_mat3_swap_row(mat.raw, row1, row2); + return mat; +} + +/*! + * @brief helper for R (row vector) * M (matrix) * C (column vector) + * + * rmc stands for Row * Matrix * Column + * + * the result is scalar because R * M = Matrix1x3 (row vector), + * then Matrix1x3 * Vec3 (column vector) = Matrix1x1 (Scalar) + * + * @param[in] r row vector or matrix1x3 + * @param[in] m matrix3x3 + * @param[in] c column vector or matrix3x1 + * + * @return scalar value e.g. Matrix1x1 + */ +CGLM_INLINE +float +glms_mat3_(rmc)(vec3s r, mat3s m, vec3s c) { + return glm_mat3_rmc(r.raw, m.raw, c.raw); +} + +/*! + * @brief Create mat3 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @return constructed matrix from raw pointer + */ +CGLM_INLINE +mat3s +glms_mat3_(make)(const float * __restrict src) { + mat3s r; + glm_mat3_make(src, r.raw); + return r; +} + +#endif /* cglms_mat3s_h */ diff --git a/cglm/include/cglm/struct/mat3x2.h b/cglm/include/cglm/struct/mat3x2.h new file mode 100644 index 0000000..a95d5e1 --- /dev/null +++ b/cglm/include/cglm/struct/mat3x2.h @@ -0,0 +1,127 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLMS_MAT3X2_ZERO_INIT + GLMS_MAT3X2_ZERO + + Functions: + CGLM_INLINE mat3x2s glms_mat3x2_zero(void); + CGLM_INLINE mat3x2s glms_mat3x2_make(const float * __restrict src); + CGLM_INLINE mat2s glms_mat3x2_mul(mat3x2s m1, mat2x3s m2); + CGLM_INLINE vec2s glms_mat3x2_mulv(mat3x2s m, vec3s v); + CGLM_INLINE mat2x3s glms_mat3x2_transpose(mat3x2s m); + CGLM_INLINE mat3x2s glms_mat3x2_scale(mat3x2s m, float s); + */ + +#ifndef cglms_mat3x2_h +#define cglms_mat3x2_h + +#include "../common.h" +#include "../types-struct.h" +#include "../mat3x2.h" + +/* api definition */ +#define glms_mat3x2_(NAME) CGLM_STRUCTAPI(mat3x2, NAME) + +#define GLMS_MAT3X2_ZERO_INIT {GLM_MAT3X2_ZERO_INIT} + +/* for C only */ +#define GLMS_MAT3X2_ZERO ((mat3x2s)GLMS_MAT3X2_ZERO_INIT) + +/*! + * @brief make given matrix zero. + * + * @param[in, out] mat matrix + */ +CGLM_INLINE +mat3x2s +glms_mat3x2_(zero)(void) { + mat3x2s r; + glm_mat3x2_zero(r.raw); + return r; +} + +/*! + * @brief Create mat3x2 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @return constructed matrix from raw pointer + */ +CGLM_INLINE +mat3x2s +glms_mat3x2_(make)(const float * __restrict src) { + mat3x2s r; + glm_mat3x2_make(src, r.raw); + return r; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * @code + * r = glms_mat3x2_mul(mat3x2s, mat2x3s); + * @endcode + * + * @param[in] m1 left matrix (mat3x2s) + * @param[in] m2 right matrix (mat2x3s) + * @returns destination matrix (mat3s) + */ +CGLM_INLINE +mat2s +glms_mat3x2_(mul)(mat3x2s m1, mat2x3s m2) { + mat2s r; + glm_mat3x2_mul(m1.raw, m2.raw, r.raw); + return r; +} + +/*! + * @brief multiply matrix with column vector and store in dest vector + * + * @param[in] m matrix (left) + * @param[in] v vector (right, column vector) + * @param[out] dest result vector + */ +CGLM_INLINE +vec2s +glms_mat3x2_(mulv)(mat3x2s m, vec3s v) { + vec2s r; + glm_mat3x2_mulv(m.raw, v.raw, r.raw); + return r; +} + +/*! + * @brief transpose matrix and store in dest + * + * @param[in] m matrix + * @param[out] dest result + */ +CGLM_INLINE +mat2x3s +glms_mat3x2_(transpose)(mat3x2s m) { + mat2x3s r; + glm_mat3x2_transpose(m.raw, r.raw); + return r; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +mat3x2s +glms_mat3x2_(scale)(mat3x2s m, float s) { + glm_mat3x2_scale(m.raw, s); + return m; +} + +#endif /* cglms_mat3x2_h */ diff --git a/cglm/include/cglm/struct/mat3x4.h b/cglm/include/cglm/struct/mat3x4.h new file mode 100644 index 0000000..ee688f0 --- /dev/null +++ b/cglm/include/cglm/struct/mat3x4.h @@ -0,0 +1,127 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLMS_MAT3X4_ZERO_INIT + GLMS_MAT3X4_ZERO + + Functions: + CGLM_INLINE mat3x4s glms_mat3x4_zero(void); + CGLM_INLINE mat3x4s glms_mat3x4_make(const float * __restrict src); + CGLM_INLINE mat4s glms_mat3x4_mul(mat3x4s m1, mat4x3s m2); + CGLM_INLINE vec4s glms_mat3x4_mulv(mat3x4s m, vec3s v); + CGLM_INLINE mat4x3s glms_mat3x4_transpose(mat3x4s m); + CGLM_INLINE mat3x4s glms_mat3x4_scale(mat3x4s m, float s); + */ + +#ifndef cglms_mat3x4_h +#define cglms_mat3x4_h + +#include "../common.h" +#include "../types-struct.h" +#include "../mat3x4.h" + +/* api definition */ +#define glms_mat3x4_(NAME) CGLM_STRUCTAPI(mat3x4, NAME) + +#define GLMS_MAT3X4_ZERO_INIT {GLM_MAT3X4_ZERO_INIT} + +/* for C only */ +#define GLMS_MAT3X4_ZERO ((mat3x4s)GLMS_MAT3X4_ZERO_INIT) + +/*! + * @brief make given matrix zero. + * + * @param[in, out] mat matrix + */ +CGLM_INLINE +mat3x4s +glms_mat3x4_(zero)(void) { + mat3x4s r; + glm_mat3x4_zero(r.raw); + return r; +} + +/*! + * @brief Create mat3x4 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @return constructed matrix from raw pointer + */ +CGLM_INLINE +mat3x4s +glms_mat3x4_(make)(const float * __restrict src) { + mat3x4s r; + glm_mat3x4_make(src, r.raw); + return r; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * @code + * r = glms_mat3x4_mul(mat3x4s, mat4x3s); + * @endcode + * + * @param[in] m1 left matrix (mat3x4s) + * @param[in] m2 right matrix (mat4x3s) + * @returns destination matrix (mat4s) + */ +CGLM_INLINE +mat4s +glms_mat3x4_(mul)(mat3x4s m1, mat4x3s m2) { + mat4s r; + glm_mat3x4_mul(m1.raw, m2.raw, r.raw); + return r; +} + +/*! + * @brief multiply matrix with column vector and store in dest column vector + * + * @param[in] m matrix (left) + * @param[in] v vector (right, column vector) + * @returns destination vector (vec4s) + */ +CGLM_INLINE +vec4s +glms_mat3x4_(mulv)(mat3x4s m, vec3s v) { + vec4s r; + glm_mat3x4_mulv(m.raw, v.raw, r.raw); + return r; +} + +/*! + * @brief transpose matrix and store in dest + * + * @param[in] m matrix + * @param[out] dest result + */ +CGLM_INLINE +mat4x3s +glms_mat3x4_(transpose)(mat3x4s m) { + mat4x3s r; + glm_mat3x4_transpose(m.raw, r.raw); + return r; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +mat3x4s +glms_mat3x4_(scale)(mat3x4s m, float s) { + glm_mat3x4_scale(m.raw, s); + return m; +} + +#endif /* cglms_mat3x4_h */ diff --git a/cglm/include/cglm/struct/mat4.h b/cglm/include/cglm/struct/mat4.h new file mode 100644 index 0000000..35b184b --- /dev/null +++ b/cglm/include/cglm/struct/mat4.h @@ -0,0 +1,477 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/*! + * Most of functions in this header are optimized manually with SIMD + * if available. You dont need to call/incude SIMD headers manually + */ + +/* + Macros: + GLMS_MAT4_IDENTITY_INIT + GLMS_MAT4_ZERO_INIT + GLMS_MAT4_IDENTITY + GLMS_MAT4_ZERO + + Functions: + CGLM_INLINE mat4s glms_mat4_ucopy(mat4s mat); + CGLM_INLINE mat4s glms_mat4_copy(mat4s mat); + CGLM_INLINE mat4s glms_mat4_identity(void); + CGLM_INLINE void glms_mat4_identity_array(mat4s * __restrict mat, size_t count); + CGLM_INLINE mat4s glms_mat4_zero(void); + CGLM_INLINE mat3s glms_mat4_pick3(mat4s mat); + CGLM_INLINE mat3s glms_mat4_pick3t(mat4s mat); + CGLM_INLINE mat4s glms_mat4_ins3(mat3s mat, mat4s dest); + CGLM_INLINE mat4s glms_mat4_mul(mat4s m1, mat4s m2); + CGLM_INLINE mat4s glms_mat4_mulN(mat4s * __restrict matrices[], uint32_t len); + CGLM_INLINE vec4s glms_mat4_mulv(mat4s m, vec4s v); + CGLM_INLINE float glms_mat4_trace(mat4s m); + CGLM_INLINE float glms_mat4_trace3(mat4s m); + CGLM_INLINE versors glms_mat4_quat(mat4s m); + CGLM_INLINE vec3s glms_mat4_mulv3(mat4s m, vec3s v, float last); + CGLM_INLINE mat4s glms_mat4_transpose(mat4s m); + CGLM_INLINE mat4s glms_mat4_scale_p(mat4s m, float s); + CGLM_INLINE mat4s glms_mat4_scale(mat4s m, float s); + CGLM_INLINE float glms_mat4_det(mat4s mat); + CGLM_INLINE mat4s glms_mat4_inv(mat4s mat); + CGLM_INLINE mat4s glms_mat4_inv_fast(mat4s mat); + CGLM_INLINE mat4s glms_mat4_swap_col(mat4s mat, int col1, int col2); + CGLM_INLINE mat4s glms_mat4_swap_row(mat4s mat, int row1, int row2); + CGLM_INLINE float glms_mat4_rmc(vec4s r, mat4s m, vec4s c); + CGLM_INLINE mat4s glms_mat4_make(const float * __restrict src); + */ + +#ifndef cglms_mat4s_h +#define cglms_mat4s_h + +#include "../common.h" +#include "../types-struct.h" +#include "../mat4.h" +#include "vec4.h" +#include "vec3.h" + +/* api definition */ +#define glms_mat4_(NAME) CGLM_STRUCTAPI(mat4, NAME) + +#define GLMS_MAT4_IDENTITY_INIT {GLM_MAT4_IDENTITY_INIT} +#define GLMS_MAT4_ZERO_INIT {GLM_MAT4_ZERO_INIT} + +/* for C only */ +#define GLMS_MAT4_IDENTITY ((mat4s)GLMS_MAT4_IDENTITY_INIT) +#define GLMS_MAT4_ZERO ((mat4s)GLMS_MAT4_ZERO_INIT) + +/*! + * @brief copy all members of [mat] to [dest] + * + * matrix may not be aligned, u stands for unaligned, this may be useful when + * copying a matrix from external source e.g. asset importer... + * + * @param[in] mat source + * @returns destination + */ +CGLM_INLINE +mat4s +glms_mat4_(ucopy)(mat4s mat) { + mat4s r; + glm_mat4_ucopy(mat.raw, r.raw); + return r; +} + +/*! + * @brief copy all members of [mat] to [dest] + * + * @param[in] mat source + * @returns destination + */ +CGLM_INLINE +mat4s +glms_mat4_(copy)(mat4s mat) { + mat4s r; + glm_mat4_copy(mat.raw, r.raw); + return r; +} + +/*! + * @brief make given matrix identity. It is identical with below, + * but it is more easy to do that with this func especially for members + * e.g. glm_mat4_identity(aStruct->aMatrix); + * + * @code + * glm_mat4_copy(GLM_MAT4_IDENTITY, mat); // C only + * + * // or + * mat4 mat = GLM_MAT4_IDENTITY_INIT; + * @endcode + * + * @returns destination + */ +CGLM_INLINE +mat4s +glms_mat4_(identity)(void) { + mat4s r; + glm_mat4_identity(r.raw); + return r; +} + +/*! + * @brief make given matrix array's each element identity matrix + * + * @param[in, out] mat matrix array (must be aligned (16/32) + * if alignment is not disabled) + * + * @param[in] count count of matrices + */ +CGLM_INLINE +void +glms_mat4_(identity_array)(mat4s * __restrict mat, size_t count) { + CGLM_ALIGN_MAT mat4s t = GLMS_MAT4_IDENTITY_INIT; + size_t i; + + for (i = 0; i < count; i++) { + glm_mat4_copy(t.raw, mat[i].raw); + } +} + +/*! + * @brief make given matrix zero. + * + * @returns matrix + */ +CGLM_INLINE +mat4s +glms_mat4_(zero)(void) { + mat4s r; + glm_mat4_zero(r.raw); + return r; +} + +/*! + * @brief copy upper-left of mat4 to mat3 + * + * @param[in] mat source + * @returns destination + */ +CGLM_INLINE +mat3s +glms_mat4_(pick3)(mat4s mat) { + mat3s r; + glm_mat4_pick3(mat.raw, r.raw); + return r; +} + +/*! + * @brief copy upper-left of mat4 to mat3 (transposed) + * + * the postfix t stands for transpose + * + * @param[in] mat source + * @returns destination + */ +CGLM_INLINE +mat3s +glms_mat4_(pick3t)(mat4s mat) { + mat3s r; + glm_mat4_pick3t(mat.raw, r.raw); + return r; +} + +/*! + * @brief copy mat3 to mat4's upper-left + * + * @param[in] mat source + * @param[in] dest destination + * @returns destination + */ +CGLM_INLINE +mat4s +glms_mat4_(ins3)(mat3s mat, mat4s dest) { + glm_mat4_ins3(mat.raw, dest.raw); + return dest; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * m1, m2 and dest matrices can be same matrix, it is possible to write this: + * + * @code + * mat4 m = GLM_MAT4_IDENTITY_INIT; + * r = glms_mat4_mul(m, m); + * @endcode + * + * @param[in] m1 left matrix + * @param[in] m2 right matrix + * @returns destination matrix + */ +CGLM_INLINE +mat4s +glms_mat4_(mul)(mat4s m1, mat4s m2) { + mat4s r; + glm_mat4_mul(m1.raw, m2.raw, r.raw); + return r; +} + +/*! + * @brief mupliply N mat4 matrices and store result in dest + * + * this function lets you multiply multiple (more than two or more...) matrices + *

multiplication will be done in loop, this may reduce instructions + * size but if len is too small then compiler may unroll whole loop, + * usage: + * @code + * mat m1, m2, m3, m4, res; + * + * res = glm_mat4_mulN((mat4 *[]){&m1, &m2, &m3, &m4}, 4); + * @endcode + * + * @warning matrices parameter is pointer array not mat4 array! + * + * @param[in] matrices mat4 * array + * @param[in] len matrices count + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_mat4_(mulN)(mat4s * __restrict matrices[], uint32_t len) { + CGLM_ALIGN_MAT mat4s r = GLMS_MAT4_IDENTITY_INIT; + size_t i; + + for (i = 0; i < len; i++) { + r = glms_mat4_(mul)(r, *matrices[i]); + } + + return r; +} + +/*! + * @brief multiply mat4 with vec4 (column vector) and store in dest vector + * + * @param[in] m mat4 (left) + * @param[in] v vec4 (right, column vector) + * @returns vec4 (result, column vector) + */ +CGLM_INLINE +vec4s +glms_mat4_(mulv)(mat4s m, vec4s v) { + vec4s r; + glm_mat4_mulv(m.raw, v.raw, r.raw); + return r; +} + +/*! + * @brief trace of matrix + * + * sum of the elements on the main diagonal from upper left to the lower right + * + * @param[in] m matrix + */ +CGLM_INLINE +float +glms_mat4_(trace)(mat4s m) { + return glm_mat4_trace(m.raw); +} + +/*! + * @brief trace of matrix (rotation part) + * + * sum of the elements on the main diagonal from upper left to the lower right + * + * @param[in] m matrix + */ +CGLM_INLINE +float +glms_mat4_(trace3)(mat4s m) { + return glm_mat4_trace3(m.raw); +} + +/*! + * @brief convert mat4's rotation part to quaternion + * + * @param[in] m affine matrix + * @returns destination quaternion + */ +CGLM_INLINE +versors +glms_mat4_(quat)(mat4s m) { + versors r; + glm_mat4_quat(m.raw, r.raw); + return r; +} + +/*! + * @brief multiply vector with mat4 + * + * @param[in] m mat4(affine transform) + * @param[in] v vec3 + * @param[in] last 4th item to make it vec4 + * @returns result vector (vec3) + */ +CGLM_INLINE +vec3s +glms_mat4_(mulv3)(mat4s m, vec3s v, float last) { + vec3s r; + glm_mat4_mulv3(m.raw, v.raw, last, r.raw); + return r; +} + +/*! + * @brief transpose mat4 and store result in same matrix + * + * @param[in] m source + * @returns result + */ +CGLM_INLINE +mat4s +glms_mat4_(transpose)(mat4s m) { + glm_mat4_transpose(m.raw); + return m; +} + +/*! + * @brief scale (multiply with scalar) matrix without simd optimization + * + * multiply matrix with scalar + * + * @param[in] m matrix + * @param[in] s scalar + * @returns matrix + */ +CGLM_INLINE +mat4s +glms_mat4_(scale_p)(mat4s m, float s) { + glm_mat4_scale_p(m.raw, s); + return m; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in] m matrix + * @param[in] s scalar + * @returns matrix + */ +CGLM_INLINE +mat4s +glms_mat4_(scale)(mat4s m, float s) { + glm_mat4_scale(m.raw, s); + return m; +} + +/*! + * @brief mat4 determinant + * + * @param[in] mat matrix + * + * @return determinant + */ +CGLM_INLINE +float +glms_mat4_(det)(mat4s mat) { + return glm_mat4_det(mat.raw); +} + +/*! + * @brief inverse mat4 and store in dest + * + * @param[in] mat matrix + * @returns inverse matrix + */ +CGLM_INLINE +mat4s +glms_mat4_(inv)(mat4s mat) { + mat4s r; + glm_mat4_inv(mat.raw, r.raw); + return r; +} + +/*! + * @brief inverse mat4 and store in dest + * + * this func uses reciprocal approximation without extra corrections + * e.g Newton-Raphson. this should work faster than normal, + * to get more precise use glm_mat4_inv version. + * + * NOTE: You will lose precision, glm_mat4_inv is more accurate + * + * @param[in] mat matrix + * @returns inverse matrix + */ +CGLM_INLINE +mat4s +glms_mat4_(inv_fast)(mat4s mat) { + mat4s r; + glm_mat4_inv_fast(mat.raw, r.raw); + return r; +} + +/*! + * @brief swap two matrix columns + * + * @param[in] mat matrix + * @param[in] col1 col1 + * @param[in] col2 col2 + * @returns matrix + */ +CGLM_INLINE +mat4s +glms_mat4_(swap_col)(mat4s mat, int col1, int col2) { + glm_mat4_swap_col(mat.raw, col1, col2); + return mat; +} + +/*! + * @brief swap two matrix rows + * + * @param[in] mat matrix + * @param[in] row1 row1 + * @param[in] row2 row2 + * @returns matrix + */ +CGLM_INLINE +mat4s +glms_mat4_(swap_row)(mat4s mat, int row1, int row2) { + glm_mat4_swap_row(mat.raw, row1, row2); + return mat; +} + +/*! + * @brief helper for R (row vector) * M (matrix) * C (column vector) + * + * rmc stands for Row * Matrix * Column + * + * the result is scalar because R * M = Matrix1x4 (row vector), + * then Matrix1x4 * Vec4 (column vector) = Matrix1x1 (Scalar) + * + * @param[in] r row vector or matrix1x4 + * @param[in] m matrix4x4 + * @param[in] c column vector or matrix4x1 + * + * @return scalar value e.g. B(s) + */ +CGLM_INLINE +float +glms_mat4_(rmc)(vec4s r, mat4s m, vec4s c) { + return glm_mat4_rmc(r.raw, m.raw, c.raw); +} + +/*! + * @brief Create mat4 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @return constructed matrix from raw pointer + */ +CGLM_INLINE +mat4s +glms_mat4_(make)(const float * __restrict src) { + mat4s r; + glm_mat4_make(src, r.raw); + return r; +} + +#endif /* cglms_mat4s_h */ diff --git a/cglm/include/cglm/struct/mat4x2.h b/cglm/include/cglm/struct/mat4x2.h new file mode 100644 index 0000000..17589bc --- /dev/null +++ b/cglm/include/cglm/struct/mat4x2.h @@ -0,0 +1,128 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLMS_MAT4X2_ZERO_INIT + GLMS_MAT4X2_ZERO + + Functions: + CGLM_INLINE mat4x2s glms_mat4x2_zero(void); + CGLM_INLINE mat4x2s glms_mat4x2_make(const float * __restrict src); + CGLM_INLINE mat2s glms_mat4x2_mul(mat4x2s m1, mat2x4s m2); + CGLM_INLINE vec2s glms_mat4x2_mulv(mat4x2s m, vec4s v); + CGLM_INLINE mat2x4s glms_mat4x2_transpose(mat4x2s m); + CGLM_INLINE mat4x2s glms_mat4x2_scale(mat4x2s m, float s); + */ + +#ifndef cglms_mat4x2_h +#define cglms_mat4x2_h + +#include "../common.h" +#include "../types-struct.h" +#include "../mat4x2.h" + +/* api definition */ +#define glms_mat4x2_(NAME) CGLM_STRUCTAPI(mat4x2, NAME) + +#define GLMS_MAT4X2_ZERO_INIT {GLM_MAT4X2_ZERO_INIT} + +/* for C only */ +#define GLMS_MAT4X2_ZERO ((mat4x2s)GLMS_MAT4X2_ZERO_INIT) + + +/*! + * @brief make given matrix zero. + * + * @param[in, out] mat matrix + */ +CGLM_INLINE +mat4x2s +glms_mat4x2_(zero)(void) { + mat4x2s r; + glm_mat4x2_zero(r.raw); + return r; +} + +/*! + * @brief Create mat4x2 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @return constructed matrix from raw pointer + */ +CGLM_INLINE +mat4x2s +glms_mat4x2_(make)(const float * __restrict src) { + mat4x2s r; + glm_mat4x2_make(src, r.raw); + return r; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * @code + * r = glms_mat4x2_mul(mat4x2s, mat2x4s); + * @endcode + * + * @param[in] m1 left matrix (mat4x2s) + * @param[in] m2 right matrix (mat2x4s) + * @returns destination matrix (mat2s) + */ +CGLM_INLINE +mat2s +glms_mat4x2_(mul)(mat4x2s m1, mat2x4s m2) { + mat2s r; + glm_mat4x2_mul(m1.raw, m2.raw, r.raw); + return r; +} + +/*! + * @brief multiply matrix with column vector and store in dest column vector + * + * @param[in] m matrix (left) + * @param[in] v vector (right, column vector) + * @returns destination vector (vec2s) + */ +CGLM_INLINE +vec2s +glms_mat4x2_(mulv)(mat4x2s m, vec4s v) { + vec2s r; + glm_mat4x2_mulv(m.raw, v.raw, r.raw); + return r; +} + +/*! + * @brief transpose matrix and store in dest + * + * @param[in] m matrix + * @param[out] dest result + */ +CGLM_INLINE +mat2x4s +glms_mat4x2_(transpose)(mat4x2s m) { + mat2x4s r; + glm_mat4x2_transpose(m.raw, r.raw); + return r; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +mat4x2s +glms_mat4x2_(scale)(mat4x2s m, float s) { + glm_mat4x2_scale(m.raw, s); + return m; +} + +#endif /* cglms_mat4x2_h */ diff --git a/cglm/include/cglm/struct/mat4x3.h b/cglm/include/cglm/struct/mat4x3.h new file mode 100644 index 0000000..37a68e1 --- /dev/null +++ b/cglm/include/cglm/struct/mat4x3.h @@ -0,0 +1,127 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLMS_MAT4X3_ZERO_INIT + GLMS_MAT4X3_ZERO + + Functions: + CGLM_INLINE mat4x3s glms_mat4x3_zero(void); + CGLM_INLINE mat4x3s glms_mat4x3_make(const float * __restrict src); + CGLM_INLINE mat3s glms_mat4x3_mul(mat4x3s m1, mat3x4s m2); + CGLM_INLINE vec3s glms_mat4x3_mulv(mat4x3s m, vec4s v); + CGLM_INLINE mat3x4s glms_mat4x3_transpose(mat4x3s m); + CGLM_INLINE mat4x3s glms_mat4x3_scale(mat4x3s m, float s); + */ + +#ifndef cglms_mat4x3_h +#define cglms_mat4x3_h + +#include "../common.h" +#include "../types-struct.h" +#include "../mat4x3.h" + +/* api definition */ +#define glms_mat4x3_(NAME) CGLM_STRUCTAPI(mat4x3, NAME) + +#define GLMS_MAT4X3_ZERO_INIT {GLM_MAT4X3_ZERO_INIT} + +/* for C only */ +#define GLMS_MAT4X3_ZERO ((mat4x3s)GLMS_MAT4X3_ZERO_INIT) + +/*! + * @brief make given matrix zero. + * + * @param[in, out] mat matrix + */ +CGLM_INLINE +mat4x3s +glms_mat4x3_(zero)(void) { + mat4x3s r; + glm_mat4x3_zero(r.raw); + return r; +} + +/*! + * @brief Create mat4x3 matrix from pointer + * + * @param[in] src pointer to an array of floats + * @return constructed matrix from raw pointer + */ +CGLM_INLINE +mat4x3s +glms_mat4x3_(make)(const float * __restrict src) { + mat4x3s r; + glm_mat4x3_make(src, r.raw); + return r; +} + +/*! + * @brief multiply m1 and m2 to dest + * + * @code + * r = glms_mat4x3_mul(mat4x3s, mat3x4s); + * @endcode + * + * @param[in] m1 left matrix (mat4x3s) + * @param[in] m2 right matrix (mat3x4s) + * @returns destination matrix (mat3s) + */ +CGLM_INLINE +mat3s +glms_mat4x3_(mul)(mat4x3s m1, mat3x4s m2) { + mat3s r; + glm_mat4x3_mul(m1.raw, m2.raw, r.raw); + return r; +} + +/*! + * @brief multiply matrix with column vector and store in dest vector + * + * @param[in] m matrix (left) + * @param[in] v vector (right, column vector) + * @returns destination vector (vec3s) + */ +CGLM_INLINE +vec3s +glms_mat4x3_(mulv)(mat4x3s m, vec4s v) { + vec3s r; + glm_mat4x3_mulv(m.raw, v.raw, r.raw); + return r; +} + +/*! + * @brief transpose matrix and store in dest + * + * @param[in] m matrix + * @param[out] dest result + */ +CGLM_INLINE +mat3x4s +glms_mat4x3_(transpose)(mat4x3s m) { + mat3x4s r; + glm_mat4x3_transpose(m.raw, r.raw); + return r; +} + +/*! + * @brief scale (multiply with scalar) matrix + * + * multiply matrix with scalar + * + * @param[in, out] m matrix + * @param[in] s scalar + */ +CGLM_INLINE +mat4x3s +glms_mat4x3_(scale)(mat4x3s m, float s) { + glm_mat4x3_scale(m.raw, s); + return m; +} + +#endif /* cglms_mat4x3_h */ diff --git a/cglm/include/cglm/struct/noise.h b/cglm/include/cglm/struct/noise.h new file mode 100644 index 0000000..3fd7d2e --- /dev/null +++ b/cglm/include/cglm/struct/noise.h @@ -0,0 +1,57 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglms_noises_h +#define cglms_noises_h + +#include "../common.h" +#include "../types-struct.h" +#include "../noise.h" +#include "vec4.h" + +/* + Functions: + CGLM_INLINE float glms_perlin_vec4(vec4s point); + */ + +/*! + * @brief Classic perlin noise + * + * @param[in] point 4D vector + * @returns perlin noise value + */ +CGLM_INLINE +float +glms_perlin_vec4(vec4s point) { + return glm_perlin_vec4(point.raw); +} + +/*! + * @brief Classic perlin noise + * + * @param[in] point 3D vector + * @returns perlin noise value + */ +CGLM_INLINE +float +glms_perlin_vec3(vec3s point) { + return glm_perlin_vec3(point.raw); +} + +/*! + * @brief Classic perlin noise + * + * @param[in] point 2D vector + * @returns perlin noise value + */ +CGLM_INLINE +float +glms_perlin_vec2(vec2s point) { + return glm_perlin_vec2(point.raw); +} + +#endif /* cglms_noises_h */ diff --git a/cglm/include/cglm/struct/plane.h b/cglm/include/cglm/struct/plane.h new file mode 100644 index 0000000..6a84ac7 --- /dev/null +++ b/cglm/include/cglm/struct/plane.h @@ -0,0 +1,40 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglms_planes_h +#define cglms_planes_h + +#include "../common.h" +#include "../types-struct.h" +#include "../plane.h" +#include "vec4.h" + +/* + Plane equation: Ax + By + Cz + D = 0; + + It stored in vec4 as [A, B, C, D]. (A, B, C) is normal and D is distance +*/ + +/* + Functions: + CGLM_INLINE vec4s glms_plane_normalize(vec4s plane); + */ + +/*! + * @brief normalizes a plane + * + * @param[in] plane plane to normalize + * @returns normalized plane + */ +CGLM_INLINE +vec4s +glms_plane_normalize(vec4s plane) { + glm_plane_normalize(plane.raw); + return plane; +} + +#endif /* cglms_planes_h */ diff --git a/cglm/include/cglm/struct/project.h b/cglm/include/cglm/struct/project.h new file mode 100644 index 0000000..8383c77 --- /dev/null +++ b/cglm/include/cglm/struct/project.h @@ -0,0 +1,162 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglms_projects_h +#define cglms_projects_h + +#include "../common.h" +#include "../types-struct.h" +#include "../project.h" +#include "vec3.h" +#include "vec4.h" +#include "mat4.h" + +#ifndef CGLM_CLIPSPACE_INCLUDE_ALL +# if CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_ZO_BIT +# include "clipspace/project_zo.h" +# elif CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_NO_BIT +# include "clipspace/project_no.h" +# endif +#else +# include "clipspace/project_zo.h" +# include "clipspace/project_no.h" +#endif + +/*! + * @brief maps the specified viewport coordinates into specified space [1] + * the matrix should contain projection matrix. + * + * if you don't have ( and don't want to have ) an inverse matrix then use + * glm_unproject version. You may use existing inverse of matrix in somewhere + * else, this is why glm_unprojecti exists to save save inversion cost + * + * [1] space: + * 1- if m = invProj: View Space + * 2- if m = invViewProj: World Space + * 3- if m = invMVP: Object Space + * + * You probably want to map the coordinates into object space + * so use invMVP as m + * + * Computing viewProj: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * glm_mat4_inv(viewProj, invMVP); + * + * @param[in] pos point/position in viewport coordinates + * @param[in] invMat matrix (see brief) + * @param[in] vp viewport as [x, y, width, height] + * @returns unprojected coordinates + */ +CGLM_INLINE +vec3s +glms_unprojecti(vec3s pos, mat4s invMat, vec4s vp) { + vec3s r; + glm_unprojecti(pos.raw, invMat.raw, vp.raw, r.raw); + return r; +} + +/*! + * @brief maps the specified viewport coordinates into specified space [1] + * the matrix should contain projection matrix. + * + * this is same as glm_unprojecti except this function get inverse matrix for + * you. + * + * [1] space: + * 1- if m = proj: View Space + * 2- if m = viewProj: World Space + * 3- if m = MVP: Object Space + * + * You probably want to map the coordinates into object space + * so use MVP as m + * + * Computing viewProj and MVP: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * + * or in struct api: + * MVP = mat4_mul(mat4_mul(proj, view), model) + * + * @param[in] pos point/position in viewport coordinates + * @param[in] m matrix (see brief) + * @param[in] vp viewport as [x, y, width, height] + * @returns unprojected coordinates + */ +CGLM_INLINE +vec3s +glms_unproject(vec3s pos, mat4s m, vec4s vp) { + vec3s r; + glm_unproject(pos.raw, m.raw, vp.raw, r.raw); + return r; +} + +/*! + * @brief map object coordinates to window coordinates + * + * Computing MVP: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * + * or in struct api: + * MVP = mat4_mul(mat4_mul(proj, view), model) + * + * @param[in] pos object coordinates + * @param[in] m MVP matrix + * @param[in] vp viewport as [x, y, width, height] + * @returns projected coordinates + */ +CGLM_INLINE +vec3s +glms_project(vec3s pos, mat4s m, vec4s vp) { + vec3s r; + glm_project(pos.raw, m.raw, vp.raw, r.raw); + return r; +} + +/*! + * @brief map object's z coordinate to window coordinates + * + * Computing MVP: + * glm_mat4_mul(proj, view, viewProj); + * glm_mat4_mul(viewProj, model, MVP); + * + * or in struct api: + * MVP = mat4_mul(mat4_mul(proj, view), model) + * + * @param[in] v object coordinates + * @param[in] m MVP matrix + * + * @returns projected z coordinate + */ +CGLM_INLINE +float +glms_project_z(vec3s v, mat4s m) { +#if CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_ZO_BIT + return glms_project_z_zo(v, m); +#elif CGLM_CONFIG_CLIP_CONTROL & CGLM_CLIP_CONTROL_NO_BIT + return glms_project_z_no(v, m); +#endif +} + +/*! + * @brief define a picking region + * + * @param[in] center center [x, y] of a picking region in window coordinates + * @param[in] size size [width, height] of the picking region in window coordinates + * @param[in] vp viewport as [x, y, width, height] + * @returns projected coordinates + */ +CGLM_INLINE +mat4s +glms_pickmatrix(vec2s center, vec2s size, vec4s vp) { + mat4s res; + glm_pickmatrix(center.raw, size.raw, vp.raw, res.raw); + return res; +} + +#endif /* cglms_projects_h */ diff --git a/cglm/include/cglm/struct/quat.h b/cglm/include/cglm/struct/quat.h new file mode 100644 index 0000000..d6789e4 --- /dev/null +++ b/cglm/include/cglm/struct/quat.h @@ -0,0 +1,601 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLMS_QUAT_IDENTITY_INIT + GLMS_QUAT_IDENTITY + + Functions: + CGLM_INLINE versors glms_quat_identity(void) + CGLM_INLINE void glms_quat_identity_array(versor *q, size_t count) + CGLM_INLINE versors glms_quat_init(float x, float y, float z, float w) + CGLM_INLINE versors glms_quatv(float angle, vec3s axis) + CGLM_INLINE versors glms_quat(float angle, float x, float y, float z) + CGLM_INLINE versors glms_quat_from_vecs(vec3s a, vec3s b) + CGLM_INLINE float glms_quat_norm(versors q) + CGLM_INLINE versors glms_quat_normalize(versors q) + CGLM_INLINE float glms_quat_dot(versors p, versors q) + CGLM_INLINE versors glms_quat_conjugate(versors q) + CGLM_INLINE versors glms_quat_inv(versors q) + CGLM_INLINE versors glms_quat_add(versors p, versors q) + CGLM_INLINE versors glms_quat_sub(versors p, versors q) + CGLM_INLINE vec3s glms_quat_imagn(versors q) + CGLM_INLINE float glms_quat_imaglen(versors q) + CGLM_INLINE float glms_quat_angle(versors q) + CGLM_INLINE vec3s glms_quat_axis(versors q) + CGLM_INLINE versors glms_quat_mul(versors p, versors q) + CGLM_INLINE mat4s glms_quat_mat4(versors q) + CGLM_INLINE mat4s glms_quat_mat4t(versors q) + CGLM_INLINE mat3s glms_quat_mat3(versors q) + CGLM_INLINE mat3s glms_quat_mat3t(versors q) + CGLM_INLINE versors glms_quat_lerp(versors from, versors to, float t) + CGLM_INLINE versors glms_quat_lerpc(versors from, versors to, float t) + CGLM_INLINE versors glms_quat_nlerp(versors from, versors to, float t) + CGLM_INLINE versors glms_quat_slerp(versors from, versors to, float t) + CGLM_INLINE versors glms_quat_slerp_longest(versors from, versors to, float t) + CGLM_INLINE mat4s. glms_quat_look(vec3s eye, versors ori) + CGLM_INLINE versors glms_quat_for(vec3s dir, vec3s fwd, vec3s up) + CGLM_INLINE versors glms_quat_forp(vec3s from, vec3s to, vec3s fwd, vec3s up) + CGLM_INLINE vec3s glms_quat_rotatev(versors q, vec3s v) + CGLM_INLINE mat4s glms_quat_rotate(mat4s m, versors q) + CGLM_INLINE mat4s glms_quat_rotate_at(mat4s m, versors q, vec3s pivot) + CGLM_INLINE mat4s glms_quat_rotate_atm(versors q, vec3s pivot) + CGLM_INLINE versors glms_quat_make(float * restrict src) + */ + +#ifndef cglms_quat_h +#define cglms_quat_h + +#include "../common.h" +#include "../types-struct.h" +#include "../plane.h" +#include "../quat.h" + +/* api definition */ +#define glms_quat_(NAME) CGLM_STRUCTAPI(quat, NAME) + +/* + * IMPORTANT: + * ---------------------------------------------------------------------------- + * cglm stores quat as [x, y, z, w] since v0.3.6 + * + * it was [w, x, y, z] before v0.3.6 it has been changed to [x, y, z, w] + * with v0.3.6 version. + * ---------------------------------------------------------------------------- + */ + +#define GLMS_QUAT_IDENTITY_INIT {GLM_QUAT_IDENTITY_INIT} +#define GLMS_QUAT_IDENTITY ((versors)GLMS_QUAT_IDENTITY_INIT) + +/*! + * @brief makes given quat to identity + * + * @returns identity quaternion + */ +CGLM_INLINE +versors +glms_quat_(identity)(void) { + versors dest; + glm_quat_identity(dest.raw); + return dest; +} + +/*! + * @brief make given quaternion array's each element identity quaternion + * + * @param[in, out] q quat array (must be aligned (16) + * if alignment is not disabled) + * + * @param[in] count count of quaternions + */ +CGLM_INLINE +void +glms_quat_(identity_array)(versors * __restrict q, size_t count) { + CGLM_ALIGN(16) versor v = GLM_QUAT_IDENTITY_INIT; + size_t i; + + for (i = 0; i < count; i++) { + glm_vec4_copy(v, q[i].raw); + } +} + +/*! + * @brief inits quaternion with raw values + * + * @param[in] x x + * @param[in] y y + * @param[in] z z + * @param[in] w w (real part) + * @returns quaternion + */ +CGLM_INLINE +versors +glms_quat_(init)(float x, float y, float z, float w) { + versors dest; + glm_quat_init(dest.raw, x, y, z, w); + return dest; +} + +/*! + * @brief creates NEW quaternion with axis vector + * + * @param[in] angle angle (radians) + * @param[in] axis axis + * @returns quaternion + */ +CGLM_INLINE +versors +glms_quatv(float angle, vec3s axis) { + versors dest; + glm_quatv(dest.raw, angle, axis.raw); + return dest; +} + +/*! + * @brief creates NEW quaternion with individual axis components + * + * @param[in] angle angle (radians) + * @param[in] x axis.x + * @param[in] y axis.y + * @param[in] z axis.z + * @returns quaternion + */ +CGLM_INLINE +versors +glms_quat(float angle, float x, float y, float z) { + versors dest; + glm_quat(dest.raw, angle, x, y, z); + return dest; +} + +/*! + * @brief compute quaternion rotating vector A to vector B + * + * @param[in] a vec3 (must have unit length) + * @param[in] b vec3 (must have unit length) + * @returns quaternion (of unit length) + */ +CGLM_INLINE +versors +glms_quat_(from_vecs)(vec3s a, vec3s b) { + versors dest; + glm_quat_from_vecs(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief returns norm (magnitude) of quaternion + * + * @param[in] q quaternion + */ +CGLM_INLINE +float +glms_quat_(norm)(versors q) { + return glm_quat_norm(q.raw); +} + +/*! + * @brief normalize quaternion + * + * @param[in] q quaternion + * @returns quaternion + */ +CGLM_INLINE +versors +glms_quat_(normalize)(versors q) { + versors dest; + glm_quat_normalize_to(q.raw, dest.raw); + return dest; +} + +/*! + * @brief dot product of two quaternion + * + * @param[in] p quaternion 1 + * @param[in] q quaternion 2 + * @returns dot product + */ +CGLM_INLINE +float +glms_quat_(dot)(versors p, versors q) { + return glm_quat_dot(p.raw, q.raw); +} + +/*! + * @brief conjugate of quaternion + * + * @param[in] q quaternion + * @returns conjugate + */ +CGLM_INLINE +versors +glms_quat_(conjugate)(versors q) { + versors dest; + glm_quat_conjugate(q.raw, dest.raw); + return dest; +} + +/*! + * @brief inverse of non-zero quaternion + * + * @param[in] q quaternion + * @returns inverse quaternion + */ +CGLM_INLINE +versors +glms_quat_(inv)(versors q) { + versors dest; + glm_quat_inv(q.raw, dest.raw); + return dest; +} + +/*! + * @brief add (componentwise) two quaternions and store result in dest + * + * @param[in] p quaternion 1 + * @param[in] q quaternion 2 + * @returns result quaternion + */ +CGLM_INLINE +versors +glms_quat_(add)(versors p, versors q) { + versors dest; + glm_quat_add(p.raw, q.raw, dest.raw); + return dest; +} + +/*! + * @brief subtract (componentwise) two quaternions and store result in dest + * + * @param[in] p quaternion 1 + * @param[in] q quaternion 2 + * @returns result quaternion + */ +CGLM_INLINE +versors +glms_quat_(sub)(versors p, versors q) { + versors dest; + glm_quat_sub(p.raw, q.raw, dest.raw); + return dest; +} + +/*! + * @brief returns normalized imaginary part of quaternion + * + * @param[in] q quaternion + */ +CGLM_INLINE +vec3s +glms_quat_(imagn)(versors q) { + vec3s dest; + glm_normalize_to(q.raw, dest.raw); + return dest; +} + +/*! + * @brief returns length of imaginary part of quaternion + * + * @param[in] q quaternion + */ +CGLM_INLINE +float +glms_quat_(imaglen)(versors q) { + return glm_quat_imaglen(q.raw); +} + +/*! + * @brief returns angle of quaternion + * + * @param[in] q quaternion + */ +CGLM_INLINE +float +glms_quat_(angle)(versors q) { + return glm_quat_angle(q.raw); +} + +/*! + * @brief axis of quaternion + * + * @param[in] q quaternion + * @returns axis of quaternion + */ +CGLM_INLINE +vec3s +glms_quat_(axis)(versors q) { + vec3s dest; + glm_quat_axis(q.raw, dest.raw); + return dest; +} + +/*! + * @brief multiplies two quaternion and stores result in dest + * this is also called Hamilton Product + * + * According to WikiPedia: + * The product of two rotation quaternions [clarification needed] will be + * equivalent to the rotation q followed by the rotation p + * + * @param[in] p quaternion 1 + * @param[in] q quaternion 2 + * @returns result quaternion + */ +CGLM_INLINE +versors +glms_quat_(mul)(versors p, versors q) { + versors dest; + glm_quat_mul(p.raw, q.raw, dest.raw); + return dest; +} + +/*! + * @brief convert quaternion to mat4 + * + * @param[in] q quaternion + * @returns result matrix + */ +CGLM_INLINE +mat4s +glms_quat_(mat4)(versors q) { + mat4s dest; + glm_quat_mat4(q.raw, dest.raw); + return dest; +} + +/*! + * @brief convert quaternion to mat4 (transposed) + * + * @param[in] q quaternion + * @returns result matrix as transposed + */ +CGLM_INLINE +mat4s +glms_quat_(mat4t)(versors q) { + mat4s dest; + glm_quat_mat4t(q.raw, dest.raw); + return dest; +} + +/*! + * @brief convert quaternion to mat3 + * + * @param[in] q quaternion + * @returns result matrix + */ +CGLM_INLINE +mat3s +glms_quat_(mat3)(versors q) { + mat3s dest; + glm_quat_mat3(q.raw, dest.raw); + return dest; +} + +/*! + * @brief convert quaternion to mat3 (transposed) + * + * @param[in] q quaternion + * @returns result matrix + */ +CGLM_INLINE +mat3s +glms_quat_(mat3t)(versors q) { + mat3s dest; + glm_quat_mat3t(q.raw, dest.raw); + return dest; +} + +/*! + * @brief interpolates between two quaternions + * using linear interpolation (LERP) + * + * @param[in] from from + * @param[in] to to + * @param[in] t interpolant (amount) + * @returns result quaternion + */ +CGLM_INLINE +versors +glms_quat_(lerp)(versors from, versors to, float t) { + versors dest; + glm_quat_lerp(from.raw, to.raw, t, dest.raw); + return dest; +} + +/*! + * @brief interpolates between two quaternions + * using linear interpolation (LERP) + * + * @param[in] from from + * @param[in] to to + * @param[in] t interpolant (amount) clamped between 0 and 1 + * @returns result quaternion + */ +CGLM_INLINE +versors +glms_quat_(lerpc)(versors from, versors to, float t) { + versors dest; + glm_quat_lerpc(from.raw, to.raw, t, dest.raw); + return dest; +} + +/*! + * @brief interpolates between two quaternions + * taking the shortest rotation path using + * normalized linear interpolation (NLERP) + * + * @param[in] from from + * @param[in] to to + * @param[in] t interpolant (amount) + * @returns result quaternion + */ +CGLM_INLINE +versors +glms_quat_(nlerp)(versors from, versors to, float t) { + versors dest; + glm_quat_nlerp(from.raw, to.raw, t, dest.raw); + return dest; +} + +/*! + * @brief interpolates between two quaternions + * using spherical linear interpolation (SLERP) + * + * @param[in] from from + * @param[in] to to + * @param[in] t amount + * @returns result quaternion + */ +CGLM_INLINE +versors +glms_quat_(slerp)(versors from, versors to, float t) { + versors dest; + glm_quat_slerp(from.raw, to.raw, t, dest.raw); + return dest; +} + +/*! + * @brief interpolates between two quaternions + * using spherical linear interpolation (SLERP) and always takes the longest path + * + * @param[in] from from + * @param[in] to to + * @param[in] t amount + * @returns result quaternion + */ +CGLM_INLINE +versors +glms_quat_(slerp_longest)(versors from, versors to, float t) { + versors dest; + glm_quat_slerp_longest(from.raw, to.raw, t, dest.raw); + return dest; +} + +/*! + * @brief creates view matrix using quaternion as camera orientation + * + * @param[in] eye eye + * @param[in] ori orientation in world space as quaternion + * @returns view matrix + */ +CGLM_INLINE +mat4s +glms_quat_(look)(vec3s eye, versors ori) { + mat4s dest; + glm_quat_look(eye.raw, ori.raw, dest.raw); + return dest; +} + +/*! + * @brief creates look rotation quaternion + * + * @param[in] dir direction to look + * @param[in] up up vector + * @returns destination quaternion + */ +CGLM_INLINE +versors +glms_quat_(for)(vec3s dir, vec3s up) { + versors dest; + glm_quat_for(dir.raw, up.raw, dest.raw); + return dest; +} + +/*! + * @brief creates look rotation quaternion using source and + * destination positions p suffix stands for position + * + * @param[in] from source point + * @param[in] to destination point + * @param[in] up up vector + * @returns destination quaternion + */ +CGLM_INLINE +versors +glms_quat_(forp)(vec3s from, vec3s to, vec3s up) { + versors dest; + glm_quat_forp(from.raw, to.raw, up.raw, dest.raw); + return dest; +} + +/*! + * @brief rotate vector using using quaternion + * + * @param[in] q quaternion + * @param[in] v vector to rotate + * @returns rotated vector + */ +CGLM_INLINE +vec3s +glms_quat_(rotatev)(versors q, vec3s v) { + vec3s dest; + glm_quat_rotatev(q.raw, v.raw, dest.raw); + return dest; +} + +/*! + * @brief rotate existing transform matrix using quaternion + * + * @param[in] m existing transform matrix + * @param[in] q quaternion + * @returns rotated matrix/transform + */ +CGLM_INLINE +mat4s +glms_quat_(rotate)(mat4s m, versors q) { + glm_quat_rotate(m.raw, q.raw, m.raw); + return m; +} + +/*! + * @brief rotate existing transform matrix using quaternion at pivot point + * + * @param[in, out] m existing transform matrix + * @param[in] q quaternion + * @returns pivot + */ +CGLM_INLINE +mat4s +glms_quat_(rotate_at)(mat4s m, versors q, vec3s pivot) { + glm_quat_rotate_at(m.raw, q.raw, pivot.raw); + return m; +} + +/*! + * @brief rotate NEW transform matrix using quaternion at pivot point + * + * this creates rotation matrix, it assumes you don't have a matrix + * + * this should work faster than glm_quat_rotate_at because it reduces + * one glm_translate. + * + * @param[in] q quaternion + * @returns pivot + */ +CGLM_INLINE +mat4s +glms_quat_(rotate_atm)(versors q, vec3s pivot) { + mat4s dest; + glm_quat_rotate_atm(dest.raw, q.raw, pivot.raw); + return dest; +} + +/*! + * @brief Create CGLM quaternion from pointer + * + * @param[in] src pointer to an array of floats + * @returns constructed quaternion from raw pointer + */ +CGLM_INLINE +versors +glms_quat_(make)(const float * __restrict src) { + versors dest; + glm_quat_make(src, dest.raw); + return dest; +} + +#endif /* cglms_quat_h */ diff --git a/cglm/include/cglm/struct/ray.h b/cglm/include/cglm/struct/ray.h new file mode 100644 index 0000000..10609b9 --- /dev/null +++ b/cglm/include/cglm/struct/ray.h @@ -0,0 +1,86 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglms_ray_h +#define cglms_ray_h + +#include "../common.h" +#include "../types-struct.h" +#include "../ray.h" + +/* api definition */ +#define glms_ray_(NAME) CGLM_STRUCTAPI(ray, NAME) + +/*! + * @brief Möller–Trumbore ray-triangle intersection algorithm + * + * @param[in] origin origin of ray + * @param[in] direction direction of ray + * @param[in] v0 first vertex of triangle + * @param[in] v1 second vertex of triangle + * @param[in] v2 third vertex of triangle + * @param[in, out] d distance to intersection + * @return whether there is intersection + */ +CGLM_INLINE +bool +glms_ray_(triangle)(vec3s origin, + vec3s direction, + vec3s v0, + vec3s v1, + vec3s v2, + float *d) { + return glm_ray_triangle(origin.raw, direction.raw, v0.raw, v1.raw, v2.raw, d); +} + +/*! + * @brief ray sphere intersection + * + * returns false if there is no intersection if true: + * + * - t1 > 0, t2 > 0: ray intersects the sphere at t1 and t2 both ahead of the origin + * - t1 < 0, t2 > 0: ray starts inside the sphere, exits at t2 + * - t1 < 0, t2 < 0: no intersection ahead of the ray ( returns false ) + * - the caller can check if the intersection points (t1 and t2) fall within a + * specific range (for example, tmin < t1, t2 < tmax) to determine if the + * intersections are within a desired segment of the ray + * + * @param[in] origin ray origin + * @param[out] dir normalized ray direction + * @param[in] s sphere [center.x, center.y, center.z, radii] + * @param[in] t1 near point1 (closer to origin) + * @param[in] t2 far point2 (farther from origin) + * + * @returns whether there is intersection + */ +CGLM_INLINE +bool +glms_ray_(sphere)(vec3s origin, + vec3s dir, + vec4s s, + float * __restrict t1, + float * __restrict t2) { + return glm_ray_sphere(origin.raw, dir.raw, s.raw, t1, t2); +} + +/*! + * @brief point using t by 𝐏(𝑡)=𝐀+𝑡𝐛 + * + * @param[in] orig origin of ray + * @param[in] dir direction of ray + * @param[in] t parameter + * @returns point point at t + */ +CGLM_INLINE +vec3s +glms_ray_(at)(vec3s orig, vec3s dir, float t) { + vec3s r; + glm_ray_at(orig.raw, dir.raw, t, r.raw); + return r; +} + +#endif /* cglms_ray_h */ diff --git a/cglm/include/cglm/struct/sphere.h b/cglm/include/cglm/struct/sphere.h new file mode 100644 index 0000000..9859c72 --- /dev/null +++ b/cglm/include/cglm/struct/sphere.h @@ -0,0 +1,93 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglms_spheres_h +#define cglms_spheres_h + +#include "../common.h" +#include "../types-struct.h" +#include "../sphere.h" +#include "mat4.h" + +/* + Sphere Representation in cglm: [center.x, center.y, center.z, radii] + + You could use this representation or you can convert it to vec4 before call + any function + */ + +/*! + * @brief helper for getting sphere radius + * + * @param[in] s sphere + * + * @return returns radii + */ +CGLM_INLINE +float +glms_sphere_radii(vec4s s) { + return glm_sphere_radii(s.raw); +} + +/*! + * @brief apply transform to sphere, it is just wrapper for glm_mat4_mulv3 + * + * @param[in] s sphere + * @param[in] m transform matrix + * @returns transformed sphere + */ +CGLM_INLINE +vec4s +glms_sphere_transform(vec4s s, mat4s m) { + vec4s r; + glm_sphere_transform(s.raw, m.raw, r.raw); + return r; +} + +/*! + * @brief merges two spheres and creates a new one + * + * two sphere must be in same space, for instance if one in world space then + * the other must be in world space too, not in local space. + * + * @param[in] s1 sphere 1 + * @param[in] s2 sphere 2 + * returns merged/extended sphere + */ +CGLM_INLINE +vec4s +glms_sphere_merge(vec4s s1, vec4s s2) { + vec4s r; + glm_sphere_merge(s1.raw, s2.raw, r.raw); + return r; +} + +/*! + * @brief check if two sphere intersects + * + * @param[in] s1 sphere + * @param[in] s2 other sphere + */ +CGLM_INLINE +bool +glms_sphere_sphere(vec4s s1, vec4s s2) { + return glm_sphere_sphere(s1.raw, s2.raw); +} + +/*! + * @brief check if sphere intersects with point + * + * @param[in] s sphere + * @param[in] point point + */ +CGLM_INLINE +bool +glms_sphere_point(vec4s s, vec3s point) { + return glm_sphere_point(s.raw, point.raw); +} + +#endif /* cglms_spheres_h */ diff --git a/cglm/include/cglm/struct/vec2-ext.h b/cglm/include/cglm/struct/vec2-ext.h new file mode 100644 index 0000000..246132f --- /dev/null +++ b/cglm/include/cglm/struct/vec2-ext.h @@ -0,0 +1,337 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/*! + * @brief SIMD like functions + */ + +/* + Functions: + CGLM_INLINE vec2s glms_vec2_fill(float val) + CGLM_INLINE bool glms_vec2_eq(vec2s v, float val) + CGLM_INLINE bool glms_vec2_eq_eps(vec2s v, float val) + CGLM_INLINE bool glms_vec2_eq_all(vec2s v) + CGLM_INLINE bool glms_vec2_eqv(vec2s a, vec2s b) + CGLM_INLINE bool glms_vec2_eqv_eps(vec2s a, vec2s b) + CGLM_INLINE float glms_vec2_max(vec2s v) + CGLM_INLINE float glms_vec2_min(vec2s v) + CGLM_INLINE bool glms_vec2_isnan(vec2s v) + CGLM_INLINE bool glms_vec2_isinf(vec2s v) + CGLM_INLINE bool glms_vec2_isvalid(vec2s v) + CGLM_INLINE vec2s glms_vec2_sign(vec2s v) + CGLM_INLINE vec2s glms_vec2_abs(vec2s v) + CGLM_INLINE vec2s glms_vec2_fract(vec2s v) + CGLM_INLINE vec2s glms_vec2_floor(vec2s v) + CGLM_INLINE vec2s glms_vec2_mods(vec2s v, float s) + CGLM_INLINE vec2s glms_vec2_steps(float edge, vec2s v) + CGLM_INLINE vec2s glms_vec2_stepr(vec2s edge, float v) + CGLM_INLINE vec2s glms_vec2_sqrt(vec2s v) + */ + +#ifndef cglms_vec2s_ext_h +#define cglms_vec2s_ext_h + +#include "../common.h" +#include "../types-struct.h" +#include "../util.h" +#include "../vec2-ext.h" + +/* api definition */ +#define glms_vec2_(NAME) CGLM_STRUCTAPI(vec2, NAME) + +/*! + * @brief fill a vector with specified value + * + * @param[in] val value + * @returns dest + */ +CGLM_INLINE +vec2s +glms_vec2_(fill)(float val) { + vec2s r; + glm_vec2_fill(r.raw, val); + return r; +} + +/*! + * @brief check if vector is equal to value (without epsilon) + * + * @param[in] v vector + * @param[in] val value + */ +CGLM_INLINE +bool +glms_vec2_(eq)(vec2s v, float val) { + return glm_vec2_eq(v.raw, val); +} + +/*! + * @brief check if vector is equal to value (with epsilon) + * + * @param[in] v vector + * @param[in] val value + */ +CGLM_INLINE +bool +glms_vec2_(eq_eps)(vec2s v, float val) { + return glm_vec2_eq_eps(v.raw, val); +} + +/*! + * @brief check if vector members are equal (without epsilon) + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glms_vec2_(eq_all)(vec2s v) { + return glm_vec2_eq_all(v.raw); +} + +/*! + * @brief check if vector is equal to another (without epsilon) + * + * @param[in] a vector + * @param[in] b vector + */ +CGLM_INLINE +bool +glms_vec2_(eqv)(vec2s a, vec2s b) { + return glm_vec2_eqv(a.raw, b.raw); +} + +/*! + * @brief check if vector is equal to another (with epsilon) + * + * @param[in] a vector + * @param[in] b vector + */ +CGLM_INLINE +bool +glms_vec2_(eqv_eps)(vec2s a, vec2s b) { + return glm_vec2_eqv_eps(a.raw, b.raw); +} + +/*! + * @brief max value of vector + * + * @param[in] v vector + */ +CGLM_INLINE +float +glms_vec2_(max)(vec2s v) { + return glm_vec2_max(v.raw); +} + +/*! + * @brief min value of vector + * + * @param[in] v vector + */ +CGLM_INLINE +float +glms_vec2_min(vec2s v) { + return glm_vec2_min(v.raw); +} + +/*! + * @brief check if one of items is NaN (not a number) + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glms_vec2_(isnan)(vec2s v) { + return glm_vec2_isnan(v.raw); +} + +/*! + * @brief check if one of items is INFINITY + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glms_vec2_(isinf)(vec2s v) { + return glm_vec2_isinf(v.raw); +} + +/*! + * @brief check if all items are valid number + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glms_vec2_isvalid(vec2s v) { + return glm_vec2_isvalid(v.raw); +} + +/*! + * @brief get sign of 32 bit float as +1, -1, 0 + * + * Important: It returns 0 for zero/NaN input + * + * @param v vector + * @returns sign vector + */ +CGLM_INLINE +vec2s +glms_vec2_(sign)(vec2s v) { + vec2s r; + glm_vec2_sign(v.raw, r.raw); + return r; +} + +/*! + * @brief fractional part of each vector item + * + * @param v vector + * @returns abs vector + */ +CGLM_INLINE +vec2s +glms_vec2_(abs)(vec2s v) { + vec2s r; + glm_vec2_abs(v.raw, r.raw); + return r; +} + +/*! + * @brief fractional part of each vector item + * + * @param[in] v vector + * @returns destination vector + */ +CGLM_INLINE +vec2s +glms_vec2_(fract)(vec2s v) { + vec2s r; + glm_vec2_fract(v.raw, r.raw); + return r; +} + +/*! + * @brief floor of each vector item + * + * @param[in] v vector + * @returns destination vector + */ +CGLM_INLINE +vec2s +glms_vec2_(floor)(vec2s v) { + vec2s r; + glm_vec2_floor(v.raw, r.raw); + return r; +} + +/*! + * @brief mod of each vector item by scalar + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec2s +glms_vec2_(mods)(vec2s v, float s) { + vec2s r; + glm_vec2_mods(v.raw, s, r.raw); + return r; +} + +/*! + * @brief threshold each vector item with scalar + * condition is: (x[i] < edge) ? 0.0 : 1.0 + * + * @param[in] edge threshold + * @param[in] x vector to test against threshold + * @returns destination + */ +CGLM_INLINE +vec2s +glms_vec2_(steps)(float edge, vec2s x) { + vec2s r; + glm_vec2_steps(edge, x.raw, r.raw); + return r; +} + +/*! + * @brief threshold a value with *vector* as the threshold + * condition is: (x < edge[i]) ? 0.0 : 1.0 + * + * @param[in] edge threshold vector + * @param[in] x value to test against threshold + * @returns destination + */ +CGLM_INLINE +vec2s +glms_vec2_(stepr)(vec2s edge, float x) { + vec2s r; + glm_vec2_stepr(edge.raw, x, r.raw); + return r; +} + +/*! + * @brief square root of each vector item + * + * @param[in] v vector + * @returns destination vector + */ +CGLM_INLINE +vec2s +glms_vec2_(sqrt)(vec2s v) { + vec2s r; + glm_vec2_sqrt(v.raw, r.raw); + return r; +} + +/*! + * @brief treat vectors as complex numbers and multiply them as such. + * + * @param[in] a left number + * @param[in] b right number + * @param[out] dest destination number + */ +CGLM_INLINE +vec2s +glms_vec2_(complex_mul)(vec2s a, vec2s b, vec2s dest) { + glm_vec2_complex_mul(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief treat vectors as complex numbers and divide them as such. + * + * @param[in] a left number (numerator) + * @param[in] b right number (denominator) + * @param[out] dest destination number + */ +CGLM_INLINE +vec2s +glms_vec2_(complex_div)(vec2s a, vec2s b, vec2s dest) { + glm_vec2_complex_div(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief treat the vector as a complex number and conjugate it as such. + * + * @param[in] a the number + * @param[out] dest destination number + */ +CGLM_INLINE +vec2s +glms_vec2_(complex_conjugate)(vec2s a, vec2s dest) { + glm_vec2_complex_conjugate(a.raw, dest.raw); + return dest; +} + +#endif /* cglms_vec2s_ext_h */ diff --git a/cglm/include/cglm/struct/vec2.h b/cglm/include/cglm/struct/vec2.h new file mode 100644 index 0000000..40ed659 --- /dev/null +++ b/cglm/include/cglm/struct/vec2.h @@ -0,0 +1,747 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLMS_VEC2_ONE_INIT + GLMS_VEC2_ZERO_INIT + GLMS_VEC2_ONE + GLMS_VEC2_ZERO + + Functions: + CGLM_INLINE vec2s glms_vec2(vec3s v3) + CGLM_INLINE void glms_vec2_pack(vec2s dst[], vec2 src[], size_t len) + CGLM_INLINE void glms_vec2_unpack(vec2 dst[], vec2s src[], size_t len) + CGLM_INLINE vec2s glms_vec2_zero(void) + CGLM_INLINE vec2s glms_vec2_one(void) + CGLM_INLINE float glms_vec2_dot(vec2s a, vec2s b) + CGLM_INLINE float glms_vec2_cross(vec2s a, vec2s b) + CGLM_INLINE float glms_vec2_norm2(vec2s v) + CGLM_INLINE float glms_vec2_norm(vec2s v) + CGLM_INLINE vec2s glms_vec2_add(vec2s a, vec2s b) + CGLM_INLINE vec2s glms_vec2_adds(vec2s a, float s) + CGLM_INLINE vec2s glms_vec2_sub(vec2s a, vec2s b) + CGLM_INLINE vec2s glms_vec2_subs(vec2s a, float s) + CGLM_INLINE vec2s glms_vec2_mul(vec2s a, vec2s b) + CGLM_INLINE vec2s glms_vec2_scale(vec2s v, float s) + CGLM_INLINE vec2s glms_vec2_scale_as(vec2s v, float s) + CGLM_INLINE vec2s glms_vec2_div(vec2s a, vec2s b) + CGLM_INLINE vec2s glms_vec2_divs(vec2s a, float s) + CGLM_INLINE vec2s glms_vec2_addadd(vec2s a, vec2s b, vec2s dest) + CGLM_INLINE vec2s glms_vec2_subadd(vec2s a, vec2s b, vec2s dest) + CGLM_INLINE vec2s glms_vec2_muladd(vec2s a, vec2s b, vec2s dest) + CGLM_INLINE vec2s glms_vec2_muladds(vec2s a, float s, vec2s dest) + CGLM_INLINE vec2s glms_vec2_maxadd(vec2s a, vec2s b, vec2s dest) + CGLM_INLINE vec2s glms_vec2_minadd(vec2s a, vec2s b, vec2s dest) + CGLM_INLINE vec2s glms_vec2_subsub(vec2s a, vec2s b, vec2s dest) + CGLM_INLINE vec2s glms_vec2_addsub(vec2s a, vec2s b, vec2s dest) + CGLM_INLINE vec2s glms_vec2_mulsub(vec2s a, vec2s b, vec2s dest) + CGLM_INLINE vec2s glms_vec2_mulsubs(vec2s a, float s, vec2s dest) + CGLM_INLINE vec2s glms_vec2_maxsub(vec2s a, vec2s b, vec2s dest) + CGLM_INLINE vec2s glms_vec2_minsub(vec2s a, vec2s b, vec2s dest) + CGLM_INLINE vec2s glms_vec2_negate(vec2s v) + CGLM_INLINE vec2s glms_vec2_normalize(vec2s v) + CGLM_INLINE vec2s glms_vec2_rotate(vec2s v, float angle, vec2s axis) + CGLM_INLINE vec2s glms_vec2_center(vec2s a, vec2s b) + CGLM_INLINE float glms_vec2_distance(vec2s a, vec2s b) + CGLM_INLINE float glms_vec2_distance2(vec2s a, vec2s b) + CGLM_INLINE vec2s glms_vec2_maxv(vec2s a, vec2s b) + CGLM_INLINE vec2s glms_vec2_minv(vec2s a, vec2s b) + CGLM_INLINE vec2s glms_vec2_clamp(vec2s v, float minVal, float maxVal) + CGLM_INLINE vec2s glms_vec2_lerp(vec2s from, vec2s to, float t) + CGLM_INLINE vec2s glms_vec2_step(vec2s edge, vec2s x) + CGLM_INLINE vec2s glms_vec2_make(float * restrict src) + CGLM_INLINE vec2s glms_vec2_reflect(vec2s v, vec2s n) + CGLM_INLINE bool glms_vec2_refract(vec2s v, vec2s n, float eta, vec2s *dest) + */ + +#ifndef cglms_vec2s_h +#define cglms_vec2s_h + +#include "../common.h" +#include "../types-struct.h" +#include "../util.h" +#include "../vec2.h" +#include "vec2-ext.h" + +#define GLMS_VEC2_ONE_INIT {GLM_VEC2_ONE_INIT} +#define GLMS_VEC2_ZERO_INIT {GLM_VEC2_ZERO_INIT} + +#define GLMS_VEC2_ONE ((vec2s)GLMS_VEC2_ONE_INIT) +#define GLMS_VEC2_ZERO ((vec2s)GLMS_VEC2_ZERO_INIT) + +/*! + * @brief init vec2 using vec2 + * + * @param[in] v3 vector3 + * @returns destination + */ +CGLM_INLINE +vec2s +glms_vec2(vec3s v3) { + vec2s r; + glm_vec2(v3.raw, r.raw); + return r; +} + +/*! + * @brief pack an array of vec2 into an array of vec2s + * + * @param[out] dst array of vec2 + * @param[in] src array of vec2s + * @param[in] len number of elements + */ +CGLM_INLINE +void +glms_vec2_(pack)(vec2s dst[], vec2 src[], size_t len) { + size_t i; + + for (i = 0; i < len; i++) { + glm_vec2_copy(src[i], dst[i].raw); + } +} + +/*! + * @brief unpack an array of vec2s into an array of vec2 + * + * @param[out] dst array of vec2s + * @param[in] src array of vec2 + * @param[in] len number of elements + */ +CGLM_INLINE +void +glms_vec2_(unpack)(vec2 dst[], vec2s src[], size_t len) { + size_t i; + + for (i = 0; i < len; i++) { + glm_vec2_copy(src[i].raw, dst[i]); + } +} + +/*! + * @brief make vector zero + * + * @returns zero vector + */ +CGLM_INLINE +vec2s +glms_vec2_(zero)(void) { + vec2s r; + glm_vec2_zero(r.raw); + return r; +} + +/*! + * @brief make vector one + * + * @returns one vector + */ +CGLM_INLINE +vec2s +glms_vec2_(one)(void) { + vec2s r; + glm_vec2_one(r.raw); + return r; +} + +/*! + * @brief vec2 dot product + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return dot product + */ +CGLM_INLINE +float +glms_vec2_(dot)(vec2s a, vec2s b) { + return glm_vec2_dot(a.raw, b.raw); +} + +/*! + * @brief vec2 cross product + * + * REF: http://allenchou.net/2013/07/cross-product-of-2d-vectors/ + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return Z component of cross product + */ +CGLM_INLINE +float +glms_vec2_(cross)(vec2s a, vec2s b) { + return glm_vec2_cross(a.raw, b.raw); +} + +/*! + * @brief norm * norm (magnitude) of vec + * + * we can use this func instead of calling norm * norm, because it would call + * sqrtf function twice but with this func we can avoid func call, maybe this is + * not good name for this func + * + * @param[in] v vector + * + * @return norm * norm + */ +CGLM_INLINE +float +glms_vec2_(norm2)(vec2s v) { + return glm_vec2_norm2(v.raw); +} + +/*! + * @brief norm (magnitude) of vec2 + * + * @param[in] v vector + * + * @return norm + */ +CGLM_INLINE +float +glms_vec2_(norm)(vec2s v) { + return glm_vec2_norm(v.raw); +} + +/*! + * @brief add a vector to b vector store result in dest + * + * @param[in] a vector1 + * @param[in] b vector2 + * @returns destination vector + */ +CGLM_INLINE +vec2s +glms_vec2_(add)(vec2s a, vec2s b) { + vec2s r; + glm_vec2_add(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief add scalar to v vector store result in dest (d = v + s) + * + * @param[in] a vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec2s +glms_vec2_(adds)(vec2s a, float s) { + vec2s r; + glm_vec2_adds(a.raw, s, r.raw); + return r; +} + +/*! + * @brief subtract b vector from a vector store result in dest + * + * @param[in] a vector1 + * @param[in] b vector2 + * @returns destination vector + */ +CGLM_INLINE +vec2s +glms_vec2_(sub)(vec2s a, vec2s b) { + vec2s r; + glm_vec2_sub(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief subtract scalar from v vector store result in dest (d = v - s) + * + * @param[in] a vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec2s +glms_vec2_(subs)(vec2s a, float s) { + vec2s r; + glm_vec2_subs(a.raw, s, r.raw); + return r; +} + +/*! + * @brief multiply two vectors (component-wise multiplication) + * + * @param a vector1 + * @param b vector2 + * @returns result = (a[0] * b[0], a[1] * b[1]) + */ +CGLM_INLINE +vec2s +glms_vec2_(mul)(vec2s a, vec2s b) { + vec2s r; + glm_vec2_mul(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief multiply/scale vec2 vector with scalar: result = v * s + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec2s +glms_vec2_(scale)(vec2s v, float s) { + vec2s r; + glm_vec2_scale(v.raw, s, r.raw); + return r; +} + +/*! + * @brief make vec2 vector scale as specified: result = unit(v) * s + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec2s +glms_vec2_(scale_as)(vec2s v, float s) { + vec2s r; + glm_vec2_scale_as(v.raw, s, r.raw); + return r; +} + +/*! + * @brief div vector with another component-wise division: d = a / b + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns result = (a[0]/b[0], a[1]/b[1]) + */ +CGLM_INLINE +vec2s +glms_vec2_(div)(vec2s a, vec2s b) { + vec2s r; + glm_vec2_div(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief div vector with scalar: d = v / s + * + * @param[in] a vector + * @param[in] s scalar + * @returns result = (a[0]/s, a[1]/s) + */ +CGLM_INLINE +vec2s +glms_vec2_(divs)(vec2s a, float s) { + vec2s r; + glm_vec2_divs(a.raw, s, r.raw); + return r; +} + +/*! + * @brief add two vectors and add result to sum + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest += (a + b) + */ +CGLM_INLINE +vec2s +glms_vec2_(addadd)(vec2s a, vec2s b, vec2s dest) { + glm_vec2_addadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief sub two vectors and add result to dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest += (a + b) + */ +CGLM_INLINE +vec2s +glms_vec2_(subadd)(vec2s a, vec2s b, vec2s dest) { + glm_vec2_subadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief mul two vectors and add result to dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest += (a * b) + */ +CGLM_INLINE +vec2s +glms_vec2_(muladd)(vec2s a, vec2s b, vec2s dest) { + glm_vec2_muladd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief mul vector with scalar and add result to sum + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @returns dest += (a * b) + */ +CGLM_INLINE +vec2s +glms_vec2_(muladds)(vec2s a, float s, vec2s dest) { + glm_vec2_muladds(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief add max of two vectors to result/dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest += max(a, b) + */ +CGLM_INLINE +vec2s +glms_vec2_(maxadd)(vec2s a, vec2s b, vec2s dest) { + glm_vec2_maxadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief add min of two vectors to result/dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest += min(a, b) + */ +CGLM_INLINE +vec2s +glms_vec2_(minadd)(vec2s a, vec2s b, vec2s dest) { + glm_vec2_minadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief sub two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest -= (a - b) + */ +CGLM_INLINE +vec2s +glms_vec2_(subsub)(vec2s a, vec2s b, vec2s dest) { + glm_vec2_subsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief add two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest -= (a + b) + */ +CGLM_INLINE +vec2s +glms_vec2_(addsub)(vec2s a, vec2s b, vec2s dest) { + glm_vec2_addsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief mul two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest -= (a * b) + */ +CGLM_INLINE +vec2s +glms_vec2_(mulsub)(vec2s a, vec2s b, vec2s dest) { + glm_vec2_mulsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief mul vector with scalar and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @returns dest -= (a * b) + */ +CGLM_INLINE +vec2s +glms_vec2_(mulsubs)(vec2s a, float s, vec2s dest) { + glm_vec2_mulsubs(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief sub max of two vectors to dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest -= max(a, b) + */ +CGLM_INLINE +vec2s +glms_vec2_(maxsub)(vec2s a, vec2s b, vec2s dest) { + glm_vec2_maxsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief sub min of two vectors to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest -= min(a, b) + */ +CGLM_INLINE +vec2s +glms_vec2_(minsub)(vec2s a, vec2s b, vec2s dest) { + glm_vec2_minsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief negate vector components + * + * @param[in] v vector + * @returns negated vector + */ +CGLM_INLINE +vec2s +glms_vec2_(negate)(vec2s v) { + glm_vec2_negate(v.raw); + return v; +} + +/*! + * @brief normalize vec2 and store result in same vec + * + * @param[in] v vector + * @returns normalized vector + */ +CGLM_INLINE +vec2s +glms_vec2_(normalize)(vec2s v) { + glm_vec2_normalize(v.raw); + return v; +} + +/*! + * @brief rotate vec2 by angle using Rodrigues' rotation formula + * + * @param[in] v vector + * @param[in] angle angle by radians + * @returns rotated vector + */ +CGLM_INLINE +vec2s +glms_vec2_(rotate)(vec2s v, float angle) { + vec2s r; + glm_vec2_rotate(v.raw, angle, r.raw); + return r; +} + +/** + * @brief find center point of two vector + * + * @param[in] a vector1 + * @param[in] b vector2 + * @returns center point + */ +CGLM_INLINE +vec2s +glms_vec2_(center)(vec2s a, vec2s b) { + vec2s r; + glm_vec2_center(a.raw, b.raw, r.raw); + return r; +} + +/** + * @brief distance between two vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @return distance + */ +CGLM_INLINE +float +glms_vec2_(distance)(vec2s a, vec2s b) { + return glm_vec2_distance(a.raw, b.raw); +} + +/** + * @brief squared distance between two vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @return squared distance (distance * distance) + */ +CGLM_INLINE +float +glms_vec2_(distance2)(vec2s a, vec2s b) { + return glm_vec2_distance2(a.raw, b.raw); +} + +/*! + * @brief max values of vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @returns destination + */ +CGLM_INLINE +vec2s +glms_vec2_(maxv)(vec2s a, vec2s b) { + vec2s r; + glm_vec2_maxv(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief min values of vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @returns destination + */ +CGLM_INLINE +vec2s +glms_vec2_(minv)(vec2s a, vec2s b) { + vec2s r; + glm_vec2_minv(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief clamp vector's individual members between min and max values + * + * @param[in] v vector + * @param[in] minVal minimum value + * @param[in] maxVal maximum value + * @returns clamped vector + */ +CGLM_INLINE +vec2s +glms_vec2_(clamp)(vec2s v, float minVal, float maxVal) { + glm_vec2_clamp(v.raw, minVal, maxVal); + return v; +} + +/*! + * @brief linear interpolation between two vectors + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) + * @returns destination + */ +CGLM_INLINE +vec2s +glms_vec2_(lerp)(vec2s from, vec2s to, float t) { + vec2s r; + glm_vec2_lerp(from.raw, to.raw, t, r.raw); + return r; +} + +/*! + * @brief threshold function + * + * @param[in] edge threshold + * @param[in] x value to test against threshold + * @returns destination + */ +CGLM_INLINE +vec2s +glms_vec2_(step)(vec2s edge, vec2s x) { + vec2s r; + glm_vec2_step(edge.raw, x.raw, r.raw); + return r; +} + +/*! + * @brief Create two dimensional vector from pointer + * + * @param[in] src pointer to an array of floats + * @returns constructed 2D vector from raw pointer + */ +CGLM_INLINE +vec2s +glms_vec2_(make)(const float * __restrict src) { + vec2s dest; + glm_vec2_make(src, dest.raw); + return dest; +} + +/*! + * @brief reflection vector using an incident ray and a surface normal + * + * @param[in] I incident vector + * @param[in] N normalized normal vector + * @returns reflection result + */ +CGLM_INLINE +vec2s +glms_vec2_(reflect)(vec2s v, vec2s n) { + vec2s dest; + glm_vec2_reflect(v.raw, n.raw, dest.raw); + return dest; +} + +/*! + * @brief computes refraction vector for an incident vector and a surface normal. + * + * calculates the refraction vector based on Snell's law. If total internal reflection + * occurs (angle too great given eta), dest is set to zero and returns false. + * Otherwise, computes refraction vector, stores it in dest, and returns true. + * + * @param[in] v normalized incident vector + * @param[in] n normalized normal vector + * @param[in] eta ratio of indices of refraction (incident/transmitted) + * @param[out] dest refraction vector if refraction occurs; zero vector otherwise + * + * @returns true if refraction occurs; false if total internal reflection occurs. + */ +CGLM_INLINE +bool +glms_vec2_(refract)(vec2s v, vec2s n, float eta, vec2s * __restrict dest) { + return glm_vec2_refract(v.raw, n.raw, eta, dest->raw); +} + +#endif /* cglms_vec2s_h */ diff --git a/cglm/include/cglm/struct/vec3-ext.h b/cglm/include/cglm/struct/vec3-ext.h new file mode 100644 index 0000000..6cd8ca0 --- /dev/null +++ b/cglm/include/cglm/struct/vec3-ext.h @@ -0,0 +1,325 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/*! + * @brief SIMD like functions + */ + +/* + Functions: + CGLM_INLINE vec3s glms_vec3_broadcast(float val); + CGLM_INLINE vec3s glms_vec3_fill(float val); + CGLM_INLINE bool glms_vec3_eq(vec3s v, float val); + CGLM_INLINE bool glms_vec3_eq_eps(vec3s v, float val); + CGLM_INLINE bool glms_vec3_eq_all(vec3s v); + CGLM_INLINE bool glms_vec3_eqv(vec3s a, vec3s b); + CGLM_INLINE bool glms_vec3_eqv_eps(vec3s a, vec3s b); + CGLM_INLINE float glms_vec3_max(vec3s v); + CGLM_INLINE float glms_vec3_min(vec3s v); + CGLM_INLINE bool glms_vec3_isnan(vec3s v); + CGLM_INLINE bool glms_vec3_isinf(vec3s v); + CGLM_INLINE bool glms_vec3_isvalid(vec3s v); + CGLM_INLINE vec3s glms_vec3_sign(vec3s v); + CGLM_INLINE vec3s glms_vec3_abs(vec3s v); + CGLM_INLINE vec3s glms_vec3_fract(vec3s v); + CGLM_INLINE vec3s glms_vec3_floor(vec3s v); + CGLM_INLINE vec3s glms_vec3_mods(vec3s v, float s); + CGLM_INLINE vec3s glms_vec3_steps(float edge, vec3s v); + CGLM_INLINE vec3s glms_vec3_stepr(vec3s edge, float v); + CGLM_INLINE float glms_vec3_hadd(vec3s v); + CGLM_INLINE vec3s glms_vec3_sqrt(vec3s v); + */ + +#ifndef cglms_vec3s_ext_h +#define cglms_vec3s_ext_h + +#include "../common.h" +#include "../types-struct.h" +#include "../util.h" +#include "../vec3-ext.h" + +/* api definition */ +#define glms_vec3_(NAME) CGLM_STRUCTAPI(vec3, NAME) + +/*! + * @brief fill a vector with specified value + * + * @param[in] val value + * @returns dest + */ +CGLM_INLINE +vec3s +glms_vec3_(broadcast)(float val) { + vec3s r; + glm_vec3_broadcast(val, r.raw); + return r; +} + +/*! + * @brief fill a vector with specified value + * + * @param[in] val value + * @returns dest + */ +CGLM_INLINE +vec3s +glms_vec3_(fill)(float val) { + vec3s r; + glm_vec3_fill(r.raw, val); + return r; +} + +/*! + * @brief check if vector is equal to value (without epsilon) + * + * @param[in] v vector + * @param[in] val value + */ +CGLM_INLINE +bool +glms_vec3_(eq)(vec3s v, float val) { + return glm_vec3_eq(v.raw, val); +} + +/*! + * @brief check if vector is equal to value (with epsilon) + * + * @param[in] v vector + * @param[in] val value + */ +CGLM_INLINE +bool +glms_vec3_(eq_eps)(vec3s v, float val) { + return glm_vec3_eq_eps(v.raw, val); +} + +/*! + * @brief check if vector members are equal (without epsilon) + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glms_vec3_(eq_all)(vec3s v) { + return glm_vec3_eq_all(v.raw); +} + +/*! + * @brief check if vector is equal to another (without epsilon) + * + * @param[in] a vector + * @param[in] b vector + */ +CGLM_INLINE +bool +glms_vec3_(eqv)(vec3s a, vec3s b) { + return glm_vec3_eqv(a.raw, b.raw); +} + +/*! + * @brief check if vector is equal to another (with epsilon) + * + * @param[in] a vector + * @param[in] b vector + */ +CGLM_INLINE +bool +glms_vec3_(eqv_eps)(vec3s a, vec3s b) { + return glm_vec3_eqv_eps(a.raw, b.raw); +} + +/*! + * @brief max value of vector + * + * @param[in] v vector + */ +CGLM_INLINE +float +glms_vec3_(max)(vec3s v) { + return glm_vec3_max(v.raw); +} + +/*! + * @brief min value of vector + * + * @param[in] v vector + */ +CGLM_INLINE +float +glms_vec3_(min)(vec3s v) { + return glm_vec3_min(v.raw); +} + +/*! + * @brief check if one of items is NaN (not a number) + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glms_vec3_(isnan)(vec3s v) { + return glm_vec3_isnan(v.raw); +} + +/*! + * @brief check if one of items is INFINITY + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glms_vec3_(isinf)(vec3s v) { + return glm_vec3_isinf(v.raw); +} + +/*! + * @brief check if all items are valid number + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glms_vec3_(isvalid)(vec3s v) { + return glm_vec3_isvalid(v.raw); +} + +/*! + * @brief get sign of 32 bit float as +1, -1, 0 + * + * Important: It returns 0 for zero/NaN input + * + * @param v vector + * @returns sign vector + */ +CGLM_INLINE +vec3s +glms_vec3_(sign)(vec3s v) { + vec3s r; + glm_vec3_sign(v.raw, r.raw); + return r; +} + +/*! + * @brief absolute value of each vector item + * + * @param[in] v vector + * @return destination vector + */ +CGLM_INLINE +vec3s +glms_vec3_(abs)(vec3s v) { + vec3s r; + glm_vec3_abs(v.raw, r.raw); + return r; +} + +/*! + * @brief fractional part of each vector item + * + * @param[in] v vector + * @return dest destination vector + */ +CGLM_INLINE +vec3s +glms_vec3_(fract)(vec3s v) { + vec3s r; + glm_vec3_fract(v.raw, r.raw); + return r; +} + +/*! + * @brief floor of each vector item + * + * @param[in] v vector + * @return dest destination vector + */ +CGLM_INLINE +vec3s +glms_vec3_(floor)(vec3s v) { + vec3s r; + glm_vec3_floor(v.raw, r.raw); + return r; +} + +/*! + * @brief mod of each vector item by scalar + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec3s +glms_vec3_(mods)(vec3s v, float s) { + vec3s r; + glm_vec3_mods(v.raw, s, r.raw); + return r; +} + +/*! + * @brief threshold each vector item with scalar + * condition is: (x[i] < edge) ? 0.0 : 1.0 + * + * @param[in] edge threshold + * @param[in] x vector to test against threshold + * @returns destination + */ +CGLM_INLINE +vec3s +glms_vec3_(steps)(float edge, vec3s x) { + vec3s r; + glm_vec3_steps(edge, x.raw, r.raw); + return r; +} + +/*! + * @brief threshold a value with *vector* as the threshold + * condition is: (x < edge[i]) ? 0.0 : 1.0 + * + * @param[in] edge threshold vector + * @param[in] x value to test against threshold + * @returns destination + */ +CGLM_INLINE +vec3s +glms_vec3_(stepr)(vec3s edge, float x) { + vec3s r; + glm_vec3_stepr(edge.raw, x, r.raw); + return r; +} + +/*! + * @brief vector reduction by summation + * @warning could overflow + * + * @param[in] v vector + * @return sum of all vector's elements + */ +CGLM_INLINE +float +glms_vec3_(hadd)(vec3s v) { + return glm_vec3_hadd(v.raw); +} + +/*! + * @brief square root of each vector item + * + * @param[in] v vector + * @returns destination vector + */ +CGLM_INLINE +vec3s +glms_vec3_(sqrt)(vec3s v) { + vec3s r; + glm_vec3_sqrt(v.raw, r.raw); + return r; +} + +#endif /* cglms_vec3s_ext_h */ diff --git a/cglm/include/cglm/struct/vec3.h b/cglm/include/cglm/struct/vec3.h new file mode 100644 index 0000000..a1d901e --- /dev/null +++ b/cglm/include/cglm/struct/vec3.h @@ -0,0 +1,1132 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLMS_VEC3_ONE_INIT + GLMS_VEC3_ZERO_INIT + GLMS_VEC3_ONE + GLMS_VEC3_ZERO + GLMS_YUP + GLMS_ZUP + GLMS_XUP + + Functions: + CGLM_INLINE vec3s glms_vec3(vec4s v4); + CGLM_INLINE void glms_vec3_pack(vec3s dst[], vec3 src[], size_t len); + CGLM_INLINE void glms_vec3_unpack(vec3 dst[], vec3s src[], size_t len); + CGLM_INLINE vec3s glms_vec3_zero(void); + CGLM_INLINE vec3s glms_vec3_one(void); + CGLM_INLINE float glms_vec3_dot(vec3s a, vec3s b); + CGLM_INLINE float glms_vec3_norm2(vec3s v); + CGLM_INLINE float glms_vec3_norm(vec3s v); + CGLM_INLINE float glms_vec3_norm_one(vec3s v); + CGLM_INLINE float glms_vec3_norm_inf(vec3s v); + CGLM_INLINE vec3s glms_vec3_add(vec3s a, vec3s b); + CGLM_INLINE vec3s glms_vec3_adds(vec3s a, float s); + CGLM_INLINE vec3s glms_vec3_sub(vec3s a, vec3s b); + CGLM_INLINE vec3s glms_vec3_subs(vec3s a, float s); + CGLM_INLINE vec3s glms_vec3_mul(vec3s a, vec3s b); + CGLM_INLINE vec3s glms_vec3_scale(vec3s v, float s); + CGLM_INLINE vec3s glms_vec3_scale_as(vec3s v, float s); + CGLM_INLINE vec3s glms_vec3_div(vec3s a, vec3s b); + CGLM_INLINE vec3s glms_vec3_divs(vec3s a, float s); + CGLM_INLINE vec3s glms_vec3_addadd(vec3s a, vec3s b, vec3s dest); + CGLM_INLINE vec3s glms_vec3_subadd(vec3s a, vec3s b, vec3s dest); + CGLM_INLINE vec3s glms_vec3_muladd(vec3s a, vec3s b, vec3s dest); + CGLM_INLINE vec3s glms_vec3_muladds(vec3s a, float s, vec3s dest); + CGLM_INLINE vec3s glms_vec3_maxadd(vec3s a, vec3s b, vec3s dest); + CGLM_INLINE vec3s glms_vec3_minadd(vec3s a, vec3s b, vec3s dest); + CGLM_INLINE vec3s glms_vec3_subsub(vec3s a, vec3s b, vec3s dest); + CGLM_INLINE vec3s glms_vec3_addsub(vec3s a, vec3s b, vec3s dest); + CGLM_INLINE vec3s glms_vec3_mulsub(vec3s a, vec3s b, vec3s dest); + CGLM_INLINE vec3s glms_vec3_mulsubs(vec3s a, float s, vec3s dest); + CGLM_INLINE vec3s glms_vec3_maxsub(vec3s a, vec3s b, vec3s dest); + CGLM_INLINE vec3s glms_vec3_minsub(vec3s a, vec3s b, vec3s dest); + CGLM_INLINE vec3s glms_vec3_flipsign(vec3s v); + CGLM_INLINE vec3s glms_vec3_negate(vec3s v); + CGLM_INLINE vec3s glms_vec3_normalize(vec3s v); + CGLM_INLINE vec3s glms_vec3_cross(vec3s a, vec3s b); + CGLM_INLINE vec3s glms_vec3_crossn(vec3s a, vec3s b); + CGLM_INLINE float glms_vec3_angle(vec3s a, vec3s b); + CGLM_INLINE vec3s glms_vec3_rotate(vec3s v, float angle, vec3s axis); + CGLM_INLINE vec3s glms_vec3_rotate_m4(mat4s m, vec3s v); + CGLM_INLINE vec3s glms_vec3_rotate_m3(mat3s m, vec3s v); + CGLM_INLINE vec3s glms_vec3_proj(vec3s a, vec3s b); + CGLM_INLINE vec3s glms_vec3_center(vec3s a, vec3s b); + CGLM_INLINE float glms_vec3_distance(vec3s a, vec3s b); + CGLM_INLINE float glms_vec3_distance2(vec3s a, vec3s b); + CGLM_INLINE vec3s glms_vec3_maxv(vec3s a, vec3s b); + CGLM_INLINE vec3s glms_vec3_minv(vec3s a, vec3s b); + CGLM_INLINE vec3s glms_vec3_ortho(vec3s v); + CGLM_INLINE vec3s glms_vec3_clamp(vec3s v, float minVal, float maxVal); + CGLM_INLINE vec3s glms_vec3_lerp(vec3s from, vec3s to, float t); + CGLM_INLINE vec3s glms_vec3_lerpc(vec3s from, vec3s to, float t); + CGLM_INLINE vec3s glms_vec3_mix(vec3s from, vec3s to, float t); + CGLM_INLINE vec3s glms_vec3_mixc(vec3s from, vec3s to, float t); + CGLM_INLINE vec3s glms_vec3_step(vec3s edge, vec3s x); + CGLM_INLINE vec3s glms_vec3_smoothstep_uni(float edge0, float edge1, vec3s x); + CGLM_INLINE vec3s glms_vec3_smoothstep(vec3s edge0, vec3s edge1, vec3s x); + CGLM_INLINE vec3s glms_vec3_smoothinterp(vec3s from, vec3s to, float t); + CGLM_INLINE vec3s glms_vec3_smoothinterpc(vec3s from, vec3s to, float t); + CGLM_INLINE vec3s glms_vec3_swizzle(vec3s v, int mask); + CGLM_INLINE vec3s glms_vec3_make(float * restrict src); + CGLM_INLINE vec3s glms_vec3_faceforward(vec3s n, vec3s v, vec3s nref); + CGLM_INLINE vec3s glms_vec3_reflect(vec3s v, vec3s n); + CGLM_INLINE bool glms_vec3_refract(vec3s v, vec3s n, float eta, vec3s *dest) + + Convenient: + CGLM_INLINE vec3s glms_cross(vec3s a, vec3s b); + CGLM_INLINE float glms_dot(vec3s a, vec3s b); + CGLM_INLINE vec3s glms_normalize(vec3s v); + + Deprecated: + glms_vec3_step_uni --> use glms_vec3_steps + */ + +#ifndef cglms_vec3s_h +#define cglms_vec3s_h + +#include "../common.h" +#include "../types-struct.h" +#include "../util.h" +#include "../vec3.h" +#include "vec3-ext.h" + +/* DEPRECATED! */ +#define glms_vec3_step_uni(edge, x) glms_vec3_steps(edge, x) + +#define GLMS_VEC3_ONE_INIT {GLM_VEC3_ONE_INIT} +#define GLMS_VEC3_ZERO_INIT {GLM_VEC3_ZERO_INIT} + +#define GLMS_VEC3_ONE ((vec3s)GLMS_VEC3_ONE_INIT) +#define GLMS_VEC3_ZERO ((vec3s)GLMS_VEC3_ZERO_INIT) + +#define GLMS_YUP ((vec3s){{0.0f, 1.0f, 0.0f}}) +#define GLMS_ZUP ((vec3s){{0.0f, 0.0f, 1.0f}}) +#define GLMS_XUP ((vec3s){{1.0f, 0.0f, 0.0f}}) + +/*! + * @brief init vec3 using vec4 + * + * @param[in] v4 vector4 + * @returns destination + */ +CGLM_INLINE +vec3s +glms_vec3(vec4s v4) { + vec3s r; + glm_vec3(v4.raw, r.raw); + return r; +} + +/*! + * @brief pack an array of vec3 into an array of vec3s + * + * @param[out] dst array of vec3 + * @param[in] src array of vec3s + * @param[in] len number of elements + */ +CGLM_INLINE +void +glms_vec3_(pack)(vec3s dst[], vec3 src[], size_t len) { + size_t i; + + for (i = 0; i < len; i++) { + glm_vec3_copy(src[i], dst[i].raw); + } +} + +/*! + * @brief unpack an array of vec3s into an array of vec3 + * + * @param[out] dst array of vec3s + * @param[in] src array of vec3 + * @param[in] len number of elements + */ +CGLM_INLINE +void +glms_vec3_(unpack)(vec3 dst[], vec3s src[], size_t len) { + size_t i; + + for (i = 0; i < len; i++) { + glm_vec3_copy(src[i].raw, dst[i]); + } +} + +/*! + * @brief make vector zero + * + * @returns zero vector + */ +CGLM_INLINE +vec3s +glms_vec3_(zero)(void) { + vec3s r; + glm_vec3_zero(r.raw); + return r; +} + +/*! + * @brief make vector one + * + * @returns one vector + */ +CGLM_INLINE +vec3s +glms_vec3_(one)(void) { + vec3s r; + glm_vec3_one(r.raw); + return r; +} + +/*! + * @brief vec3 dot product + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return dot product + */ +CGLM_INLINE +float +glms_vec3_(dot)(vec3s a, vec3s b) { + return glm_vec3_dot(a.raw, b.raw); +} + +/*! + * @brief norm * norm (magnitude) of vec + * + * we can use this func instead of calling norm * norm, because it would call + * sqrtf function twice but with this func we can avoid func call, maybe this is + * not good name for this func + * + * @param[in] v vector + * + * @return norm * norm + */ +CGLM_INLINE +float +glms_vec3_(norm2)(vec3s v) { + return glm_vec3_norm2(v.raw); +} + +/*! + * @brief norm (magnitude) of vec3 + * + * @param[in] v vector + * + * @return norm + */ +CGLM_INLINE +float +glms_vec3_(norm)(vec3s v) { + return glm_vec3_norm(v.raw); +} + +/*! + * @brief L1 norm of vec3 + * Also known as Manhattan Distance or Taxicab norm. + * L1 Norm is the sum of the magnitudes of the vectors in a space. + * It is calculated as the sum of the absolute values of the vector components. + * In this norm, all the components of the vector are weighted equally. + * + * This computes: + * R = |v[0]| + |v[1]| + |v[2]| + * + * @param[in] v vector + * + * @return L1 norm + */ +CGLM_INLINE +float +glms_vec3_(norm_one)(vec3s v) { + return glm_vec3_norm_one(v.raw); +} + +/*! + * @brief Infinity norm of vec3 + * Also known as Maximum norm. + * Infinity Norm is the largest magnitude among each element of a vector. + * It is calculated as the maximum of the absolute values of the vector components. + * + * This computes: + * inf norm = max(|v[0]|, |v[1]|, |v[2]|) + * + * @param[in] v vector + * + * @return Infinity norm + */ +CGLM_INLINE +float +glms_vec3_(norm_inf)(vec3s v) { + return glm_vec3_norm_inf(v.raw); +} + +/*! + * @brief add a vector to b vector store result in dest + * + * @param[in] a vector1 + * @param[in] b vector2 + * @returns destination vector + */ +CGLM_INLINE +vec3s +glms_vec3_(add)(vec3s a, vec3s b) { + vec3s r; + glm_vec3_add(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief add scalar to v vector store result in dest (d = v + s) + * + * @param[in] a vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec3s +glms_vec3_(adds)(vec3s a, float s) { + vec3s r; + glm_vec3_adds(a.raw, s, r.raw); + return r; +} + +/*! + * @brief subtract b vector from a vector store result in dest + * + * @param[in] a vector1 + * @param[in] b vector2 + * @returns destination vector + */ +CGLM_INLINE +vec3s +glms_vec3_(sub)(vec3s a, vec3s b) { + vec3s r; + glm_vec3_sub(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief subtract scalar from v vector store result in dest (d = v - s) + * + * @param[in] a vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec3s +glms_vec3_(subs)(vec3s a, float s) { + vec3s r; + glm_vec3_subs(a.raw, s, r.raw); + return r; +} + +/*! + * @brief multiply two vectors (component-wise multiplication) + * + * @param a vector1 + * @param b vector2 + * @returns v3 = (a[0] * b[0], a[1] * b[1], a[2] * b[2]) + */ +CGLM_INLINE +vec3s +glms_vec3_(mul)(vec3s a, vec3s b) { + vec3s r; + glm_vec3_mul(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief multiply/scale vec3 vector with scalar: result = v * s + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec3s +glms_vec3_(scale)(vec3s v, float s) { + vec3s r; + glm_vec3_scale(v.raw, s, r.raw); + return r; +} + +/*! + * @brief make vec3 vector scale as specified: result = unit(v) * s + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec3s +glms_vec3_(scale_as)(vec3s v, float s) { + vec3s r; + glm_vec3_scale_as(v.raw, s, r.raw); + return r; +} + +/*! + * @brief div vector with another component-wise division: d = a / b + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns result = (a[0]/b[0], a[1]/b[1], a[2]/b[2]) + */ +CGLM_INLINE +vec3s +glms_vec3_(div)(vec3s a, vec3s b) { + vec3s r; + glm_vec3_div(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief div vector with scalar: d = v / s + * + * @param[in] a vector + * @param[in] s scalar + * @returns result = (a[0]/s, a[1]/s, a[2]/s) + */ +CGLM_INLINE +vec3s +glms_vec3_(divs)(vec3s a, float s) { + vec3s r; + glm_vec3_divs(a.raw, s, r.raw); + return r; +} + +/*! + * @brief add two vectors and add result to sum + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest += (a + b) + */ +CGLM_INLINE +vec3s +glms_vec3_(addadd)(vec3s a, vec3s b, vec3s dest) { + glm_vec3_addadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief sub two vectors and add result to dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest += (a + b) + */ +CGLM_INLINE +vec3s +glms_vec3_(subadd)(vec3s a, vec3s b, vec3s dest) { + glm_vec3_subadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief mul two vectors and add result to dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest += (a * b) + */ +CGLM_INLINE +vec3s +glms_vec3_(muladd)(vec3s a, vec3s b, vec3s dest) { + glm_vec3_muladd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief mul vector with scalar and add result to sum + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @returns dest += (a * b) + */ +CGLM_INLINE +vec3s +glms_vec3_(muladds)(vec3s a, float s, vec3s dest) { + glm_vec3_muladds(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief add max of two vectors to result/dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest += max(a, b) + */ +CGLM_INLINE +vec3s +glms_vec3_(maxadd)(vec3s a, vec3s b, vec3s dest) { + glm_vec3_maxadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief add min of two vectors to result/dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest += min(a, b) + */ +CGLM_INLINE +vec3s +glms_vec3_(minadd)(vec3s a, vec3s b, vec3s dest) { + glm_vec3_minadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief sub two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest -= (a - b) + */ +CGLM_INLINE +vec3s +glms_vec3_(subsub)(vec3s a, vec3s b, vec3s dest) { + glm_vec3_subsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief add two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest -= (a + b) + */ +CGLM_INLINE +vec3s +glms_vec3_(addsub)(vec3s a, vec3s b, vec3s dest) { + glm_vec3_addsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief mul two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest -= (a * b) + */ +CGLM_INLINE +vec3s +glms_vec3_(mulsub)(vec3s a, vec3s b, vec3s dest) { + glm_vec3_mulsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief mul vector with scalar and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @returns dest -= (a * b) + */ +CGLM_INLINE +vec3s +glms_vec3_(mulsubs)(vec3s a, float s, vec3s dest) { + glm_vec3_mulsubs(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief sub max of two vectors to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest -= max(a, b) + */ +CGLM_INLINE +vec3s +glms_vec3_(maxsub)(vec3s a, vec3s b, vec3s dest) { + glm_vec3_maxsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief sub min of two vectors to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest -= min(a, b) + */ +CGLM_INLINE +vec3s +glms_vec3_(minsub)(vec3s a, vec3s b, vec3s dest) { + glm_vec3_minsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief negate vector components and store result in dest + * + * @param[in] v vector + * @returns result vector + */ +CGLM_INLINE +vec3s +glms_vec3_(flipsign)(vec3s v) { + glm_vec3_flipsign(v.raw); + return v; +} + +/*! + * @brief negate vector components + * + * @param[in] v vector + * @returns negated vector + */ +CGLM_INLINE +vec3s +glms_vec3_(negate)(vec3s v) { + glm_vec3_negate(v.raw); + return v; +} + +/*! + * @brief normalize vec3 and store result in same vec + * + * @param[in] v vector + * @returns normalized vector + */ +CGLM_INLINE +vec3s +glms_vec3_(normalize)(vec3s v) { + glm_vec3_normalize(v.raw); + return v; +} + +/*! + * @brief cross product of two vector (RH) + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns destination + */ +CGLM_INLINE +vec3s +glms_vec3_(cross)(vec3s a, vec3s b) { + vec3s r; + glm_vec3_cross(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief cross product of two vector (RH) and normalize the result + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns destination + */ +CGLM_INLINE +vec3s +glms_vec3_(crossn)(vec3s a, vec3s b) { + vec3s r; + glm_vec3_crossn(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief angle between two vector + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return angle as radians + */ +CGLM_INLINE +float +glms_vec3_(angle)(vec3s a, vec3s b) { + return glm_vec3_angle(a.raw, b.raw); +} + +/*! + * @brief rotate vec3 around axis by angle using Rodrigues' rotation formula + * + * @param[in] v vector + * @param[in] axis axis vector (must be unit vector) + * @param[in] angle angle by radians + * @returns rotated vector + */ +CGLM_INLINE +vec3s +glms_vec3_(rotate)(vec3s v, float angle, vec3s axis) { + glm_vec3_rotate(v.raw, angle, axis.raw); + return v; +} + +/*! + * @brief apply rotation matrix to vector + * + * matrix format should be (no perspective): + * a b c x + * e f g y + * i j k z + * 0 0 0 w + * + * @param[in] m affine matrix or rot matrix + * @param[in] v vector + * @returns rotated vector + */ +CGLM_INLINE +vec3s +glms_vec3_(rotate_m4)(mat4s m, vec3s v) { + vec3s r; + glm_vec3_rotate_m4(m.raw, v.raw, r.raw); + return r; +} + +/*! + * @brief apply rotation matrix to vector + * + * @param[in] m affine matrix or rot matrix + * @param[in] v vector + * @returns rotated vector + */ +CGLM_INLINE +vec3s +glms_vec3_(rotate_m3)(mat3s m, vec3s v) { + vec3s r; + glm_vec3_rotate_m3(m.raw, v.raw, r.raw); + return r; +} + +/*! + * @brief project a vector onto b vector + * + * @param[in] a vector1 + * @param[in] b vector2 + * @returns projected vector + */ +CGLM_INLINE +vec3s +glms_vec3_(proj)(vec3s a, vec3s b) { + vec3s r; + glm_vec3_proj(a.raw, b.raw, r.raw); + return r; +} + +/** + * @brief find center point of two vector + * + * @param[in] a vector1 + * @param[in] b vector2 + * @returns center point + */ +CGLM_INLINE +vec3s +glms_vec3_(center)(vec3s a, vec3s b) { + vec3s r; + glm_vec3_center(a.raw, b.raw, r.raw); + return r; +} + +/** + * @brief distance between two vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @return distance + */ +CGLM_INLINE +float +glms_vec3_(distance)(vec3s a, vec3s b) { + return glm_vec3_distance(a.raw, b.raw); +} + +/** + * @brief squared distance between two vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @return squared distance (distance * distance) + */ +CGLM_INLINE +float +glms_vec3_(distance2)(vec3s a, vec3s b) { + return glm_vec3_distance2(a.raw, b.raw); +} + +/*! + * @brief max values of vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @returns destination + */ +CGLM_INLINE +vec3s +glms_vec3_(maxv)(vec3s a, vec3s b) { + vec3s r; + glm_vec3_maxv(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief min values of vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @returns destination + */ +CGLM_INLINE +vec3s +glms_vec3_(minv)(vec3s a, vec3s b) { + vec3s r; + glm_vec3_minv(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief possible orthogonal/perpendicular vector + * + * @param[in] v vector + * @returns orthogonal/perpendicular vector + */ +CGLM_INLINE +vec3s +glms_vec3_(ortho)(vec3s v) { + vec3s r; + glm_vec3_ortho(v.raw, r.raw); + return r; +} + +/*! + * @brief clamp vector's individual members between min and max values + * + * @param[in] v vector + * @param[in] minVal minimum value + * @param[in] maxVal maximum value + * @returns clamped vector + */ +CGLM_INLINE +vec3s +glms_vec3_(clamp)(vec3s v, float minVal, float maxVal) { + glm_vec3_clamp(v.raw, minVal, maxVal); + return v; +} + +/*! + * @brief linear interpolation between two vectors + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) + * @returns destination + */ +CGLM_INLINE +vec3s +glms_vec3_(lerp)(vec3s from, vec3s to, float t) { + vec3s r; + glm_vec3_lerp(from.raw, to.raw, t, r.raw); + return r; +} + +/*! + * @brief linear interpolation between two vectors (clamped) + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) clamped between 0 and 1 + * @returns destination + */ +CGLM_INLINE +vec3s +glms_vec3_(lerpc)(vec3s from, vec3s to, float t) { + vec3s r; + glm_vec3_lerpc(from.raw, to.raw, t, r.raw); + return r; +} + +/*! + * @brief linear interpolation between two vectors + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) + * @returns destination + */ +CGLM_INLINE +vec3s +glms_vec3_(mix)(vec3s from, vec3s to, float t) { + vec3s r; + glm_vec3_mix(from.raw, to.raw, t, r.raw); + return r; +} + +/*! + * @brief linear interpolation between two vectors (clamped) + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) clamped between 0 and 1 + * @returns destination + */ +CGLM_INLINE +vec3s +glms_vec3_(mixc)(vec3s from, vec3s to, float t) { + vec3s r; + glm_vec3_mixc(from.raw, to.raw, t, r.raw); + return r; +} + +/*! + * @brief threshold function + * + * @param[in] edge threshold + * @param[in] x value to test against threshold + * @returns 0.0 if x < edge, else 1.0 + */ +CGLM_INLINE +vec3s +glms_vec3_(step)(vec3s edge, vec3s x) { + vec3s r; + glm_vec3_step(edge.raw, x.raw, r.raw); + return r; +} + +/*! + * @brief threshold function with a smooth transition (unidimensional) + * + * @param[in] edge0 low threshold + * @param[in] edge1 high threshold + * @param[in] x value to test against threshold + * @returns destination + */ +CGLM_INLINE +vec3s +glms_vec3_(smoothstep_uni)(float edge0, float edge1, vec3s x) { + vec3s r; + glm_vec3_smoothstep_uni(edge0, edge1, x.raw, r.raw); + return r; +} + +/*! + * @brief threshold function with a smooth transition + * + * @param[in] edge0 low threshold + * @param[in] edge1 high threshold + * @param[in] x value to test against threshold + * @returns destination + */ +CGLM_INLINE +vec3s +glms_vec3_(smoothstep)(vec3s edge0, vec3s edge1, vec3s x) { + vec3s r; + glm_vec3_smoothstep(edge0.raw, edge1.raw, x.raw, r.raw); + return r; +} + +/*! + * @brief smooth Hermite interpolation between two vectors + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) + * @returns destination + */ +CGLM_INLINE +vec3s +glms_vec3_(smoothinterp)(vec3s from, vec3s to, float t) { + vec3s r; + glm_vec3_smoothinterp(from.raw, to.raw, t, r.raw); + return r; +} + +/*! + * @brief smooth Hermite interpolation between two vectors (clamped) + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) clamped between 0 and 1 + * @returns destination + */ +CGLM_INLINE +vec3s +glms_vec3_(smoothinterpc)(vec3s from, vec3s to, float t) { + vec3s r; + glm_vec3_smoothinterpc(from.raw, to.raw, t, r.raw); + return r; +} + +/*! + * @brief vec3 cross product + * + * this is just convenient wrapper + * + * @param[in] a source 1 + * @param[in] b source 2 + * @returns destination + */ +CGLM_INLINE +vec3s +glms_cross(vec3s a, vec3s b) { + vec3s r; + glm_cross(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief vec3 dot product + * + * this is just convenient wrapper + * + * @param[in] a vector1 + * @param[in] b vector2 + * @return dot product + */ +CGLM_INLINE +float +glms_dot(vec3s a, vec3s b) { + return glm_dot(a.raw, b.raw); +} + +/*! + * @brief normalize vec3 and store result in same vec + * + * this is just convenient wrapper + * + * @param[in] v vector + * @returns normalized vector + */ +CGLM_INLINE +vec3s +glms_normalize(vec3s v) { + glm_normalize(v.raw); + return v; +} + +/*! + * @brief swizzle vector components + * + * you can use existing masks e.g. GLM_XXX, GLM_ZYX + * + * @param[in] v source + * @param[in] mask mask + * @returns swizzled vector + */ +CGLM_INLINE +vec3s +glms_vec3_(swizzle)(vec3s v, int mask) { + vec3s dest; + glm_vec3_swizzle(v.raw, mask, dest.raw); + return dest; +} + +/*! + * @brief Create three dimensional vector from pointer + * + * @param[in] src pointer to an array of floats + * @returns constructed 3D vector from raw pointer + */ +CGLM_INLINE +vec3s +glms_vec3_(make)(const float * __restrict src) { + vec3s dest; + glm_vec3_make(src, dest.raw); + return dest; +} + +/*! + * @brief a vector pointing in the same direction as another + * + * orients a vector to point away from a surface as defined by its normal + * + * @param[in] n vector to orient. + * @param[in] v incident vector + * @param[in] nref reference vector + * @returns oriented vector, pointing away from the surface. + */ +CGLM_INLINE +vec3s +glms_vec3_(faceforward)(vec3s n, vec3s v, vec3s nref) { + vec3s dest; + glm_vec3_faceforward(n.raw, v.raw, nref.raw, dest.raw); + return dest; +} + +/*! + * @brief reflection vector using an incident ray and a surface normal + * + * @param[in] I incident vector + * @param[in] N normalized normal vector + * @returns reflection result + */ +CGLM_INLINE +vec3s +glms_vec3_(reflect)(vec3s v, vec3s n) { + vec3s dest; + glm_vec3_reflect(v.raw, n.raw, dest.raw); + return dest; +} + +/*! + * @brief computes refraction vector for an incident vector and a surface normal. + * + * calculates the refraction vector based on Snell's law. If total internal reflection + * occurs (angle too great given eta), dest is set to zero and returns false. + * Otherwise, computes refraction vector, stores it in dest, and returns true. + * + * @param[in] v normalized incident vector + * @param[in] n normalized normal vector + * @param[in] eta ratio of indices of refraction (incident/transmitted) + * @param[out] dest refraction vector if refraction occurs; zero vector otherwise + * + * @returns true if refraction occurs; false if total internal reflection occurs. + */ +CGLM_INLINE +bool +glms_vec3_(refract)(vec3s v, vec3s n, float eta, vec3s * __restrict dest) { + return glm_vec3_refract(v.raw, n.raw, eta, dest->raw); +} + +#endif /* cglms_vec3s_h */ diff --git a/cglm/include/cglm/struct/vec4-ext.h b/cglm/include/cglm/struct/vec4-ext.h new file mode 100644 index 0000000..f57348e --- /dev/null +++ b/cglm/include/cglm/struct/vec4-ext.h @@ -0,0 +1,325 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/*! + * @brief SIMD like functions + */ + +/* + Functions: + CGLM_INLINE vec4s glms_vec4_broadcast(float val); + CGLM_INLINE vec4s glms_vec4_fill(float val); + CGLM_INLINE bool glms_vec4_eq(vec4s v, float val); + CGLM_INLINE bool glms_vec4_eq_eps(vec4s v, float val); + CGLM_INLINE bool glms_vec4_eq_all(vec4s v); + CGLM_INLINE bool glms_vec4_eqv(vec4s a, vec4s b); + CGLM_INLINE bool glms_vec4_eqv_eps(vec4s a, vec4s b); + CGLM_INLINE float glms_vec4_max(vec4s v); + CGLM_INLINE float glms_vec4_min(vec4s v); + CGLM_INLINE bool glms_vec4_isnan(vec4s v); + CGLM_INLINE bool glms_vec4_isinf(vec4s v); + CGLM_INLINE bool glms_vec4_isvalid(vec4s v); + CGLM_INLINE vec4s glms_vec4_sign(vec4s v); + CGLM_INLINE vec4s glms_vec4_abs(vec4s v); + CGLM_INLINE vec4s glms_vec4_fract(vec4s v); + CGLM_INLINE float glms_vec4_floor(vec4s v); + CGLM_INLINE float glms_vec4_mods(vec4s v, float s); + CGLM_INLINE float glms_vec4_steps(float edge, vec4s v); + CGLM_INLINE void glms_vec4_stepr(vec4s edge, float v); + CGLM_INLINE float glms_vec4_hadd(vec4s v); + CGLM_INLINE vec4s glms_vec4_sqrt(vec4s v); + */ + +#ifndef cglms_vec4s_ext_h +#define cglms_vec4s_ext_h + +#include "../common.h" +#include "../types-struct.h" +#include "../util.h" +#include "../vec4-ext.h" + +/* api definition */ +#define glms_vec4_(NAME) CGLM_STRUCTAPI(vec4, NAME) + +/*! + * @brief fill a vector with specified value + * + * @param val value + * @returns dest + */ +CGLM_INLINE +vec4s +glms_vec4_(broadcast)(float val) { + vec4s r; + glm_vec4_broadcast(val, r.raw); + return r; +} + +/*! + * @brief fill a vector with specified value + * + * @param val value + * @returns dest + */ +CGLM_INLINE +vec4s +glms_vec4_(fill)(float val) { + vec4s r; + glm_vec4_fill(r.raw, val); + return r; +} + +/*! + * @brief check if vector is equal to value (without epsilon) + * + * @param v vector + * @param val value + */ +CGLM_INLINE +bool +glms_vec4_(eq)(vec4s v, float val) { + return glm_vec4_eq(v.raw, val); +} + +/*! + * @brief check if vector is equal to value (with epsilon) + * + * @param v vector + * @param val value + */ +CGLM_INLINE +bool +glms_vec4_(eq_eps)(vec4s v, float val) { + return glm_vec4_eq_eps(v.raw, val); +} + +/*! + * @brief check if vector members are equal (without epsilon) + * + * @param v vector + */ +CGLM_INLINE +bool +glms_vec4_(eq_all)(vec4s v) { + return glm_vec4_eq_all(v.raw); +} + +/*! + * @brief check if vector is equal to another (without epsilon) + * + * @param a vector + * @param b vector + */ +CGLM_INLINE +bool +glms_vec4_(eqv)(vec4s a, vec4s b) { + return glm_vec4_eqv(a.raw, b.raw); +} + +/*! + * @brief check if vector is equal to another (with epsilon) + * + * @param a vector + * @param b vector + */ +CGLM_INLINE +bool +glms_vec4_(eqv_eps)(vec4s a, vec4s b) { + return glm_vec4_eqv_eps(a.raw, b.raw); +} + +/*! + * @brief max value of vector + * + * @param v vector + */ +CGLM_INLINE +float +glms_vec4_(max)(vec4s v) { + return glm_vec4_max(v.raw); +} + +/*! + * @brief min value of vector + * + * @param v vector + */ +CGLM_INLINE +float +glms_vec4_(min)(vec4s v) { + return glm_vec4_min(v.raw); +} + +/*! + * @brief check if one of items is NaN (not a number) + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glms_vec4_(isnan)(vec4s v) { + return glm_vec4_isnan(v.raw); +} + +/*! + * @brief check if one of items is INFINITY + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glms_vec4_(isinf)(vec4s v) { + return glm_vec4_isinf(v.raw); +} + +/*! + * @brief check if all items are valid number + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glms_vec4_(isvalid)(vec4s v) { + return glm_vec4_isvalid(v.raw); +} + +/*! + * @brief get sign of 32 bit float as +1, -1, 0 + * + * Important: It returns 0 for zero/NaN input + * + * @param v vector + * @returns sign vector + */ +CGLM_INLINE +vec4s +glms_vec4_(sign)(vec4s v) { + vec4s r; + glm_vec4_sign(v.raw, r.raw); + return r; +} + +/*! + * @brief absolute value of each vector item + * + * @param[in] v vector + * @returns destination vector + */ +CGLM_INLINE +vec4s +glms_vec4_(abs)(vec4s v) { + vec4s r; + glm_vec4_abs(v.raw, r.raw); + return r; +} + +/*! + * @brief fractional part of each vector item + * + * @param[in] v vector + * @returns dest destination vector + */ +CGLM_INLINE +vec4s +glms_vec4_(fract)(vec4s v) { + vec4s r; + glm_vec4_fract(v.raw, r.raw); + return r; +} + +/*! + * @brief floor of each vector item + * + * @param[in] v vector + * @returns dest destination vector + */ +CGLM_INLINE +vec4s +glms_vec4_(floor)(vec4s v) { + vec4s r; + glm_vec4_floor(v.raw, r.raw); + return r; +} + +/*! + * @brief mod of each vector item by scalar + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec4s +glms_vec4_(mods)(vec4s v, float s) { + vec4s r; + glm_vec4_mods(v.raw, s, r.raw); + return r; +} + +/*! + * @brief threshold each vector item with scalar + * condition is: (x[i] < edge) ? 0.0 : 1.0 + * + * @param[in] edge threshold + * @param[in] x vector to test against threshold + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4_(steps)(float edge, vec4s x) { + vec4s r; + glm_vec4_steps(edge, x.raw, r.raw); + return r; +} + +/*! + * @brief threshold a value with *vector* as the threshold + * condition is: (x < edge[i]) ? 0.0 : 1.0 + * + * @param[in] edge threshold vector + * @param[in] x value to test against threshold + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4_(stepr)(vec4s edge, float x) { + vec4s r; + glm_vec4_stepr(edge.raw, x, r.raw); + return r; +} + +/*! + * @brief vector reduction by summation + * @warning could overflow + * + * @param[in] v vector + * @return sum of all vector's elements + */ +CGLM_INLINE +float +glms_vec4_(hadd)(vec4s v) { + return glm_vec4_hadd(v.raw); +} + +/*! + * @brief square root of each vector item + * + * @param[in] v vector + * @returns destination vector + */ +CGLM_INLINE +vec4s +glms_vec4_(sqrt)(vec4s v) { + vec4s r; + glm_vec4_sqrt(v.raw, r.raw); + return r; +} + +#endif /* cglms_vec4s_ext_h */ diff --git a/cglm/include/cglm/struct/vec4.h b/cglm/include/cglm/struct/vec4.h new file mode 100644 index 0000000..a64c1a3 --- /dev/null +++ b/cglm/include/cglm/struct/vec4.h @@ -0,0 +1,961 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLMS_VEC4_ONE_INIT + GLMS_VEC4_BLACK_INIT + GLMS_VEC4_ZERO_INIT + GLMS_VEC4_ONE + GLMS_VEC4_BLACK + GLMS_VEC4_ZERO + + Functions: + CGLM_INLINE vec4s glms_vec4(vec3s v3, float last); + CGLM_INLINE vec3s glms_vec4_copy3(vec4s v); + CGLM_INLINE vec4s glms_vec4_copy(vec4s v); + CGLM_INLINE vec4s glms_vec4_ucopy(vec4s v); + CGLM_INLINE void glms_vec4_pack(vec4s dst[], vec4 src[], size_t len); + CGLM_INLINE void glms_vec4_unpack(vec4 dst[], vec4s src[], size_t len); + CGLM_INLINE float glms_vec4_dot(vec4s a, vec4s b); + CGLM_INLINE float glms_vec4_norm2(vec4s v); + CGLM_INLINE float glms_vec4_norm(vec4s v); + CGLM_INLINE float glms_vec4_norm_one(vec4s v); + CGLM_INLINE float glms_vec4_norm_inf(vec4s v); + CGLM_INLINE vec4s glms_vec4_add(vec4s a, vec4s b); + CGLM_INLINE vec4s glms_vec4_adds(vec4s v, float s); + CGLM_INLINE vec4s glms_vec4_sub(vec4s a, vec4s b); + CGLM_INLINE vec4s glms_vec4_subs(vec4s v, float s); + CGLM_INLINE vec4s glms_vec4_mul(vec4s a, vec4s b); + CGLM_INLINE vec4s glms_vec4_scale(vec4s v, float s); + CGLM_INLINE vec4s glms_vec4_scale_as(vec4s v, float s); + CGLM_INLINE vec4s glms_vec4_div(vec4s a, vec4s b); + CGLM_INLINE vec4s glms_vec4_divs(vec4s v, float s); + CGLM_INLINE vec4s glms_vec4_addadd(vec4s a, vec4s b, vec4s dest); + CGLM_INLINE vec4s glms_vec4_subadd(vec4s a, vec4s b, vec4s dest); + CGLM_INLINE vec4s glms_vec4_muladd(vec4s a, vec4s b, vec4s dest); + CGLM_INLINE vec4s glms_vec4_muladds(vec4s a, float s, vec4s dest); + CGLM_INLINE vec4s glms_vec4_maxadd(vec4s a, vec4s b, vec4s dest); + CGLM_INLINE vec4s glms_vec4_minadd(vec4s a, vec4s b, vec4s dest); + CGLM_INLINE vec4s glms_vec4_subsub(vec4s a, vec4s b, vec4s dest); + CGLM_INLINE vec4s glms_vec4_addsub(vec4s a, vec4s b, vec4s dest); + CGLM_INLINE vec4s glms_vec4_mulsub(vec4s a, vec4s b, vec4s dest); + CGLM_INLINE vec4s glms_vec4_mulsubs(vec4s a, float s, vec4s dest); + CGLM_INLINE vec4s glms_vec4_maxsub(vec4s a, vec4s b, vec4s dest); + CGLM_INLINE vec4s glms_vec4_minsub(vec4s a, vec4s b, vec4s dest); + CGLM_INLINE vec4s glms_vec4_negate(vec4s v); + CGLM_INLINE vec4s glms_vec4_normalize(vec4s v); + CGLM_INLINE float glms_vec4_distance(vec4s a, vec4s b); + CGLM_INLINE float glms_vec4_distance2(vec4s a, vec4s b); + CGLM_INLINE vec4s glms_vec4_maxv(vec4s a, vec4s b); + CGLM_INLINE vec4s glms_vec4_minv(vec4s a, vec4s b); + CGLM_INLINE vec4s glms_vec4_clamp(vec4s v, float minVal, float maxVal); + CGLM_INLINE vec4s glms_vec4_lerp(vec4s from, vec4s to, float t); + CGLM_INLINE vec4s glms_vec4_lerpc(vec4s from, vec4s to, float t); + CGLM_INLINE vec4s glms_vec4_mix(vec4s from, vec4s to, float t); + CGLM_INLINE vec4s glms_vec4_mixc(vec4s from, vec4s to, float t); + CGLM_INLINE vec4s glms_vec4_step(vec4s edge, vec4s x); + CGLM_INLINE vec4s glms_vec4_smoothstep_uni(float edge0, float edge1, vec4s x); + CGLM_INLINE vec4s glms_vec4_smoothstep(vec4s edge0, vec4s edge1, vec4s x); + CGLM_INLINE vec4s glms_vec4_smoothinterp(vec4s from, vec4s to, float t); + CGLM_INLINE vec4s glms_vec4_smoothinterpc(vec4s from, vec4s to, float t); + CGLM_INLINE vec4s glms_vec4_cubic(float s); + CGLM_INLINE vec4s glms_vec4_swizzle(vec4s v, int mask); + CGLM_INLINE vec4s glms_vec4_make(float * restrict src); + CGLM_INLINE vec4s glms_vec4_reflect(vec4s v, vec4s n); + CGLM_INLINE bool glms_vec4_refract(vec4s v, vec4s n, float eta, vec4s *dest) + + Deprecated: + glms_vec4_step_uni --> use glms_vec4_steps + */ + +#ifndef cglms_vec4s_h +#define cglms_vec4s_h + +#include "../common.h" +#include "../types-struct.h" +#include "../util.h" +#include "../vec4.h" +#include "vec4-ext.h" + +/* DEPRECATED! */ +#define glms_vec4_step_uni(edge, x) glms_vec4_steps(edge, x) + +#define GLMS_VEC4_ONE_INIT {GLM_VEC4_ONE_INIT} +#define GLMS_VEC4_BLACK_INIT {GLM_VEC4_BLACK_INIT} +#define GLMS_VEC4_ZERO_INIT {GLM_VEC4_ZERO_INIT} + +#define GLMS_VEC4_ONE ((vec4s)GLM_VEC4_ONE_INIT) +#define GLMS_VEC4_BLACK ((vec4s)GLM_VEC4_BLACK_INIT) +#define GLMS_VEC4_ZERO ((vec4s)GLM_VEC4_ZERO_INIT) + +/*! + * @brief init vec4 using vec3 + * + * @param[in] v3 vector3 + * @param[in] last last item + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4(vec3s v3, float last) { + vec4s r; + glm_vec4(v3.raw, last, r.raw); + return r; +} + +/*! + * @brief copy first 3 members of [a] to [dest] + * + * @param[in] v source + * @returns vec3 + */ +CGLM_INLINE +vec3s +glms_vec4_(copy3)(vec4s v) { + vec3s r; + glm_vec4_copy3(v.raw, r.raw); + return r; +} + +/*! + * @brief copy all members of [a] to [dest] + * + * @param[in] v source + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4_(copy)(vec4s v) { + vec4s r; + glm_vec4_copy(v.raw, r.raw); + return r; +} + +/*! + * @brief copy all members of [a] to [dest] + * + * alignment is not required + * + * @param[in] v source + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4_(ucopy)(vec4s v) { + vec4s r; + glm_vec4_ucopy(v.raw, r.raw); + return r; +} + +/*! + * @brief pack an array of vec4 into an array of vec4s + * + * @param[out] dst array of vec4 + * @param[in] src array of vec4s + * @param[in] len number of elements + */ +CGLM_INLINE +void +glms_vec4_(pack)(vec4s dst[], vec4 src[], size_t len) { + size_t i; + + for (i = 0; i < len; i++) { + glm_vec4_copy(src[i], dst[i].raw); + } +} + +/*! + * @brief unpack an array of vec4s into an array of vec4 + * + * @param[out] dst array of vec4s + * @param[in] src array of vec4 + * @param[in] len number of elements + */ +CGLM_INLINE +void +glms_vec4_(unpack)(vec4 dst[], vec4s src[], size_t len) { + size_t i; + + for (i = 0; i < len; i++) { + glm_vec4_copy(src[i].raw, dst[i]); + } +} + +/*! + * @brief make vector zero + * + * @returns zero vector + */ +CGLM_INLINE +vec4s +glms_vec4_(zero)(void) { + vec4s r; + glm_vec4_zero(r.raw); + return r; +} + +/*! + * @brief make vector one + * + * @returns one vector + */ +CGLM_INLINE +vec4s +glms_vec4_(one)(void) { + vec4s r; + glm_vec4_one(r.raw); + return r; +} + +/*! + * @brief vec4 dot product + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return dot product + */ +CGLM_INLINE +float +glms_vec4_(dot)(vec4s a, vec4s b) { + return glm_vec4_dot(a.raw, b.raw); +} + +/*! + * @brief norm * norm (magnitude) of vec + * + * we can use this func instead of calling norm * norm, because it would call + * sqrtf function twice but with this func we can avoid func call, maybe this is + * not good name for this func + * + * @param[in] v vec4 + * + * @return norm * norm + */ +CGLM_INLINE +float +glms_vec4_(norm2)(vec4s v) { + return glm_vec4_norm2(v.raw); +} + +/*! + * @brief norm (magnitude) of vec4 + * + * @param[in] v vector + * + * @return norm + */ +CGLM_INLINE +float +glms_vec4_(norm)(vec4s v) { + return glm_vec4_norm(v.raw); +} + +/*! + * @brief L1 norm of vec4 + * Also known as Manhattan Distance or Taxicab norm. + * L1 Norm is the sum of the magnitudes of the vectors in a space. + * It is calculated as the sum of the absolute values of the vector components. + * In this norm, all the components of the vector are weighted equally. + * + * This computes: + * R = |v[0]| + |v[1]| + |v[2]| + |v[3]| + * + * @param[in] v vector + * + * @return L1 norm + */ +CGLM_INLINE +float +glms_vec4_(norm_one)(vec4s v) { + return glm_vec4_norm_one(v.raw); +} + +/*! + * @brief Infinity norm of vec4 + * Also known as Maximum norm. + * Infinity Norm is the largest magnitude among each element of a vector. + * It is calculated as the maximum of the absolute values of the vector components. + * + * This computes: + * inf norm = max(|v[0]|, |v[1]|, |v[2]|, |v[3]|) + * + * @param[in] v vector + * + * @return Infinity norm + */ +CGLM_INLINE +float +glms_vec4_(norm_inf)(vec4s v) { + return glm_vec4_norm_inf(v.raw); +} + +/*! + * @brief add b vector to a vector store result in dest + * + * @param[in] a vector1 + * @param[in] b vector2 + * @returns destination vector + */ +CGLM_INLINE +vec4s +glms_vec4_(add)(vec4s a, vec4s b) { + vec4s r; + glm_vec4_add(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief add scalar to v vector store result in dest (d = v + vec(s)) + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec4s +glms_vec4_(adds)(vec4s v, float s) { + vec4s r; + glm_vec4_adds(v.raw, s, r.raw); + return r; +} + +/*! + * @brief subtract b vector from a vector store result in dest (d = a - b) + * + * @param[in] a vector1 + * @param[in] b vector2 + * @returns destination vector + */ +CGLM_INLINE +vec4s +glms_vec4_(sub)(vec4s a, vec4s b) { + vec4s r; + glm_vec4_sub(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief subtract scalar from v vector store result in dest (d = v - vec(s)) + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec4s +glms_vec4_(subs)(vec4s v, float s) { + vec4s r; + glm_vec4_subs(v.raw, s, r.raw); + return r; +} + +/*! + * @brief multiply two vectors (component-wise multiplication) + * + * @param a vector1 + * @param b vector2 + * @returns dest = (a[0] * b[0], a[1] * b[1], a[2] * b[2], a[3] * b[3]) + */ +CGLM_INLINE +vec4s +glms_vec4_(mul)(vec4s a, vec4s b) { + vec4s r; + glm_vec4_mul(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief multiply/scale vec4 vector with scalar: result = v * s + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec4s +glms_vec4_(scale)(vec4s v, float s) { + vec4s r; + glm_vec4_scale(v.raw, s, r.raw); + return r; +} + +/*! + * @brief make vec4 vector scale as specified: result = unit(v) * s + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec4s +glms_vec4_(scale_as)(vec4s v, float s) { + vec4s r; + glm_vec4_scale_as(v.raw, s, r.raw); + return r; +} + +/*! + * @brief div vector with another component-wise division: d = a / b + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns result = (a[0]/b[0], a[1]/b[1], a[2]/b[2], a[3]/b[3]) + */ +CGLM_INLINE +vec4s +glms_vec4_(div)(vec4s a, vec4s b) { + vec4s r; + glm_vec4_div(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief div vec4 vector with scalar: d = v / s + * + * @param[in] v vector + * @param[in] s scalar + * @returns destination vector + */ +CGLM_INLINE +vec4s +glms_vec4_(divs)(vec4s v, float s) { + vec4s r; + glm_vec4_divs(v.raw, s, r.raw); + return r; +} + +/*! + * @brief add two vectors and add result to sum + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest += (a + b) + */ +CGLM_INLINE +vec4s +glms_vec4_(addadd)(vec4s a, vec4s b, vec4s dest) { + glm_vec4_addadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief sub two vectors and add result to dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest += (a - b) + */ +CGLM_INLINE +vec4s +glms_vec4_(subadd)(vec4s a, vec4s b, vec4s dest) { + glm_vec4_subadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief mul two vectors and add result to dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest += (a * b) + */ +CGLM_INLINE +vec4s +glms_vec4_(muladd)(vec4s a, vec4s b, vec4s dest) { + glm_vec4_muladd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief mul vector with scalar and add result to sum + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @returns dest += (a * b) + */ +CGLM_INLINE +vec4s +glms_vec4_(muladds)(vec4s a, float s, vec4s dest) { + glm_vec4_muladds(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief add max of two vectors to result/dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest += max(a, b) + */ +CGLM_INLINE +vec4s +glms_vec4_(maxadd)(vec4s a, vec4s b, vec4s dest) { + glm_vec4_maxadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief add min of two vectors to result/dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest += min(a, b) + */ +CGLM_INLINE +vec4s +glms_vec4_(minadd)(vec4s a, vec4s b, vec4s dest) { + glm_vec4_minadd(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief sub two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest -= (a + b) + */ +CGLM_INLINE +vec4s +glms_vec4_(subsub)(vec4s a, vec4s b, vec4s dest) { + glm_vec4_subsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief add two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest -= (a + b) + */ +CGLM_INLINE +vec4s +glms_vec4_(addsub)(vec4s a, vec4s b, vec4s dest) { + glm_vec4_addsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief mul two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest -= (a * b) + */ +CGLM_INLINE +vec4s +glms_vec4_(mulsub)(vec4s a, vec4s b, vec4s dest) { + glm_vec4_mulsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief mul vector with scalar and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @returns dest -= (a * b) + */ +CGLM_INLINE +vec4s +glms_vec4_(mulsubs)(vec4s a, float s, vec4s dest) { + glm_vec4_mulsubs(a.raw, s, dest.raw); + return dest; +} + +/*! + * @brief sub max of two vectors to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest -= max(a, b) + */ +CGLM_INLINE +vec4s +glms_vec4_(maxsub)(vec4s a, vec4s b, vec4s dest) { + glm_vec4_maxsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief sub min of two vectors to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @returns dest -= min(a, b) + */ +CGLM_INLINE +vec4s +glms_vec4_(minsub)(vec4s a, vec4s b, vec4s dest) { + glm_vec4_minsub(a.raw, b.raw, dest.raw); + return dest; +} + +/*! + * @brief negate vector components and store result in dest + * + * @param[in] v vector + * @returns result vector + */ +CGLM_INLINE +vec4s +glms_vec4_(negate)(vec4s v) { + glm_vec4_negate(v.raw); + return v; +} + +/*! + * @brief normalize vec4 and store result in same vec + * + * @param[in] v vector + * @returns normalized vector + */ +CGLM_INLINE +vec4s +glms_vec4_(normalize)(vec4s v) { + glm_vec4_normalize(v.raw); + return v; +} + +/** + * @brief distance between two vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @return returns distance + */ +CGLM_INLINE +float +glms_vec4_(distance)(vec4s a, vec4s b) { + return glm_vec4_distance(a.raw, b.raw); +} + +/** + * @brief squared distance between two vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @return returns squared distance + */ +CGLM_INLINE +float +glms_vec4_(distance2)(vec4s a, vec4s b) { + return glm_vec4_distance2(a.raw, b.raw); +} + +/*! + * @brief max values of vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4_(maxv)(vec4s a, vec4s b) { + vec4s r; + glm_vec4_maxv(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief min values of vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4_(minv)(vec4s a, vec4s b) { + vec4s r; + glm_vec4_minv(a.raw, b.raw, r.raw); + return r; +} + +/*! + * @brief clamp vector's individual members between min and max values + * + * @param[in] v vector + * @param[in] minVal minimum value + * @param[in] maxVal maximum value + * @returns clamped vector + */ +CGLM_INLINE +vec4s +glms_vec4_(clamp)(vec4s v, float minVal, float maxVal) { + glm_vec4_clamp(v.raw, minVal, maxVal); + return v; +} + +/*! + * @brief linear interpolation between two vectors + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4_(lerp)(vec4s from, vec4s to, float t) { + vec4s r; + glm_vec4_lerp(from.raw, to.raw, t, r.raw); + return r; +} + +/*! + * @brief linear interpolation between two vectors (clamped) + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) clamped between 0 and 1 + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4_(lerpc)(vec4s from, vec4s to, float t) { + vec4s r; + glm_vec4_lerpc(from.raw, to.raw, t, r.raw); + return r; +} + +/*! + * @brief linear interpolation between two vectors + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4_(mix)(vec4s from, vec4s to, float t) { + vec4s r; + glm_vec4_mix(from.raw, to.raw, t, r.raw); + return r; +} + +/*! + * @brief linear interpolation between two vectors (clamped) + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) clamped between 0 and 1 + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4_(mixc)(vec4s from, vec4s to, float t) { + vec4s r; + glm_vec4_mixc(from.raw, to.raw, t, r.raw); + return r; +} + +/*! + * @brief threshold function + * + * @param[in] edge threshold + * @param[in] x value to test against threshold + * @returns 0.0 if x < edge, else 1.0 + */ +CGLM_INLINE +vec4s +glms_vec4_(step)(vec4s edge, vec4s x) { + vec4s r; + glm_vec4_step(edge.raw, x.raw, r.raw); + return r; +} + +/*! + * @brief threshold function with a smooth transition (unidimensional) + * + * @param[in] edge0 low threshold + * @param[in] edge1 high threshold + * @param[in] x value to test against threshold + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4_(smoothstep_uni)(float edge0, float edge1, vec4s x) { + vec4s r; + glm_vec4_smoothstep_uni(edge0, edge1, x.raw, r.raw); + return r; +} + +/*! + * @brief threshold function with a smooth transition + * + * @param[in] edge0 low threshold + * @param[in] edge1 high threshold + * @param[in] x value to test against threshold + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4_(smoothstep)(vec4s edge0, vec4s edge1, vec4s x) { + vec4s r; + glm_vec4_smoothstep(edge0.raw, edge1.raw, x.raw, r.raw); + return r; +} + +/*! + * @brief smooth Hermite interpolation between two vectors + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4_(smoothinterp)(vec4s from, vec4s to, float t) { + vec4s r; + glm_vec4_smoothinterp(from.raw, to.raw, t, r.raw); + return r; +} + +/*! + * @brief smooth Hermite interpolation between two vectors (clamped) + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) clamped between 0 and 1 + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4_(smoothinterpc)(vec4s from, vec4s to, float t) { + vec4s r; + glm_vec4_smoothinterpc(from.raw, to.raw, t, r.raw); + return r; +} + +/*! + * @brief helper to fill vec4 as [S^3, S^2, S, 1] + * + * @param[in] s parameter + * @returns destination + */ +CGLM_INLINE +vec4s +glms_vec4_(cubic)(float s) { + vec4s r; + glm_vec4_cubic(s, r.raw); + return r; +} + +/*! + * @brief swizzle vector components + * + * you can use existing masks e.g. GLM_XXXX, GLM_WZYX + * + * @param[in] v source + * @param[in] mask mask + * @returns swizzled vector + */ +CGLM_INLINE +vec4s +glms_vec4_(swizzle)(vec4s v, int mask) { + vec4s dest; + glm_vec4_swizzle(v.raw, mask, dest.raw); + return dest; +} + +/*! + * @brief Create four dimensional vector from pointer + * + * @param[in] src pointer to an array of floats + * @returns constructed 4D vector from raw pointer + */ +CGLM_INLINE +vec4s +glms_vec4_(make)(const float * __restrict src) { + vec4s dest; + glm_vec4_make(src, dest.raw); + return dest; +} + +/*! + * @brief reflection vector using an incident ray and a surface normal + * + * @param[in] v incident vector + * @param[in] n normalized normal vector + * @returns reflection result + */ +CGLM_INLINE +vec4s +glms_vec4_(reflect)(vec4s v, vec4s n) { + vec4s dest; + glm_vec4_reflect(v.raw, n.raw, dest.raw); + return dest; +} + +/*! + * @brief computes refraction vector for an incident vector and a surface normal. + * + * calculates the refraction vector based on Snell's law. If total internal reflection + * occurs (angle too great given eta), dest is set to zero and returns false. + * Otherwise, computes refraction vector, stores it in dest, and returns true. + * + * this implementation does not explicitly preserve the 'w' component of the + * incident vector 'I' in the output 'dest', users requiring the preservation of + * the 'w' component should manually adjust 'dest' after calling this function. + * + * @param[in] v normalized incident vector + * @param[in] n normalized normal vector + * @param[in] eta ratio of indices of refraction (incident/transmitted) + * @param[out] dest refraction vector if refraction occurs; zero vector otherwise + * + * @returns true if refraction occurs; false if total internal reflection occurs. + */ +CGLM_INLINE +bool +glms_vec4_(refract)(vec4s v, vec4s n, float eta, vec4s * __restrict dest) { + return glm_vec4_refract(v.raw, n.raw, eta, dest->raw); +} + +#endif /* cglms_vec4s_h */ diff --git a/cglm/include/cglm/types-struct.h b/cglm/include/cglm/types-struct.h new file mode 100644 index 0000000..d93152e --- /dev/null +++ b/cglm/include/cglm/types-struct.h @@ -0,0 +1,303 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_types_struct_h +#define cglm_types_struct_h + +#include "types.h" + +/* + * Anonymous structs are available since C11, but we'd like to be compatible + * with C99 and C89 too. So let's figure out if we should be using them or not. + * It's simply a convenience feature, you can e.g. build the library with + * anonymous structs and your application without them and they'll still be + * compatible, cglm doesn't use the anonymous structs internally. + */ +#ifndef CGLM_USE_ANONYMOUS_STRUCT + /* If the user doesn't explicitly specify if they want anonymous structs or + * not, then we'll try to intuit an appropriate choice. */ +# if defined(CGLM_NO_ANONYMOUS_STRUCT) + /* The user has defined CGLM_NO_ANONYMOUS_STRUCT. This used to be the + * only #define governing the use of anonymous structs, so for backward + * compatibility, we still honor that choice and disable them. */ +# define CGLM_USE_ANONYMOUS_STRUCT 0 + /* Disable anonymous structs if strict ANSI mode is enabled for C89 or C99 */ +# elif defined(__STRICT_ANSI__) && \ + (!defined(__STDC_VERSION__) || (__STDC_VERSION__ < 201112L)) + /* __STRICT_ANSI__ is defined and we're in C89 + * or C99 mode (C11 or later not detected) */ +# define CGLM_USE_ANONYMOUS_STRUCT 0 +# elif (defined(__STDC_VERSION__) && __STDC_VERSION__ >= 201112L) || \ + (defined(__cplusplus) && __cplusplus >= 201103L) + /* We're compiling for C11 or this is the MSVC compiler. In either + * case, anonymous structs are available, so use them. */ +# define CGLM_USE_ANONYMOUS_STRUCT 1 +# elif defined(__GNUC__) && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 6)) + /* GCC 4.6 and onwards support anonymous structs as an extension */ +# define CGLM_USE_ANONYMOUS_STRUCT 1 +# elif defined(__clang__) && __clang_major__ >= 3 + /* Clang 3.0 and onwards support anonymous structs as an extension */ +# define CGLM_USE_ANONYMOUS_STRUCT 1 +# elif defined(_MSC_VER) && (_MSC_VER >= 1900) /* Visual Studio 2015 */ + /* We can support anonymous structs + * since Visual Studio 2015 or 2017 (1910) maybe? */ +# define CGLM_USE_ANONYMOUS_STRUCT 1 +# else + /* Otherwise, we're presumably building for C99 or C89 and can't rely + * on anonymous structs being available. Turn them off. */ +# define CGLM_USE_ANONYMOUS_STRUCT 0 +# endif +#endif + +typedef union vec2s { + vec2 raw; +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + float x; + float y; + }; + + struct { + float r; + float i; + }; + + struct { + float u; + float v; + }; + + struct { + float s; + float t; + }; +#endif +} vec2s; + +typedef union vec3s { + vec3 raw; +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + float x; + float y; + float z; + }; + + struct { + float r; + float g; + float b; + }; +#endif +} vec3s; + +typedef union ivec2s { + ivec2 raw; +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + int x; + int y; + }; + + struct { + int r; + int i; + }; + + struct { + int u; + int v; + }; + + struct { + int s; + int t; + }; +#endif +} ivec2s; + +typedef union ivec3s { + ivec3 raw; +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + int x; + int y; + int z; + }; + + struct { + int r; + int g; + int b; + }; +#endif +} ivec3s; + +typedef union ivec4s { + ivec4 raw; +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + int x; + int y; + int z; + int w; + }; + + struct { + int r; + int g; + int b; + int a; + }; +#endif +} ivec4s; + +typedef union CGLM_ALIGN_IF(16) vec4s { + vec4 raw; +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + float x; + float y; + float z; + float w; + }; + + struct { + float r; + float g; + float b; + float a; + }; +#endif +} vec4s; + +typedef union CGLM_ALIGN_IF(16) versors { + vec4 raw; +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + float x; + float y; + float z; + float w; + }; + + struct { + vec3s imag; + float real; + }; +#endif +} versors; + +typedef union mat2s { + mat2 raw; + vec2s col[2]; +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + float m00, m01; + float m10, m11; + }; +#endif +} mat2s; + +typedef union mat2x3s { + mat2x3 raw; + vec3s col[2]; /* [col (2), row (3)] */ +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + float m00, m01, m02; + float m10, m11, m12; + }; +#endif +} mat2x3s; + +typedef union mat2x4s { + mat2x4 raw; + vec4s col[2]; /* [col (2), row (4)] */ +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + float m00, m01, m02, m03; + float m10, m11, m12, m13; + }; +#endif +} mat2x4s; + +typedef union mat3s { + mat3 raw; + vec3s col[3]; +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + float m00, m01, m02; + float m10, m11, m12; + float m20, m21, m22; + }; +#endif +} mat3s; + +typedef union mat3x2s { + mat3x2 raw; + vec2s col[3]; /* [col (3), row (2)] */ +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + float m00, m01; + float m10, m11; + float m20, m21; + }; +#endif +} mat3x2s; + +typedef union mat3x4s { + mat3x4 raw; + vec4s col[3]; /* [col (3), row (4)] */ +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + float m00, m01, m02, m03; + float m10, m11, m12, m13; + float m20, m21, m22, m23; + }; +#endif +} mat3x4s; + +typedef union CGLM_ALIGN_MAT mat4s { + mat4 raw; + vec4s col[4]; +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + float m00, m01, m02, m03; + float m10, m11, m12, m13; + float m20, m21, m22, m23; + float m30, m31, m32, m33; + }; +#endif +} mat4s; + +typedef union mat4x2s { + mat4x2 raw; + vec2s col[4]; /* [col (4), row (2)] */ +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + float m00, m01; + float m10, m11; + float m20, m21; + float m30, m31; + }; +#endif +} mat4x2s; + +typedef union mat4x3s { + mat4x3 raw; + vec3s col[4]; /* [col (4), row (3)] */ +#if CGLM_USE_ANONYMOUS_STRUCT + struct { + float m00, m01, m02; + float m10, m11, m12; + float m20, m21, m22; + float m30, m31, m32; + }; +#endif +} mat4x3s; + +#endif /* cglm_types_struct_h */ diff --git a/cglm/include/cglm/types.h b/cglm/include/cglm/types.h new file mode 100644 index 0000000..26e6467 --- /dev/null +++ b/cglm/include/cglm/types.h @@ -0,0 +1,132 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_types_h +#define cglm_types_h + +#if (defined(__STDC_VERSION__) && __STDC_VERSION__ >= 201112L) +# include +#endif + +#if defined(_MSC_VER) +/* do not use alignment for older visual studio versions */ +/* also ARM32 also causes similar error, disable it for now on ARM32 too */ +# if _MSC_VER < 1913 || _M_ARM /* Visual Studio 2017 version 15.6 */ +# define CGLM_ALL_UNALIGNED +# define CGLM_ALIGN(X) /* no alignment */ +# else +# define CGLM_ALIGN(X) __declspec(align(X)) +# endif +#else +# define CGLM_ALIGN(X) __attribute((aligned(X))) +#endif + +#ifndef CGLM_ALL_UNALIGNED +# define CGLM_ALIGN_IF(X) CGLM_ALIGN(X) +#else +# define CGLM_ALIGN_IF(X) /* no alignment */ +#endif + +#ifdef __AVX__ +# define CGLM_ALIGN_MAT CGLM_ALIGN(32) +#else +# define CGLM_ALIGN_MAT CGLM_ALIGN(16) +#endif + +#ifndef CGLM_HAVE_BUILTIN_ASSUME_ALIGNED + +# if defined(__has_builtin) +# if __has_builtin(__builtin_assume_aligned) +# define CGLM_HAVE_BUILTIN_ASSUME_ALIGNED 1 +# endif +# elif defined(__GNUC__) && defined(__GNUC_MINOR__) +# if __GNUC__ >= 4 && __GNUC_MINOR__ >= 7 +# define CGLM_HAVE_BUILTIN_ASSUME_ALIGNED 1 +# endif +# endif + +# ifndef CGLM_HAVE_BUILTIN_ASSUME_ALIGNED +# define CGLM_HAVE_BUILTIN_ASSUME_ALIGNED 0 +# endif + +#endif + +#if CGLM_HAVE_BUILTIN_ASSUME_ALIGNED +# define CGLM_ASSUME_ALIGNED(expr, alignment) \ + __builtin_assume_aligned((expr), (alignment)) +#else +# define CGLM_ASSUME_ALIGNED(expr, alignment) (expr) +#endif + +#if (defined(__STDC_VERSION__) && __STDC_VERSION__ >= 201112L) +# define CGLM_CASTPTR_ASSUME_ALIGNED(expr, type) \ + ((type*)CGLM_ASSUME_ALIGNED((expr), alignof(type))) +#elif defined(_MSC_VER) +# define CGLM_CASTPTR_ASSUME_ALIGNED(expr, type) \ + ((type*)CGLM_ASSUME_ALIGNED((expr), __alignof(type))) +#else +# define CGLM_CASTPTR_ASSUME_ALIGNED(expr, type) \ + ((type*)CGLM_ASSUME_ALIGNED((expr), __alignof__(type))) +#endif + +typedef int ivec2[2]; +typedef int ivec3[3]; +typedef int ivec4[4]; + +typedef float vec2[2]; +typedef float vec3[3]; +typedef CGLM_ALIGN_IF(16) float vec4[4]; +typedef vec4 versor; /* |x, y, z, w| -> w is the last */ +typedef vec3 mat3[3]; +typedef vec2 mat3x2[3]; /* [col (3), row (2)] */ +typedef vec4 mat3x4[3]; /* [col (3), row (4)] */ +typedef CGLM_ALIGN_IF(16) vec2 mat2[2]; +typedef vec3 mat2x3[2]; /* [col (2), row (3)] */ +typedef vec4 mat2x4[2]; /* [col (2), row (4)] */ +typedef CGLM_ALIGN_MAT vec4 mat4[4]; +typedef vec2 mat4x2[4]; /* [col (4), row (2)] */ +typedef vec3 mat4x3[4]; /* [col (4), row (3)] */ + +/* + Important: cglm stores quaternion as [x, y, z, w] in memory since v0.4.0 + it was [w, x, y, z] before v0.4.0 ( v0.3.5 and earlier ). w is real part. +*/ + +#define GLM_E 2.71828182845904523536028747135266250 /* e */ +#define GLM_LOG2E 1.44269504088896340735992468100189214 /* log2(e) */ +#define GLM_LOG10E 0.434294481903251827651128918916605082 /* log10(e) */ +#define GLM_LN2 0.693147180559945309417232121458176568 /* loge(2) */ +#define GLM_LN10 2.30258509299404568401799145468436421 /* loge(10) */ +#define GLM_PI 3.14159265358979323846264338327950288 /* pi */ +#define GLM_PI_2 1.57079632679489661923132169163975144 /* pi/2 */ +#define GLM_PI_4 0.785398163397448309615660845819875721 /* pi/4 */ +#define GLM_1_PI 0.318309886183790671537767526745028724 /* 1/pi */ +#define GLM_2_PI 0.636619772367581343075535053490057448 /* 2/pi */ +#define GLM_2_SQRTPI 1.12837916709551257389615890312154517 /* 2/sqrt(pi) */ +#define GLM_SQRT2 1.41421356237309504880168872420969808 /* sqrt(2) */ +#define GLM_SQRT1_2 0.707106781186547524400844362104849039 /* 1/sqrt(2) */ + +#define GLM_Ef ((float)GLM_E) +#define GLM_LOG2Ef ((float)GLM_LOG2E) +#define GLM_LOG10Ef ((float)GLM_LOG10E) +#define GLM_LN2f ((float)GLM_LN2) +#define GLM_LN10f ((float)GLM_LN10) +#define GLM_PIf ((float)GLM_PI) +#define GLM_PI_2f ((float)GLM_PI_2) +#define GLM_PI_4f ((float)GLM_PI_4) +#define GLM_1_PIf ((float)GLM_1_PI) +#define GLM_2_PIf ((float)GLM_2_PI) +#define GLM_2_SQRTPIf ((float)GLM_2_SQRTPI) +#define GLM_SQRT2f ((float)GLM_SQRT2) +#define GLM_SQRT1_2f ((float)GLM_SQRT1_2) + +/* DEPRECATED! use GLM_PI and friends */ +#define CGLM_PI GLM_PIf +#define CGLM_PI_2 GLM_PI_2f +#define CGLM_PI_4 GLM_PI_4f + +#endif /* cglm_types_h */ diff --git a/cglm/include/cglm/util.h b/cglm/include/cglm/util.h new file mode 100644 index 0000000..8c5f2cb --- /dev/null +++ b/cglm/include/cglm/util.h @@ -0,0 +1,375 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE int glm_sign(int val); + CGLM_INLINE float glm_signf(float val); + CGLM_INLINE float glm_rad(float deg); + CGLM_INLINE float glm_deg(float rad); + CGLM_INLINE void glm_make_rad(float *deg); + CGLM_INLINE void glm_make_deg(float *rad); + CGLM_INLINE float glm_pow2(float x); + CGLM_INLINE float glm_min(float a, float b); + CGLM_INLINE float glm_max(float a, float b); + CGLM_INLINE float glm_clamp(float val, float minVal, float maxVal); + CGLM_INLINE float glm_clamp_zo(float val, float minVal, float maxVal); + CGLM_INLINE float glm_lerp(float from, float to, float t); + CGLM_INLINE float glm_lerpc(float from, float to, float t); + CGLM_INLINE float glm_step(float edge, float x); + CGLM_INLINE float glm_smooth(float t); + CGLM_INLINE float glm_smoothstep(float edge0, float edge1, float x); + CGLM_INLINE float glm_smoothinterp(float from, float to, float t); + CGLM_INLINE float glm_smoothinterpc(float from, float to, float t); + CGLM_INLINE bool glm_eq(float a, float b); + CGLM_INLINE float glm_percent(float from, float to, float current); + CGLM_INLINE float glm_percentc(float from, float to, float current); + */ + +#ifndef cglm_util_h +#define cglm_util_h + +#include "common.h" + +#define GLM_MIN(X, Y) (((X) < (Y)) ? (X) : (Y)) +#define GLM_MAX(X, Y) (((X) > (Y)) ? (X) : (Y)) + +/*! + * @brief get sign of 32 bit integer as +1, -1, 0 + * + * Important: It returns 0 for zero input + * + * @param val integer value + */ +CGLM_INLINE +int +glm_sign(int val) { + return ((val >> 31) - (-val >> 31)); +} + +/*! + * @brief get sign of 32 bit float as +1, -1, 0 + * + * Important: It returns 0 for zero/NaN input + * + * @param val float value + */ +CGLM_INLINE +float +glm_signf(float val) { + return (float)((val > 0.0f) - (val < 0.0f)); +} + +/*! + * @brief convert degree to radians + * + * @param[in] deg angle in degrees + */ +CGLM_INLINE +float +glm_rad(float deg) { + return deg * GLM_PIf / 180.0f; +} + +/*! + * @brief convert radians to degree + * + * @param[in] rad angle in radians + */ +CGLM_INLINE +float +glm_deg(float rad) { + return rad * 180.0f / GLM_PIf; +} + +/*! + * @brief convert existing degree to radians. this will override degrees value + * + * @param[in, out] deg pointer to angle in degrees + */ +CGLM_INLINE +void +glm_make_rad(float *deg) { + *deg = *deg * GLM_PIf / 180.0f; +} + +/*! + * @brief convert existing radians to degree. this will override radians value + * + * @param[in, out] rad pointer to angle in radians + */ +CGLM_INLINE +void +glm_make_deg(float *rad) { + *rad = *rad * 180.0f / GLM_PIf; +} + +/*! + * @brief multiplies given parameter with itself = x * x or powf(x, 2) + * + * @param[in] x x + */ +CGLM_INLINE +float +glm_pow2(float x) { + return x * x; +} + +/*! + * @brief find minimum of given two values + * + * @param[in] a number 1 + * @param[in] b number 2 + */ +CGLM_INLINE +float +glm_min(float a, float b) { + if (a < b) + return a; + return b; +} + +/*! + * @brief find maximum of given two values + * + * @param[in] a number 1 + * @param[in] b number 2 + */ +CGLM_INLINE +float +glm_max(float a, float b) { + if (a > b) + return a; + return b; +} + +/*! + * @brief find minimum of given two values + * + * @param[in] a number 1 + * @param[in] b number 2 + * + * @return smallest of the two values + */ +CGLM_INLINE +int +glm_imin(int a, int b) { + if (a < b) + return a; + return b; +} + +/*! + * @brief find maximum of given two values + * + * @param[in] a number 1 + * @param[in] b number 2 + * + * @return largest of the two values + */ +CGLM_INLINE +int +glm_imax(int a, int b) { + if (a > b) + return a; + return b; +} + +/*! + * @brief clamp a number between min and max + * + * @param[in] val value to clamp + * @param[in] minVal minimum value + * @param[in] maxVal maximum value + */ +CGLM_INLINE +float +glm_clamp(float val, float minVal, float maxVal) { + return glm_min(glm_max(val, minVal), maxVal); +} + +/*! + * @brief clamp a number to zero and one + * + * @param[in] val value to clamp + */ +CGLM_INLINE +float +glm_clamp_zo(float val) { + return glm_clamp(val, 0.0f, 1.0f); +} + +/*! + * @brief linear interpolation between two numbers + * + * formula: from + t * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) + */ +CGLM_INLINE +float +glm_lerp(float from, float to, float t) { + return from + t * (to - from); +} + +/*! + * @brief clamped linear interpolation between two numbers + * + * formula: from + t * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) clamped between 0 and 1 + */ +CGLM_INLINE +float +glm_lerpc(float from, float to, float t) { + return glm_lerp(from, to, glm_clamp_zo(t)); +} + +/*! + * @brief threshold function + * + * @param[in] edge threshold + * @param[in] x value to test against threshold + * @return returns 0.0 if x < edge, else 1.0 + */ +CGLM_INLINE +float +glm_step(float edge, float x) { + /* branching - no type conversion */ + return (x < edge) ? 0.0f : 1.0f; + /* + * An alternative implementation without branching + * but with type conversion could be: + * return !(x < edge); + */ +} + +/*! + * @brief smooth Hermite interpolation + * + * formula: t^2 * (3-2t) + * + * @param[in] t interpolant (amount) + */ +CGLM_INLINE +float +glm_smooth(float t) { + return t * t * (3.0f - 2.0f * t); +} + +/*! + * @brief threshold function with a smooth transition (according to OpenCL specs) + * + * formula: t^2 * (3-2t) + * + * @param[in] edge0 low threshold + * @param[in] edge1 high threshold + * @param[in] x interpolant (amount) + */ +CGLM_INLINE +float +glm_smoothstep(float edge0, float edge1, float x) { + float t; + t = glm_clamp_zo((x - edge0) / (edge1 - edge0)); + return glm_smooth(t); +} + +/*! + * @brief smoothstep interpolation between two numbers + * + * formula: from + smoothstep(t) * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) + */ +CGLM_INLINE +float +glm_smoothinterp(float from, float to, float t) { + return from + glm_smooth(t) * (to - from); +} + +/*! + * @brief clamped smoothstep interpolation between two numbers + * + * formula: from + smoothstep(t) * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) clamped between 0 and 1 + */ +CGLM_INLINE +float +glm_smoothinterpc(float from, float to, float t) { + return glm_smoothinterp(from, to, glm_clamp_zo(t)); +} + +/*! + * @brief check if two float equal with using EPSILON + * + * @param[in] a a + * @param[in] b b + */ +CGLM_INLINE +bool +glm_eq(float a, float b) { + return fabsf(a - b) <= GLM_FLT_EPSILON; +} + +/*! + * @brief percentage of current value between start and end value + * + * maybe fraction could be alternative name. + * + * @param[in] from from value + * @param[in] to to value + * @param[in] current current value + */ +CGLM_INLINE +float +glm_percent(float from, float to, float current) { + float t; + + if ((t = to - from) == 0.0f) + return 1.0f; + + return (current - from) / t; +} + +/*! + * @brief clamped percentage of current value between start and end value + * + * @param[in] from from value + * @param[in] to to value + * @param[in] current current value + */ +CGLM_INLINE +float +glm_percentc(float from, float to, float current) { + return glm_clamp_zo(glm_percent(from, to, current)); +} + +/*! +* @brief swap two float values +* +* @param[in] a float value 1 (pointer) +* @param[in] b float value 2 (pointer) +*/ +CGLM_INLINE +void +glm_swapf(float * __restrict a, float * __restrict b) { + float t; + t = *a; + *a = *b; + *b = t; +} + +#endif /* cglm_util_h */ diff --git a/cglm/include/cglm/vec2-ext.h b/cglm/include/cglm/vec2-ext.h new file mode 100644 index 0000000..6186f07 --- /dev/null +++ b/cglm/include/cglm/vec2-ext.h @@ -0,0 +1,337 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Functions: + CGLM_INLINE void glm_vec2_fill(vec2 v, float val) + CGLM_INLINE bool glm_vec2_eq(vec2 v, float val); + CGLM_INLINE bool glm_vec2_eq_eps(vec2 v, float val); + CGLM_INLINE bool glm_vec2_eq_all(vec2 v); + CGLM_INLINE bool glm_vec2_eqv(vec2 a, vec2 b); + CGLM_INLINE bool glm_vec2_eqv_eps(vec2 a, vec2 b); + CGLM_INLINE float glm_vec2_max(vec2 v); + CGLM_INLINE float glm_vec2_min(vec2 v); + CGLM_INLINE bool glm_vec2_isnan(vec2 v); + CGLM_INLINE bool glm_vec2_isinf(vec2 v); + CGLM_INLINE bool glm_vec2_isvalid(vec2 v); + CGLM_INLINE void glm_vec2_sign(vec2 v, vec2 dest); + CGLM_INLINE void glm_vec2_abs(vec2 v, vec2 dest); + CGLM_INLINE void glm_vec2_fract(vec2 v, vec2 dest); + CGLM_INLINE void glm_vec2_floor(vec2 v, vec2 dest); + CGLM_INLINE float glm_vec2_mods(vec2 v, float s, vec2 dest); + CGLM_INLINE float glm_vec2_steps(float edge, vec2 v, vec2 dest); + CGLM_INLINE void glm_vec2_stepr(vec2 edge, float v, vec2 dest); + CGLM_INLINE void glm_vec2_sqrt(vec2 v, vec2 dest); + CGLM_INLINE void glm_vec2_complex_mul(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE void glm_vec2_complex_div(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE void glm_vec2_complex_conjugate(vec2 a, vec2 dest) + */ + +#ifndef cglm_vec2_ext_h +#define cglm_vec2_ext_h + +#include "common.h" +#include "util.h" + +/*! + * @brief fill a vector with specified value + * + * @param[out] v dest + * @param[in] val value + */ +CGLM_INLINE +void +glm_vec2_fill(vec2 v, float val) { + v[0] = v[1] = val; +} + +/*! + * @brief check if vector is equal to value (without epsilon) + * + * @param[in] v vector + * @param[in] val value + */ +CGLM_INLINE +bool +glm_vec2_eq(vec2 v, float val) { + return v[0] == val && v[0] == v[1]; +} + +/*! + * @brief check if vector is equal to value (with epsilon) + * + * @param[in] v vector + * @param[in] val value + */ +CGLM_INLINE +bool +glm_vec2_eq_eps(vec2 v, float val) { + return fabsf(v[0] - val) <= GLM_FLT_EPSILON + && fabsf(v[1] - val) <= GLM_FLT_EPSILON; +} + +/*! + * @brief check if vector members are equal (without epsilon) + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glm_vec2_eq_all(vec2 v) { + return glm_vec2_eq_eps(v, v[0]); +} + +/*! + * @brief check if vector is equal to another (without epsilon) + * + * @param[in] a vector + * @param[in] b vector + */ +CGLM_INLINE +bool +glm_vec2_eqv(vec2 a, vec2 b) { + return a[0] == b[0] && a[1] == b[1]; +} + +/*! + * @brief check if vector is equal to another (with epsilon) + * + * @param[in] a vector + * @param[in] b vector + */ +CGLM_INLINE +bool +glm_vec2_eqv_eps(vec2 a, vec2 b) { + return fabsf(a[0] - b[0]) <= GLM_FLT_EPSILON + && fabsf(a[1] - b[1]) <= GLM_FLT_EPSILON; +} + +/*! + * @brief max value of vector + * + * @param[in] v vector + */ +CGLM_INLINE +float +glm_vec2_max(vec2 v) { + return glm_max(v[0], v[1]); +} + +/*! + * @brief min value of vector + * + * @param[in] v vector + */ +CGLM_INLINE +float +glm_vec2_min(vec2 v) { + return glm_min(v[0], v[1]); +} + +/*! + * @brief check if one of items is NaN (not a number) + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glm_vec2_isnan(vec2 v) { +#ifndef CGLM_FAST_MATH + return isnan(v[0]) || isnan(v[1]); +#else + return false; +#endif +} + +/*! + * @brief check if one of items is INFINITY + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glm_vec2_isinf(vec2 v) { +#ifndef CGLM_FAST_MATH + return isinf(v[0]) || isinf(v[1]); +#else + return false; +#endif +} + +/*! + * @brief check if all items are valid number + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glm_vec2_isvalid(vec2 v) { + return !glm_vec2_isnan(v) && !glm_vec2_isinf(v); +} + +/*! + * @brief get sign of 32 bit float as +1, -1, 0 + * + * Important: It returns 0 for zero/NaN input + * + * @param v vector + */ +CGLM_INLINE +void +glm_vec2_sign(vec2 v, vec2 dest) { + dest[0] = glm_signf(v[0]); + dest[1] = glm_signf(v[1]); +} + +/*! + * @brief absolute value of v + * + * @param[in] v vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec2_abs(vec2 v, vec2 dest) { + dest[0] = fabsf(v[0]); + dest[1] = fabsf(v[1]); +} + +/*! + * @brief fractional part of each vector item + * + * @param[in] v vector + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec2_fract(vec2 v, vec2 dest) { + dest[0] = fminf(v[0] - floorf(v[0]), 0.999999940395355224609375f); + dest[1] = fminf(v[1] - floorf(v[1]), 0.999999940395355224609375f); +} + +/*! + * @brief floor of each vector item + * + * @param[in] v vector + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec2_floor(vec2 v, vec2 dest) { + dest[0] = floorf(v[0]); + dest[1] = floorf(v[1]); +} + +/*! + * @brief mod of each vector item, result is written to dest (dest = v % s) + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec2_mods(vec2 v, float s, vec2 dest) { + dest[0] = fmodf(v[0], s); + dest[1] = fmodf(v[1], s); +} + +/*! + * @brief square root of each vector item + * + * @param[in] v vector + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec2_sqrt(vec2 v, vec2 dest) { + dest[0] = sqrtf(v[0]); + dest[1] = sqrtf(v[1]); +} + +/*! + * @brief treat vectors as complex numbers and multiply them as such. + * + * @param[in] a left number + * @param[in] b right number + * @param[out] dest destination number + */ +CGLM_INLINE +void +glm_vec2_complex_mul(vec2 a, vec2 b, vec2 dest) { + float tr, ti; + tr = a[0] * b[0] - a[1] * b[1]; + ti = a[0] * b[1] + a[1] * b[0]; + dest[0] = tr; + dest[1] = ti; +} + +/*! + * @brief threshold each vector item with scalar + * condition is: (x[i] < edge) ? 0.0 : 1.0 + * + * @param[in] edge threshold + * @param[in] x vector to test against threshold + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec2_steps(float edge, vec2 x, vec2 dest) { + dest[0] = glm_step(edge, x[0]); + dest[1] = glm_step(edge, x[1]); +} + +/*! + * @brief threshold a value with *vector* as the threshold + * condition is: (x < edge[i]) ? 0.0 : 1.0 + * + * @param[in] edge threshold vector + * @param[in] x value to test against threshold + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec2_stepr(vec2 edge, float x, vec2 dest) { + dest[0] = glm_step(edge[0], x); + dest[1] = glm_step(edge[1], x); +} + +/*! + * @brief treat vectors as complex numbers and divide them as such. + * + * @param[in] a left number (numerator) + * @param[in] b right number (denominator) + * @param[out] dest destination number + */ +CGLM_INLINE +void +glm_vec2_complex_div(vec2 a, vec2 b, vec2 dest) { + float tr, ti; + float const ibnorm2 = 1.0f / (b[0] * b[0] + b[1] * b[1]); + tr = ibnorm2 * (a[0] * b[0] + a[1] * b[1]); + ti = ibnorm2 * (a[1] * b[0] - a[0] * b[1]); + dest[0] = tr; + dest[1] = ti; +} + +/*! + * @brief treat the vector as a complex number and conjugate it as such. + * + * @param[in] a the number + * @param[out] dest destination number + */ +CGLM_INLINE +void +glm_vec2_complex_conjugate(vec2 a, vec2 dest) { + dest[0] = a[0]; + dest[1] = -a[1]; +} + +#endif /* cglm_vec2_ext_h */ diff --git a/cglm/include/cglm/vec2.h b/cglm/include/cglm/vec2.h new file mode 100644 index 0000000..655fb4b --- /dev/null +++ b/cglm/include/cglm/vec2.h @@ -0,0 +1,798 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_VEC2_ONE_INIT + GLM_VEC2_ZERO_INIT + GLM_VEC2_ONE + GLM_VEC2_ZERO + + Functions: + CGLM_INLINE void glm_vec2(float * __restrict v, vec2 dest) + CGLM_INLINE void glm_vec2_copy(vec2 a, vec2 dest) + CGLM_INLINE void glm_vec2_zero(vec2 v) + CGLM_INLINE void glm_vec2_one(vec2 v) + CGLM_INLINE float glm_vec2_dot(vec2 a, vec2 b) + CGLM_INLINE float glm_vec2_cross(vec2 a, vec2 b) + CGLM_INLINE float glm_vec2_norm2(vec2 v) + CGLM_INLINE float glm_vec2_norm(vec2 vec) + CGLM_INLINE void glm_vec2_add(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE void glm_vec2_adds(vec2 v, float s, vec2 dest) + CGLM_INLINE void glm_vec2_sub(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE void glm_vec2_subs(vec2 v, float s, vec2 dest) + CGLM_INLINE void glm_vec2_mul(vec2 a, vec2 b, vec2 d) + CGLM_INLINE void glm_vec2_scale(vec2 v, float s, vec2 dest) + CGLM_INLINE void glm_vec2_scale_as(vec2 v, float s, vec2 dest) + CGLM_INLINE void glm_vec2_div(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE void glm_vec2_divs(vec2 v, float s, vec2 dest) + CGLM_INLINE void glm_vec2_addadd(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE void glm_vec2_subadd(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE void glm_vec2_muladd(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE void glm_vec2_muladds(vec2 a, float s, vec2 dest) + CGLM_INLINE void glm_vec2_maxadd(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE void glm_vec2_minadd(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE void glm_vec2_subsub(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE void glm_vec2_addsub(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE void glm_vec2_mulsub(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE void glm_vec2_mulsubs(vec2 a, float s, vec2 dest) + CGLM_INLINE void glm_vec2_maxsub(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE void glm_vec2_minsub(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE void glm_vec2_negate_to(vec2 v, vec2 dest) + CGLM_INLINE void glm_vec2_negate(vec2 v) + CGLM_INLINE void glm_vec2_normalize(vec2 v) + CGLM_INLINE void glm_vec2_normalize_to(vec2 vec, vec2 dest) + CGLM_INLINE void glm_vec2_rotate(vec2 v, float angle, vec2 dest) + CGLM_INLINE void glm_vec2_center(vec2 a, vec2 b, vec2 dest) + CGLM_INLINE float glm_vec2_distance2(vec2 a, vec2 b) + CGLM_INLINE float glm_vec2_distance(vec2 a, vec2 b) + CGLM_INLINE void glm_vec2_maxv(vec2 v1, vec2 v2, vec2 dest) + CGLM_INLINE void glm_vec2_minv(vec2 v1, vec2 v2, vec2 dest) + CGLM_INLINE void glm_vec2_clamp(vec2 v, float minVal, float maxVal) + CGLM_INLINE void glm_vec2_swizzle(vec2 v, int mask, vec2 dest) + CGLM_INLINE void glm_vec2_lerp(vec2 from, vec2 to, float t, vec2 dest) + CGLM_INLINE void glm_vec2_step(vec2 edge, vec2 x, vec2 dest) + CGLM_INLINE void glm_vec2_make(float * restrict src, vec2 dest) + CGLM_INLINE void glm_vec2_reflect(vec2 v, vec2 n, vec2 dest) + CGLM_INLINE void glm_vec2_refract(vec2 v, vec2 n, float eta, vec2 dest) + */ + +#ifndef cglm_vec2_h +#define cglm_vec2_h + +#include "common.h" +#include "util.h" +#include "vec2-ext.h" + +#define GLM_VEC2_ONE_INIT {1.0f, 1.0f} +#define GLM_VEC2_ZERO_INIT {0.0f, 0.0f} + +#define GLM_VEC2_ONE ((vec2)GLM_VEC2_ONE_INIT) +#define GLM_VEC2_ZERO ((vec2)GLM_VEC2_ZERO_INIT) + +/*! + * @brief init vec2 using another vector + * + * @param[in] v a vector + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec2(float * __restrict v, vec2 dest) { + dest[0] = v[0]; + dest[1] = v[1]; +} + +/*! + * @brief copy all members of [a] to [dest] + * + * @param[in] a source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec2_copy(vec2 a, vec2 dest) { + dest[0] = a[0]; + dest[1] = a[1]; +} + +/*! + * @brief make vector zero + * + * @param[in, out] v vector + */ +CGLM_INLINE +void +glm_vec2_zero(vec2 v) { + v[0] = v[1] = 0.0f; +} + +/*! + * @brief make vector one + * + * @param[in, out] v vector + */ +CGLM_INLINE +void +glm_vec2_one(vec2 v) { + v[0] = v[1] = 1.0f; +} + +/*! + * @brief vec2 dot product + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return dot product + */ +CGLM_INLINE +float +glm_vec2_dot(vec2 a, vec2 b) { + return a[0] * b[0] + a[1] * b[1]; +} + +/*! + * @brief vec2 cross product + * + * REF: http://allenchou.net/2013/07/cross-product-of-2d-vectors/ + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return Z component of cross product + */ +CGLM_INLINE +float +glm_vec2_cross(vec2 a, vec2 b) { + /* just calculate the z-component */ + return a[0] * b[1] - a[1] * b[0]; +} + +/*! + * @brief norm * norm (magnitude) of vec + * + * we can use this func instead of calling norm * norm, because it would call + * sqrtf function twice but with this func we can avoid func call, maybe this is + * not good name for this func + * + * @param[in] v vector + * + * @return norm * norm + */ +CGLM_INLINE +float +glm_vec2_norm2(vec2 v) { + return glm_vec2_dot(v, v); +} + +/*! + * @brief norm (magnitude) of vec2 + * + * @param[in] vec vector + * + * @return norm + */ +CGLM_INLINE +float +glm_vec2_norm(vec2 vec) { + return sqrtf(glm_vec2_norm2(vec)); +} + +/*! + * @brief add a vector to b vector store result in dest + * + * @param[in] a vector1 + * @param[in] b vector2 + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec2_add(vec2 a, vec2 b, vec2 dest) { + dest[0] = a[0] + b[0]; + dest[1] = a[1] + b[1]; +} + +/*! + * @brief add scalar to v vector store result in dest (d = v + s) + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec2_adds(vec2 v, float s, vec2 dest) { + dest[0] = v[0] + s; + dest[1] = v[1] + s; +} + +/*! + * @brief subtract b vector from a vector store result in dest + * + * @param[in] a vector1 + * @param[in] b vector2 + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec2_sub(vec2 a, vec2 b, vec2 dest) { + dest[0] = a[0] - b[0]; + dest[1] = a[1] - b[1]; +} + +/*! + * @brief subtract scalar from v vector store result in dest (d = v - s) + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec2_subs(vec2 v, float s, vec2 dest) { + dest[0] = v[0] - s; + dest[1] = v[1] - s; +} + +/*! + * @brief multiply two vectors (component-wise multiplication) + * + * @param a v1 + * @param b v2 + * @param dest v3 = (a[0] * b[0], a[1] * b[1]) + */ +CGLM_INLINE +void +glm_vec2_mul(vec2 a, vec2 b, vec2 dest) { + dest[0] = a[0] * b[0]; + dest[1] = a[1] * b[1]; +} + +/*! + * @brief multiply/scale vector with scalar: result = v * s + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec2_scale(vec2 v, float s, vec2 dest) { + dest[0] = v[0] * s; + dest[1] = v[1] * s; +} + +/*! + * @brief scale as vector specified: result = unit(v) * s + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec2_scale_as(vec2 v, float s, vec2 dest) { + float norm; + norm = glm_vec2_norm(v); + + if (CGLM_UNLIKELY(norm < FLT_EPSILON)) { + glm_vec2_zero(dest); + return; + } + + glm_vec2_scale(v, s / norm, dest); +} + +/*! + * @brief div vector with another component-wise division: d = a / b + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest result = (a[0]/b[0], a[1]/b[1]) + */ +CGLM_INLINE +void +glm_vec2_div(vec2 a, vec2 b, vec2 dest) { + dest[0] = a[0] / b[0]; + dest[1] = a[1] / b[1]; +} + +/*! + * @brief div vector with scalar: d = v / s + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest result = (a[0]/s, a[1]/s) + */ +CGLM_INLINE +void +glm_vec2_divs(vec2 v, float s, vec2 dest) { + dest[0] = v[0] / s; + dest[1] = v[1] / s; +} + +/*! + * @brief add two vectors and add result to sum + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest += (a + b) + */ +CGLM_INLINE +void +glm_vec2_addadd(vec2 a, vec2 b, vec2 dest) { + dest[0] += a[0] + b[0]; + dest[1] += a[1] + b[1]; +} + +/*! + * @brief sub two vectors and add result to dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest += (a + b) + */ +CGLM_INLINE +void +glm_vec2_subadd(vec2 a, vec2 b, vec2 dest) { + dest[0] += a[0] - b[0]; + dest[1] += a[1] - b[1]; +} + +/*! + * @brief mul two vectors and add result to dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest += (a * b) + */ +CGLM_INLINE +void +glm_vec2_muladd(vec2 a, vec2 b, vec2 dest) { + dest[0] += a[0] * b[0]; + dest[1] += a[1] * b[1]; +} + +/*! + * @brief mul vector with scalar and add result to sum + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest += (a * b) + */ +CGLM_INLINE +void +glm_vec2_muladds(vec2 a, float s, vec2 dest) { + dest[0] += a[0] * s; + dest[1] += a[1] * s; +} + +/*! + * @brief add max of two vectors to result/dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest += max(a, b) + */ +CGLM_INLINE +void +glm_vec2_maxadd(vec2 a, vec2 b, vec2 dest) { + dest[0] += glm_max(a[0], b[0]); + dest[1] += glm_max(a[1], b[1]); +} + +/*! + * @brief add min of two vectors to result/dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest += min(a, b) + */ +CGLM_INLINE +void +glm_vec2_minadd(vec2 a, vec2 b, vec2 dest) { + dest[0] += glm_min(a[0], b[0]); + dest[1] += glm_min(a[1], b[1]); +} + +/*! + * @brief sub two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest -= (a - b) + */ +CGLM_INLINE +void +glm_vec2_subsub(vec2 a, vec2 b, vec2 dest) { + dest[0] -= a[0] - b[0]; + dest[1] -= a[1] - b[1]; +} + +/*! + * @brief add two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest -= (a + b) + */ +CGLM_INLINE +void +glm_vec2_addsub(vec2 a, vec2 b, vec2 dest) { + dest[0] -= a[0] + b[0]; + dest[1] -= a[1] + b[1]; +} + +/*! + * @brief mul two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest -= (a * b) + */ +CGLM_INLINE +void +glm_vec2_mulsub(vec2 a, vec2 b, vec2 dest) { + dest[0] -= a[0] * b[0]; + dest[1] -= a[1] * b[1]; +} + +/*! + * @brief mul vector with scalar and sub result to sum + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest -= (a * b) + */ +CGLM_INLINE +void +glm_vec2_mulsubs(vec2 a, float s, vec2 dest) { + dest[0] -= a[0] * s; + dest[1] -= a[1] * s; +} + +/*! + * @brief sub max of two vectors to result/dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest -= max(a, b) + */ +CGLM_INLINE +void +glm_vec2_maxsub(vec2 a, vec2 b, vec2 dest) { + dest[0] -= glm_max(a[0], b[0]); + dest[1] -= glm_max(a[1], b[1]); +} + +/*! + * @brief sub min of two vectors to result/dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest -= min(a, b) + */ +CGLM_INLINE +void +glm_vec2_minsub(vec2 a, vec2 b, vec2 dest) { + dest[0] -= glm_min(a[0], b[0]); + dest[1] -= glm_min(a[1], b[1]); +} + +/*! + * @brief negate vector components and store result in dest + * + * @param[in] v vector + * @param[out] dest result vector + */ +CGLM_INLINE +void +glm_vec2_negate_to(vec2 v, vec2 dest) { + dest[0] = -v[0]; + dest[1] = -v[1]; +} + +/*! + * @brief negate vector components + * + * @param[in, out] v vector + */ +CGLM_INLINE +void +glm_vec2_negate(vec2 v) { + glm_vec2_negate_to(v, v); +} + +/*! + * @brief normalize vector and store result in same vec + * + * @param[in, out] v vector + */ +CGLM_INLINE +void +glm_vec2_normalize(vec2 v) { + float norm; + + norm = glm_vec2_norm(v); + + if (CGLM_UNLIKELY(norm < FLT_EPSILON)) { + v[0] = v[1] = 0.0f; + return; + } + + glm_vec2_scale(v, 1.0f / norm, v); +} + +/*! + * @brief normalize vector to dest + * + * @param[in] v source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec2_normalize_to(vec2 v, vec2 dest) { + float norm; + + norm = glm_vec2_norm(v); + + if (CGLM_UNLIKELY(norm < FLT_EPSILON)) { + glm_vec2_zero(dest); + return; + } + + glm_vec2_scale(v, 1.0f / norm, dest); +} + +/*! + * @brief rotate vec2 around origin by angle (CCW: counterclockwise) + * + * Formula: + * 𝑥2 = cos(a)𝑥1 − sin(a)𝑦1 + * 𝑦2 = sin(a)𝑥1 + cos(a)𝑦1 + * + * @param[in] v vector to rotate + * @param[in] angle angle by radians + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec2_rotate(vec2 v, float angle, vec2 dest) { + float c, s, x1, y1; + + c = cosf(angle); + s = sinf(angle); + + x1 = v[0]; + y1 = v[1]; + + dest[0] = c * x1 - s * y1; + dest[1] = s * x1 + c * y1; +} + +/** + * @brief find center point of two vector + * + * @param[in] a vector1 + * @param[in] b vector2 + * @param[out] dest center point + */ +CGLM_INLINE +void +glm_vec2_center(vec2 a, vec2 b, vec2 dest) { + glm_vec2_add(a, b, dest); + glm_vec2_scale(dest, 0.5f, dest); +} + +/** + * @brief squared distance between two vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @return returns squared distance (distance * distance) + */ +CGLM_INLINE +float +glm_vec2_distance2(vec2 a, vec2 b) { + return glm_pow2(b[0] - a[0]) + glm_pow2(b[1] - a[1]); +} + +/** + * @brief distance between two vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @return returns distance + */ +CGLM_INLINE +float +glm_vec2_distance(vec2 a, vec2 b) { + return sqrtf(glm_vec2_distance2(a, b)); +} + +/*! + * @brief max values of vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec2_maxv(vec2 a, vec2 b, vec2 dest) { + dest[0] = glm_max(a[0], b[0]); + dest[1] = glm_max(a[1], b[1]); +} + +/*! + * @brief min values of vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec2_minv(vec2 a, vec2 b, vec2 dest) { + dest[0] = glm_min(a[0], b[0]); + dest[1] = glm_min(a[1], b[1]); +} + +/*! + * @brief clamp vector's individual members between min and max values + * + * @param[in, out] v vector + * @param[in] minval minimum value + * @param[in] maxval maximum value + */ +CGLM_INLINE +void +glm_vec2_clamp(vec2 v, float minval, float maxval) { + v[0] = glm_clamp(v[0], minval, maxval); + v[1] = glm_clamp(v[1], minval, maxval); +} + +/*! + * @brief swizzle vector components + * + * @param[in] v source + * @param[in] mask mask + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec2_swizzle(vec2 v, int mask, vec2 dest) { + vec2 t; + + t[0] = v[(mask & (3 << 0))]; + t[1] = v[(mask & (3 << 2)) >> 2]; + + glm_vec2_copy(t, dest); +} + +/*! + * @brief linear interpolation between two vector + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) clamped between 0 and 1 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec2_lerp(vec2 from, vec2 to, float t, vec2 dest) { + vec2 s, v; + + /* from + s * (to - from) */ + glm_vec2_fill(s, glm_clamp_zo(t)); + glm_vec2_sub(to, from, v); + glm_vec2_mul(s, v, v); + glm_vec2_add(from, v, dest); +} + +/*! + * @brief threshold function + * + * @param[in] edge threshold + * @param[in] x value to test against threshold + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec2_step(vec2 edge, vec2 x, vec2 dest) { + dest[0] = glm_step(edge[0], x[0]); + dest[1] = glm_step(edge[1], x[1]); +} + +/*! + * @brief Create two dimensional vector from pointer + * + * @param[in] src pointer to an array of floats + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec2_make(const float * __restrict src, vec2 dest) { + dest[0] = src[0]; dest[1] = src[1]; +} + +/*! + * @brief reflection vector using an incident ray and a surface normal + * + * @param[in] v incident vector + * @param[in] n normalized normal vector + * @param[out] dest destination vector for the reflection result + */ +CGLM_INLINE +void +glm_vec2_reflect(vec2 v, vec2 n, vec2 dest) { + vec2 temp; + glm_vec2_scale(n, 2.0f * glm_vec2_dot(v, n), temp); + glm_vec2_sub(v, temp, dest); +} + +/*! + * @brief computes refraction vector for an incident vector and a surface normal. + * + * calculates the refraction vector based on Snell's law. If total internal reflection + * occurs (angle too great given eta), dest is set to zero and returns false. + * Otherwise, computes refraction vector, stores it in dest, and returns true. + * + * @param[in] v normalized incident vector + * @param[in] n normalized normal vector + * @param[in] eta ratio of indices of refraction (incident/transmitted) + * @param[out] dest refraction vector if refraction occurs; zero vector otherwise + * + * @returns true if refraction occurs; false if total internal reflection occurs. + */ +CGLM_INLINE +bool +glm_vec2_refract(vec2 v, vec2 n, float eta, vec2 dest) { + float ndi, eni, k; + + ndi = glm_vec2_dot(n, v); + eni = eta * ndi; + k = 1.0f - eta * eta + eni * eni; + + if (k < 0.0f) { + glm_vec2_zero(dest); + return false; + } + + glm_vec2_scale(v, eta, dest); + glm_vec2_mulsubs(n, eni + sqrtf(k), dest); + return true; +} + +#endif /* cglm_vec2_h */ diff --git a/cglm/include/cglm/vec3-ext.h b/cglm/include/cglm/vec3-ext.h new file mode 100644 index 0000000..4413cc2 --- /dev/null +++ b/cglm/include/cglm/vec3-ext.h @@ -0,0 +1,345 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/*! + * @brief SIMD like functions + */ + +/* + Functions: + CGLM_INLINE void glm_vec3_broadcast(float val, vec3 d); + CGLM_INLINE void glm_vec3_fill(vec3 v, float val); + CGLM_INLINE bool glm_vec3_eq(vec3 v, float val); + CGLM_INLINE bool glm_vec3_eq_eps(vec3 v, float val); + CGLM_INLINE bool glm_vec3_eq_all(vec3 v); + CGLM_INLINE bool glm_vec3_eqv(vec3 a, vec3 b); + CGLM_INLINE bool glm_vec3_eqv_eps(vec3 a, vec3 b); + CGLM_INLINE float glm_vec3_max(vec3 v); + CGLM_INLINE float glm_vec3_min(vec3 v); + CGLM_INLINE bool glm_vec3_isnan(vec3 v); + CGLM_INLINE bool glm_vec3_isinf(vec3 v); + CGLM_INLINE bool glm_vec3_isvalid(vec3 v); + CGLM_INLINE void glm_vec3_sign(vec3 v, vec3 dest); + CGLM_INLINE void glm_vec3_abs(vec3 v, vec3 dest); + CGLM_INLINE void glm_vec3_fract(vec3 v, vec3 dest); + CGLM_INLINE void glm_vec3_floor(vec3 v, vec3 dest); + CGLM_INLINE float glm_vec3_mods(vec3 v, float s, vec3 dest); + CGLM_INLINE float glm_vec3_steps(float edge, vec3 v, vec3 dest); + CGLM_INLINE void glm_vec3_stepr(vec3 edge, float v, vec3 dest); + CGLM_INLINE float glm_vec3_hadd(vec3 v); + CGLM_INLINE void glm_vec3_sqrt(vec3 v, vec3 dest); + */ + +#ifndef cglm_vec3_ext_h +#define cglm_vec3_ext_h + +#include "common.h" +#include "util.h" + +/*! + * @brief fill a vector with specified value + * + * @param[in] val value + * @param[out] d dest + */ +CGLM_INLINE +void +glm_vec3_broadcast(float val, vec3 d) { + d[0] = d[1] = d[2] = val; +} + +/*! + * @brief fill a vector with specified value + * + * @param[out] v dest + * @param[in] val value + */ +CGLM_INLINE +void +glm_vec3_fill(vec3 v, float val) { + v[0] = v[1] = v[2] = val; +} + +/*! + * @brief check if vector is equal to value (without epsilon) + * + * @param[in] v vector + * @param[in] val value + */ +CGLM_INLINE +bool +glm_vec3_eq(vec3 v, float val) { + return v[0] == val && v[0] == v[1] && v[0] == v[2]; +} + +/*! + * @brief check if vector is equal to value (with epsilon) + * + * @param[in] v vector + * @param[in] val value + */ +CGLM_INLINE +bool +glm_vec3_eq_eps(vec3 v, float val) { + return fabsf(v[0] - val) <= GLM_FLT_EPSILON + && fabsf(v[1] - val) <= GLM_FLT_EPSILON + && fabsf(v[2] - val) <= GLM_FLT_EPSILON; +} + +/*! + * @brief check if vector members are equal (without epsilon) + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glm_vec3_eq_all(vec3 v) { + return glm_vec3_eq_eps(v, v[0]); +} + +/*! + * @brief check if vector is equal to another (without epsilon) + * + * @param[in] a vector + * @param[in] b vector + */ +CGLM_INLINE +bool +glm_vec3_eqv(vec3 a, vec3 b) { + return a[0] == b[0] + && a[1] == b[1] + && a[2] == b[2]; +} + +/*! + * @brief check if vector is equal to another (with epsilon) + * + * @param[in] a vector + * @param[in] b vector + */ +CGLM_INLINE +bool +glm_vec3_eqv_eps(vec3 a, vec3 b) { + return fabsf(a[0] - b[0]) <= GLM_FLT_EPSILON + && fabsf(a[1] - b[1]) <= GLM_FLT_EPSILON + && fabsf(a[2] - b[2]) <= GLM_FLT_EPSILON; +} + +/*! + * @brief max value of vector + * + * @param[in] v vector + */ +CGLM_INLINE +float +glm_vec3_max(vec3 v) { + float max; + + max = v[0]; + if (v[1] > max) + max = v[1]; + if (v[2] > max) + max = v[2]; + + return max; +} + +/*! + * @brief min value of vector + * + * @param[in] v vector + */ +CGLM_INLINE +float +glm_vec3_min(vec3 v) { + float min; + + min = v[0]; + if (v[1] < min) + min = v[1]; + if (v[2] < min) + min = v[2]; + + return min; +} + +/*! + * @brief check if one of items is NaN (not a number) + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glm_vec3_isnan(vec3 v) { +#ifndef CGLM_FAST_MATH + return isnan(v[0]) || isnan(v[1]) || isnan(v[2]); +#else + return false; +#endif +} + +/*! + * @brief check if one of items is INFINITY + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glm_vec3_isinf(vec3 v) { +#ifndef CGLM_FAST_MATH + return isinf(v[0]) || isinf(v[1]) || isinf(v[2]); +#else + return false; +#endif +} + +/*! + * @brief check if all items are valid number + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glm_vec3_isvalid(vec3 v) { + return !glm_vec3_isnan(v) && !glm_vec3_isinf(v); +} + +/*! + * @brief get sign of 32 bit float as +1, -1, 0 + * + * Important: It returns 0 for zero/NaN input + * + * @param v vector + */ +CGLM_INLINE +void +glm_vec3_sign(vec3 v, vec3 dest) { + dest[0] = glm_signf(v[0]); + dest[1] = glm_signf(v[1]); + dest[2] = glm_signf(v[2]); +} + +/*! + * @brief absolute value of each vector item + * + * @param[in] v vector + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec3_abs(vec3 v, vec3 dest) { + dest[0] = fabsf(v[0]); + dest[1] = fabsf(v[1]); + dest[2] = fabsf(v[2]); +} + +/*! + * @brief fractional part of each vector item + * + * @param[in] v vector + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec3_fract(vec3 v, vec3 dest) { + dest[0] = fminf(v[0] - floorf(v[0]), 0.999999940395355224609375f); + dest[1] = fminf(v[1] - floorf(v[1]), 0.999999940395355224609375f); + dest[2] = fminf(v[2] - floorf(v[2]), 0.999999940395355224609375f); +} + +/*! + * @brief floor of each vector item + * + * @param[in] v vector + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec3_floor(vec3 v, vec3 dest) { + dest[0] = floorf(v[0]); + dest[1] = floorf(v[1]); + dest[2] = floorf(v[2]); +} + +/*! + * @brief mod of each vector item, result is written to dest (dest = v % s) + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec3_mods(vec3 v, float s, vec3 dest) { + dest[0] = fmodf(v[0], s); + dest[1] = fmodf(v[1], s); + dest[2] = fmodf(v[2], s); +} + +/*! + * @brief threshold each vector item with scalar + * condition is: (x[i] < edge) ? 0.0 : 1.0 + * + * @param[in] edge threshold + * @param[in] x vector to test against threshold + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_steps(float edge, vec3 x, vec3 dest) { + dest[0] = glm_step(edge, x[0]); + dest[1] = glm_step(edge, x[1]); + dest[2] = glm_step(edge, x[2]); +} + +/*! + * @brief threshold a value with *vector* as the threshold + * condition is: (x < edge[i]) ? 0.0 : 1.0 + * + * @param[in] edge threshold vector + * @param[in] x value to test against threshold + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_stepr(vec3 edge, float x, vec3 dest) { + dest[0] = glm_step(edge[0], x); + dest[1] = glm_step(edge[1], x); + dest[2] = glm_step(edge[2], x); +} + +/*! + * @brief vector reduction by summation + * @warning could overflow + * + * @param[in] v vector + * @return sum of all vector's elements + */ +CGLM_INLINE +float +glm_vec3_hadd(vec3 v) { + return v[0] + v[1] + v[2]; +} + +/*! + * @brief square root of each vector item + * + * @param[in] v vector + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec3_sqrt(vec3 v, vec3 dest) { + dest[0] = sqrtf(v[0]); + dest[1] = sqrtf(v[1]); + dest[2] = sqrtf(v[2]); +} + +#endif /* cglm_vec3_ext_h */ diff --git a/cglm/include/cglm/vec3.h b/cglm/include/cglm/vec3.h new file mode 100644 index 0000000..1350818 --- /dev/null +++ b/cglm/include/cglm/vec3.h @@ -0,0 +1,1264 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_VEC3_ONE_INIT + GLM_VEC3_ZERO_INIT + GLM_VEC3_ONE + GLM_VEC3_ZERO + GLM_YUP + GLM_ZUP + GLM_XUP + + Functions: + CGLM_INLINE void glm_vec3(vec4 v4, vec3 dest); + CGLM_INLINE void glm_vec3_copy(vec3 a, vec3 dest); + CGLM_INLINE void glm_vec3_zero(vec3 v); + CGLM_INLINE void glm_vec3_one(vec3 v); + CGLM_INLINE float glm_vec3_dot(vec3 a, vec3 b); + CGLM_INLINE float glm_vec3_norm2(vec3 v); + CGLM_INLINE float glm_vec3_norm(vec3 v); + CGLM_INLINE float glm_vec3_norm_one(vec3 v); + CGLM_INLINE float glm_vec3_norm_inf(vec3 v); + CGLM_INLINE void glm_vec3_add(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_adds(vec3 a, float s, vec3 dest); + CGLM_INLINE void glm_vec3_sub(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_subs(vec3 a, float s, vec3 dest); + CGLM_INLINE void glm_vec3_mul(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_scale(vec3 v, float s, vec3 dest); + CGLM_INLINE void glm_vec3_scale_as(vec3 v, float s, vec3 dest); + CGLM_INLINE void glm_vec3_div(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_divs(vec3 a, float s, vec3 dest); + CGLM_INLINE void glm_vec3_addadd(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_subadd(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_muladd(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_muladds(vec3 a, float s, vec3 dest); + CGLM_INLINE void glm_vec3_maxadd(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_minadd(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_subsub(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_addsub(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_mulsub(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_mulsubs(vec3 a, float s, vec3 dest); + CGLM_INLINE void glm_vec3_maxsub(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_minsub(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_flipsign(vec3 v); + CGLM_INLINE void glm_vec3_flipsign_to(vec3 v, vec3 dest); + CGLM_INLINE void glm_vec3_negate_to(vec3 v, vec3 dest); + CGLM_INLINE void glm_vec3_negate(vec3 v); + CGLM_INLINE void glm_vec3_inv(vec3 v); + CGLM_INLINE void glm_vec3_inv_to(vec3 v, vec3 dest); + CGLM_INLINE void glm_vec3_normalize(vec3 v); + CGLM_INLINE void glm_vec3_normalize_to(vec3 v, vec3 dest); + CGLM_INLINE void glm_vec3_cross(vec3 a, vec3 b, vec3 d); + CGLM_INLINE void glm_vec3_crossn(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE float glm_vec3_angle(vec3 a, vec3 b); + CGLM_INLINE void glm_vec3_rotate(vec3 v, float angle, vec3 axis); + CGLM_INLINE void glm_vec3_rotate_m4(mat4 m, vec3 v, vec3 dest); + CGLM_INLINE void glm_vec3_rotate_m3(mat3 m, vec3 v, vec3 dest); + CGLM_INLINE void glm_vec3_proj(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_center(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE float glm_vec3_distance(vec3 a, vec3 b); + CGLM_INLINE float glm_vec3_distance2(vec3 a, vec3 b); + CGLM_INLINE void glm_vec3_maxv(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_minv(vec3 a, vec3 b, vec3 dest); + CGLM_INLINE void glm_vec3_ortho(vec3 v, vec3 dest); + CGLM_INLINE void glm_vec3_clamp(vec3 v, float minVal, float maxVal); + CGLM_INLINE void glm_vec3_lerp(vec3 from, vec3 to, float t, vec3 dest); + CGLM_INLINE void glm_vec3_lerpc(vec3 from, vec3 to, float t, vec3 dest); + CGLM_INLINE void glm_vec3_mix(vec3 from, vec3 to, float t, vec3 dest); + CGLM_INLINE void glm_vec3_mixc(vec3 from, vec3 to, float t, vec3 dest); + CGLM_INLINE void glm_vec3_step(vec3 edge, vec3 x, vec3 dest); + CGLM_INLINE void glm_vec3_smoothstep_uni(float edge0, float edge1, vec3 x, vec3 dest); + CGLM_INLINE void glm_vec3_smoothstep(vec3 edge0, vec3 edge1, vec3 x, vec3 dest); + CGLM_INLINE void glm_vec3_smoothinterp(vec3 from, vec3 to, float t, vec3 dest); + CGLM_INLINE void glm_vec3_smoothinterpc(vec3 from, vec3 to, float t, vec3 dest); + CGLM_INLINE void glm_vec3_swizzle(vec3 v, int mask, vec3 dest); + CGLM_INLINE void glm_vec3_make(float * restrict src, vec3 dest); + CGLM_INLINE void glm_vec3_faceforward(vec3 n, vec3 v, vec3 nref, vec3 dest); + CGLM_INLINE void glm_vec3_reflect(vec3 v, vec3 n, vec3 dest); + CGLM_INLINE void glm_vec3_refract(vec3 v, vec3 n, float eta, vec3 dest); + + Convenient: + CGLM_INLINE void glm_cross(vec3 a, vec3 b, vec3 d); + CGLM_INLINE float glm_dot(vec3 a, vec3 b); + CGLM_INLINE void glm_normalize(vec3 v); + CGLM_INLINE void glm_normalize_to(vec3 v, vec3 dest); + + DEPRECATED: + glm_vec3_dup + glm_vec3_flipsign + glm_vec3_flipsign_to + glm_vec3_inv + glm_vec3_inv_to + glm_vec3_mulv + glm_vec3_step_uni --> use glm_vec3_steps + */ + +#ifndef cglm_vec3_h +#define cglm_vec3_h + +#include "common.h" +#include "vec4.h" +#include "vec3-ext.h" +#include "util.h" + +/* DEPRECATED! use _copy, _ucopy versions */ +#define glm_vec3_dup(v, dest) glm_vec3_copy(v, dest) +#define glm_vec3_flipsign(v) glm_vec3_negate(v) +#define glm_vec3_flipsign_to(v, dest) glm_vec3_negate_to(v, dest) +#define glm_vec3_inv(v) glm_vec3_negate(v) +#define glm_vec3_inv_to(v, dest) glm_vec3_negate_to(v, dest) +#define glm_vec3_mulv(a, b, d) glm_vec3_mul(a, b, d) +#define glm_vec3_step_uni(edge, x, dest) glm_vec3_steps(edge, x, dest) + +#define GLM_VEC3_ONE_INIT {1.0f, 1.0f, 1.0f} +#define GLM_VEC3_ZERO_INIT {0.0f, 0.0f, 0.0f} + +#define GLM_VEC3_ONE ((vec3)GLM_VEC3_ONE_INIT) +#define GLM_VEC3_ZERO ((vec3)GLM_VEC3_ZERO_INIT) + +#define GLM_YUP ((vec3){0.0f, 1.0f, 0.0f}) +#define GLM_ZUP ((vec3){0.0f, 0.0f, 1.0f}) +#define GLM_XUP ((vec3){1.0f, 0.0f, 0.0f}) +#define GLM_FORWARD ((vec3){0.0f, 0.0f, -1.0f}) + +#define GLM_XXX GLM_SHUFFLE3(0, 0, 0) +#define GLM_YYY GLM_SHUFFLE3(1, 1, 1) +#define GLM_ZZZ GLM_SHUFFLE3(2, 2, 2) +#define GLM_ZYX GLM_SHUFFLE3(0, 1, 2) + +/*! + * @brief init vec3 using vec4 + * + * @param[in] v4 vector4 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3(vec4 v4, vec3 dest) { + dest[0] = v4[0]; + dest[1] = v4[1]; + dest[2] = v4[2]; +} + +/*! + * @brief copy all members of [a] to [dest] + * + * @param[in] a source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_copy(vec3 a, vec3 dest) { + dest[0] = a[0]; + dest[1] = a[1]; + dest[2] = a[2]; +} + +/*! + * @brief make vector zero + * + * @param[in, out] v vector + */ +CGLM_INLINE +void +glm_vec3_zero(vec3 v) { + v[0] = v[1] = v[2] = 0.0f; +} + +/*! + * @brief make vector one + * + * @param[in, out] v vector + */ +CGLM_INLINE +void +glm_vec3_one(vec3 v) { + v[0] = v[1] = v[2] = 1.0f; +} + +/*! + * @brief vec3 dot product + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return dot product + */ +CGLM_INLINE +float +glm_vec3_dot(vec3 a, vec3 b) { + return a[0] * b[0] + a[1] * b[1] + a[2] * b[2]; +} + +/*! + * @brief norm * norm (magnitude) of vec + * + * we can use this func instead of calling norm * norm, because it would call + * sqrtf function twice but with this func we can avoid func call, maybe this is + * not good name for this func + * + * @param[in] v vector + * + * @return norm * norm + */ +CGLM_INLINE +float +glm_vec3_norm2(vec3 v) { + return glm_vec3_dot(v, v); +} + +/*! + * @brief euclidean norm (magnitude), also called L2 norm + * this will give magnitude of vector in euclidean space + * + * @param[in] v vector + * + * @return norm + */ +CGLM_INLINE +float +glm_vec3_norm(vec3 v) { + return sqrtf(glm_vec3_norm2(v)); +} + +/*! + * @brief L1 norm of vec3 + * Also known as Manhattan Distance or Taxicab norm. + * L1 Norm is the sum of the magnitudes of the vectors in a space. + * It is calculated as the sum of the absolute values of the vector components. + * In this norm, all the components of the vector are weighted equally. + * + * This computes: + * R = |v[0]| + |v[1]| + |v[2]| + * + * @param[in] v vector + * + * @return L1 norm + */ +CGLM_INLINE +float +glm_vec3_norm_one(vec3 v) { + vec3 t; + glm_vec3_abs(v, t); + return glm_vec3_hadd(t); +} + +/*! + * @brief infinity norm of vec3 + * Also known as Maximum norm. + * Infinity Norm is the largest magnitude among each element of a vector. + * It is calculated as the maximum of the absolute values of the vector components. + * + * This computes: + * inf norm = max(|v[0]|, |v[1]|, |v[2]|) + * + * @param[in] v vector + * + * @return infinity norm + */ +CGLM_INLINE +float +glm_vec3_norm_inf(vec3 v) { + vec3 t; + glm_vec3_abs(v, t); + return glm_vec3_max(t); +} + +/*! + * @brief add a vector to b vector store result in dest + * + * @param[in] a vector1 + * @param[in] b vector2 + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec3_add(vec3 a, vec3 b, vec3 dest) { + dest[0] = a[0] + b[0]; + dest[1] = a[1] + b[1]; + dest[2] = a[2] + b[2]; +} + +/*! + * @brief add scalar to v vector store result in dest (d = v + s) + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec3_adds(vec3 v, float s, vec3 dest) { + dest[0] = v[0] + s; + dest[1] = v[1] + s; + dest[2] = v[2] + s; +} + +/*! + * @brief subtract b vector from a vector store result in dest + * + * @param[in] a vector1 + * @param[in] b vector2 + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec3_sub(vec3 a, vec3 b, vec3 dest) { + dest[0] = a[0] - b[0]; + dest[1] = a[1] - b[1]; + dest[2] = a[2] - b[2]; +} + +/*! + * @brief subtract scalar from v vector store result in dest (d = v - s) + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec3_subs(vec3 v, float s, vec3 dest) { + dest[0] = v[0] - s; + dest[1] = v[1] - s; + dest[2] = v[2] - s; +} + +/*! + * @brief multiply two vectors (component-wise multiplication) + * + * @param a vector1 + * @param b vector2 + * @param dest v3 = (a[0] * b[0], a[1] * b[1], a[2] * b[2]) + */ +CGLM_INLINE +void +glm_vec3_mul(vec3 a, vec3 b, vec3 dest) { + dest[0] = a[0] * b[0]; + dest[1] = a[1] * b[1]; + dest[2] = a[2] * b[2]; +} + +/*! + * @brief multiply/scale vec3 vector with scalar: result = v * s + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec3_scale(vec3 v, float s, vec3 dest) { + dest[0] = v[0] * s; + dest[1] = v[1] * s; + dest[2] = v[2] * s; +} + +/*! + * @brief make vec3 vector scale as specified: result = unit(v) * s + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec3_scale_as(vec3 v, float s, vec3 dest) { + float norm; + norm = glm_vec3_norm(v); + + if (CGLM_UNLIKELY(norm < FLT_EPSILON)) { + glm_vec3_zero(dest); + return; + } + + glm_vec3_scale(v, s / norm, dest); +} + +/*! + * @brief div vector with another component-wise division: d = a / b + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest result = (a[0]/b[0], a[1]/b[1], a[2]/b[2]) + */ +CGLM_INLINE +void +glm_vec3_div(vec3 a, vec3 b, vec3 dest) { + dest[0] = a[0] / b[0]; + dest[1] = a[1] / b[1]; + dest[2] = a[2] / b[2]; +} + +/*! + * @brief div vector with scalar: d = v / s + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest result = (a[0]/s, a[1]/s, a[2]/s) + */ +CGLM_INLINE +void +glm_vec3_divs(vec3 v, float s, vec3 dest) { + dest[0] = v[0] / s; + dest[1] = v[1] / s; + dest[2] = v[2] / s; +} + +/*! + * @brief add two vectors and add result to sum + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest += (a + b) + */ +CGLM_INLINE +void +glm_vec3_addadd(vec3 a, vec3 b, vec3 dest) { + dest[0] += a[0] + b[0]; + dest[1] += a[1] + b[1]; + dest[2] += a[2] + b[2]; +} + +/*! + * @brief sub two vectors and add result to dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest += (a + b) + */ +CGLM_INLINE +void +glm_vec3_subadd(vec3 a, vec3 b, vec3 dest) { + dest[0] += a[0] - b[0]; + dest[1] += a[1] - b[1]; + dest[2] += a[2] - b[2]; +} + +/*! + * @brief mul two vectors and add result to dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest += (a * b) + */ +CGLM_INLINE +void +glm_vec3_muladd(vec3 a, vec3 b, vec3 dest) { + dest[0] += a[0] * b[0]; + dest[1] += a[1] * b[1]; + dest[2] += a[2] * b[2]; +} + +/*! + * @brief mul vector with scalar and add result to sum + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest += (a * b) + */ +CGLM_INLINE +void +glm_vec3_muladds(vec3 a, float s, vec3 dest) { + dest[0] += a[0] * s; + dest[1] += a[1] * s; + dest[2] += a[2] * s; +} + +/*! + * @brief add max of two vectors to result/dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest += max(a, b) + */ +CGLM_INLINE +void +glm_vec3_maxadd(vec3 a, vec3 b, vec3 dest) { + dest[0] += glm_max(a[0], b[0]); + dest[1] += glm_max(a[1], b[1]); + dest[2] += glm_max(a[2], b[2]); +} + +/*! + * @brief add min of two vectors to result/dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest += min(a, b) + */ +CGLM_INLINE +void +glm_vec3_minadd(vec3 a, vec3 b, vec3 dest) { + dest[0] += glm_min(a[0], b[0]); + dest[1] += glm_min(a[1], b[1]); + dest[2] += glm_min(a[2], b[2]); +} + +/*! + * @brief sub two vectors and sub result to dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest -= (a - b) + */ +CGLM_INLINE +void +glm_vec3_subsub(vec3 a, vec3 b, vec3 dest) { + dest[0] -= a[0] - b[0]; + dest[1] -= a[1] - b[1]; + dest[2] -= a[2] - b[2]; +} + +/*! + * @brief add two vectors and sub result to dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest -= (a + b) + */ +CGLM_INLINE +void +glm_vec3_addsub(vec3 a, vec3 b, vec3 dest) { + dest[0] -= a[0] + b[0]; + dest[1] -= a[1] + b[1]; + dest[2] -= a[2] + b[2]; +} + +/*! + * @brief mul two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest -= (a * b) + */ +CGLM_INLINE +void +glm_vec3_mulsub(vec3 a, vec3 b, vec3 dest) { + dest[0] -= a[0] * b[0]; + dest[1] -= a[1] * b[1]; + dest[2] -= a[2] * b[2]; +} + +/*! + * @brief mul vector with scalar and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest -= (a * b) + */ +CGLM_INLINE +void +glm_vec3_mulsubs(vec3 a, float s, vec3 dest) { + dest[0] -= a[0] * s; + dest[1] -= a[1] * s; + dest[2] -= a[2] * s; +} + +/*! + * @brief sub max of two vectors to result/dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest -= max(a, b) + */ +CGLM_INLINE +void +glm_vec3_maxsub(vec3 a, vec3 b, vec3 dest) { + dest[0] -= glm_max(a[0], b[0]); + dest[1] -= glm_max(a[1], b[1]); + dest[2] -= glm_max(a[2], b[2]); +} + +/*! + * @brief sub min of two vectors to result/dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest -= min(a, b) + */ +CGLM_INLINE +void +glm_vec3_minsub(vec3 a, vec3 b, vec3 dest) { + dest[0] -= glm_min(a[0], b[0]); + dest[1] -= glm_min(a[1], b[1]); + dest[2] -= glm_min(a[2], b[2]); +} + +/*! + * @brief negate vector components and store result in dest + * + * @param[in] v vector + * @param[out] dest result vector + */ +CGLM_INLINE +void +glm_vec3_negate_to(vec3 v, vec3 dest) { + dest[0] = -v[0]; + dest[1] = -v[1]; + dest[2] = -v[2]; +} + +/*! + * @brief negate vector components + * + * @param[in, out] v vector + */ +CGLM_INLINE +void +glm_vec3_negate(vec3 v) { + glm_vec3_negate_to(v, v); +} + +/*! + * @brief normalize vec3 and store result in same vec + * + * @param[in, out] v vector + */ +CGLM_INLINE +void +glm_vec3_normalize(vec3 v) { + float norm; + + norm = glm_vec3_norm(v); + + if (CGLM_UNLIKELY(norm < FLT_EPSILON)) { + v[0] = v[1] = v[2] = 0.0f; + return; + } + + glm_vec3_scale(v, 1.0f / norm, v); +} + +/*! + * @brief normalize vec3 to dest + * + * @param[in] v source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_normalize_to(vec3 v, vec3 dest) { + float norm; + + norm = glm_vec3_norm(v); + + if (CGLM_UNLIKELY(norm < FLT_EPSILON)) { + glm_vec3_zero(dest); + return; + } + + glm_vec3_scale(v, 1.0f / norm, dest); +} + +/*! + * @brief cross product of two vector (RH) + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_cross(vec3 a, vec3 b, vec3 dest) { + vec3 c; + /* (u2.v3 - u3.v2, u3.v1 - u1.v3, u1.v2 - u2.v1) */ + c[0] = a[1] * b[2] - a[2] * b[1]; + c[1] = a[2] * b[0] - a[0] * b[2]; + c[2] = a[0] * b[1] - a[1] * b[0]; + glm_vec3_copy(c, dest); +} + +/*! + * @brief cross product of two vector (RH) and normalize the result + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_crossn(vec3 a, vec3 b, vec3 dest) { + glm_vec3_cross(a, b, dest); + glm_vec3_normalize(dest); +} + +/*! + * @brief angle between two vector + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return angle as radians + */ +CGLM_INLINE +float +glm_vec3_angle(vec3 a, vec3 b) { + float norm, dot; + + /* maybe compiler generate approximation instruction (rcp) */ + norm = 1.0f / (glm_vec3_norm(a) * glm_vec3_norm(b)); + dot = glm_vec3_dot(a, b) * norm; + + if (dot > 1.0f) + return 0.0f; + else if (dot < -1.0f) + return CGLM_PI; + + return acosf(dot); +} + +/*! + * @brief rotate vec3 around axis by angle using Rodrigues' rotation formula + * + * @param[in, out] v vector + * @param[in] axis axis vector (must be unit vector) + * @param[in] angle angle by radians + */ +CGLM_INLINE +void +glm_vec3_rotate(vec3 v, float angle, vec3 axis) { + vec3 v1, v2, k; + float c, s; + + c = cosf(angle); + s = sinf(angle); + + glm_vec3_normalize_to(axis, k); + + /* Right Hand, Rodrigues' rotation formula: + v = v*cos(t) + (kxv)sin(t) + k*(k.v)(1 - cos(t)) + */ + glm_vec3_scale(v, c, v1); + + glm_vec3_cross(k, v, v2); + glm_vec3_scale(v2, s, v2); + + glm_vec3_add(v1, v2, v1); + + glm_vec3_scale(k, glm_vec3_dot(k, v) * (1.0f - c), v2); + glm_vec3_add(v1, v2, v); +} + +/*! + * @brief apply rotation matrix to vector + * + * matrix format should be (no perspective): + * a b c x + * e f g y + * i j k z + * 0 0 0 w + * + * @param[in] m affine matrix or rot matrix + * @param[in] v vector + * @param[out] dest rotated vector + */ +CGLM_INLINE +void +glm_vec3_rotate_m4(mat4 m, vec3 v, vec3 dest) { + vec4 x, y, z, res; + + glm_vec4_normalize_to(m[0], x); + glm_vec4_normalize_to(m[1], y); + glm_vec4_normalize_to(m[2], z); + + glm_vec4_scale(x, v[0], res); + glm_vec4_muladds(y, v[1], res); + glm_vec4_muladds(z, v[2], res); + + glm_vec3(res, dest); +} + +/*! + * @brief apply rotation matrix to vector + * + * @param[in] m affine matrix or rot matrix + * @param[in] v vector + * @param[out] dest rotated vector + */ +CGLM_INLINE +void +glm_vec3_rotate_m3(mat3 m, vec3 v, vec3 dest) { + vec4 res, x, y, z; + + glm_vec4(m[0], 0.0f, x); + glm_vec4(m[1], 0.0f, y); + glm_vec4(m[2], 0.0f, z); + + glm_vec4_normalize(x); + glm_vec4_normalize(y); + glm_vec4_normalize(z); + + glm_vec4_scale(x, v[0], res); + glm_vec4_muladds(y, v[1], res); + glm_vec4_muladds(z, v[2], res); + + glm_vec3(res, dest); +} + +/*! + * @brief project a vector onto b vector + * + * @param[in] a vector1 + * @param[in] b vector2 + * @param[out] dest projected vector + */ +CGLM_INLINE +void +glm_vec3_proj(vec3 a, vec3 b, vec3 dest) { + glm_vec3_scale(b, + glm_vec3_dot(a, b) / glm_vec3_norm2(b), + dest); +} + +/** + * @brief find center point of two vector + * + * @param[in] a vector1 + * @param[in] b vector2 + * @param[out] dest center point + */ +CGLM_INLINE +void +glm_vec3_center(vec3 a, vec3 b, vec3 dest) { + glm_vec3_add(a, b, dest); + glm_vec3_scale(dest, 0.5f, dest); +} + +/** + * @brief squared distance between two vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @return returns squared distance (distance * distance) + */ +CGLM_INLINE +float +glm_vec3_distance2(vec3 a, vec3 b) { + return glm_pow2(a[0] - b[0]) + + glm_pow2(a[1] - b[1]) + + glm_pow2(a[2] - b[2]); +} + +/** + * @brief distance between two vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @return returns distance + */ +CGLM_INLINE +float +glm_vec3_distance(vec3 a, vec3 b) { + return sqrtf(glm_vec3_distance2(a, b)); +} + +/*! + * @brief max values of vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_maxv(vec3 a, vec3 b, vec3 dest) { + dest[0] = glm_max(a[0], b[0]); + dest[1] = glm_max(a[1], b[1]); + dest[2] = glm_max(a[2], b[2]); +} + +/*! + * @brief min values of vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_minv(vec3 a, vec3 b, vec3 dest) { + dest[0] = glm_min(a[0], b[0]); + dest[1] = glm_min(a[1], b[1]); + dest[2] = glm_min(a[2], b[2]); +} + +/*! + * @brief possible orthogonal/perpendicular vector + * + * @param[in] v vector + * @param[out] dest orthogonal/perpendicular vector + */ +CGLM_INLINE +void +glm_vec3_ortho(vec3 v, vec3 dest) { + float ignore; + float f = modff(fabsf(v[0]) + 0.5f, &ignore); + vec3 result = {-v[1], v[0] - f * v[2], f * v[1]}; + glm_vec3_copy(result, dest); +} + +/*! + * @brief clamp vector's individual members between min and max values + * + * @param[in, out] v vector + * @param[in] minVal minimum value + * @param[in] maxVal maximum value + */ +CGLM_INLINE +void +glm_vec3_clamp(vec3 v, float minVal, float maxVal) { + v[0] = glm_clamp(v[0], minVal, maxVal); + v[1] = glm_clamp(v[1], minVal, maxVal); + v[2] = glm_clamp(v[2], minVal, maxVal); +} + +/*! + * @brief linear interpolation between two vectors + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_lerp(vec3 from, vec3 to, float t, vec3 dest) { + vec3 s, v; + + /* from + s * (to - from) */ + glm_vec3_broadcast(t, s); + glm_vec3_sub(to, from, v); + glm_vec3_mul(s, v, v); + glm_vec3_add(from, v, dest); +} + +/*! + * @brief linear interpolation between two vectors (clamped) + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) clamped between 0 and 1 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_lerpc(vec3 from, vec3 to, float t, vec3 dest) { + glm_vec3_lerp(from, to, glm_clamp_zo(t), dest); +} + +/*! + * @brief linear interpolation between two vectors + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_mix(vec3 from, vec3 to, float t, vec3 dest) { + glm_vec3_lerp(from, to, t, dest); +} + +/*! + * @brief linear interpolation between two vectors (clamped) + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) clamped between 0 and 1 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_mixc(vec3 from, vec3 to, float t, vec3 dest) { + glm_vec3_lerpc(from, to, t, dest); +} + +/*! + * @brief threshold function + * + * @param[in] edge threshold + * @param[in] x value to test against threshold + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_step(vec3 edge, vec3 x, vec3 dest) { + dest[0] = glm_step(edge[0], x[0]); + dest[1] = glm_step(edge[1], x[1]); + dest[2] = glm_step(edge[2], x[2]); +} + +/*! + * @brief threshold function with a smooth transition (unidimensional) + * + * @param[in] edge0 low threshold + * @param[in] edge1 high threshold + * @param[in] x value to test against threshold + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_smoothstep_uni(float edge0, float edge1, vec3 x, vec3 dest) { + dest[0] = glm_smoothstep(edge0, edge1, x[0]); + dest[1] = glm_smoothstep(edge0, edge1, x[1]); + dest[2] = glm_smoothstep(edge0, edge1, x[2]); +} + +/*! + * @brief threshold function with a smooth transition + * + * @param[in] edge0 low threshold + * @param[in] edge1 high threshold + * @param[in] x value to test against threshold + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_smoothstep(vec3 edge0, vec3 edge1, vec3 x, vec3 dest) { + dest[0] = glm_smoothstep(edge0[0], edge1[0], x[0]); + dest[1] = glm_smoothstep(edge0[1], edge1[1], x[1]); + dest[2] = glm_smoothstep(edge0[2], edge1[2], x[2]); +} + +/*! + * @brief smooth Hermite interpolation between two vectors + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_smoothinterp(vec3 from, vec3 to, float t, vec3 dest) { + vec3 s, v; + + /* from + s * (to - from) */ + glm_vec3_broadcast(glm_smooth(t), s); + glm_vec3_sub(to, from, v); + glm_vec3_mul(s, v, v); + glm_vec3_add(from, v, dest); +} + +/*! + * @brief smooth Hermite interpolation between two vectors (clamped) + * + * formula: from + s * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) clamped between 0 and 1 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_smoothinterpc(vec3 from, vec3 to, float t, vec3 dest) { + glm_vec3_smoothinterp(from, to, glm_clamp_zo(t), dest); +} + +/*! + * @brief swizzle vector components + * + * you can use existing masks e.g. GLM_XXX, GLM_ZYX + * + * @param[in] v source + * @param[in] mask mask + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec3_swizzle(vec3 v, int mask, vec3 dest) { + vec3 t; + + t[0] = v[(mask & (3 << 0))]; + t[1] = v[(mask & (3 << 2)) >> 2]; + t[2] = v[(mask & (3 << 4)) >> 4]; + + glm_vec3_copy(t, dest); +} + +/*! + * @brief vec3 cross product + * + * this is just convenient wrapper + * + * @param[in] a source 1 + * @param[in] b source 2 + * @param[out] d destination + */ +CGLM_INLINE +void +glm_cross(vec3 a, vec3 b, vec3 d) { + glm_vec3_cross(a, b, d); +} + +/*! + * @brief vec3 dot product + * + * this is just convenient wrapper + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return dot product + */ +CGLM_INLINE +float +glm_dot(vec3 a, vec3 b) { + return glm_vec3_dot(a, b); +} + +/*! + * @brief normalize vec3 and store result in same vec + * + * this is just convenient wrapper + * + * @param[in, out] v vector + */ +CGLM_INLINE +void +glm_normalize(vec3 v) { + glm_vec3_normalize(v); +} + +/*! + * @brief normalize vec3 to dest + * + * this is just convenient wrapper + * + * @param[in] v source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_normalize_to(vec3 v, vec3 dest) { + glm_vec3_normalize_to(v, dest); +} + +/*! + * @brief Create three dimensional vector from pointer + * + * @param[in] src pointer to an array of floats + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec3_make(const float * __restrict src, vec3 dest) { + dest[0] = src[0]; + dest[1] = src[1]; + dest[2] = src[2]; +} + +/*! + * @brief a vector pointing in the same direction as another + * + * orients a vector to point away from a surface as defined by its normal + * + * @param[in] n vector to orient + * @param[in] v incident vector + * @param[in] nref reference vector + * @param[out] dest oriented vector, pointing away from the surface + */ +CGLM_INLINE +void +glm_vec3_faceforward(vec3 n, vec3 v, vec3 nref, vec3 dest) { + if (glm_vec3_dot(v, nref) < 0.0f) { + /* N is facing away from I */ + glm_vec3_copy(n, dest); + } else { + /* N is facing towards I, negate it */ + glm_vec3_negate_to(n, dest); + } +} + +/*! + * @brief reflection vector using an incident ray and a surface normal + * + * @param[in] v incident vector + * @param[in] n normalized normal vector + * @param[out] dest reflection result + */ +CGLM_INLINE +void +glm_vec3_reflect(vec3 v, vec3 n, vec3 dest) { + vec3 temp; + glm_vec3_scale(n, 2.0f * glm_vec3_dot(v, n), temp); + glm_vec3_sub(v, temp, dest); +} + +/*! + * @brief computes refraction vector for an incident vector and a surface normal. + * + * calculates the refraction vector based on Snell's law. If total internal reflection + * occurs (angle too great given eta), dest is set to zero and returns false. + * Otherwise, computes refraction vector, stores it in dest, and returns true. + * + * @param[in] v normalized incident vector + * @param[in] n normalized normal vector + * @param[in] eta ratio of indices of refraction (incident/transmitted) + * @param[out] dest refraction vector if refraction occurs; zero vector otherwise + * + * @returns true if refraction occurs; false if total internal reflection occurs. + */ +CGLM_INLINE +bool +glm_vec3_refract(vec3 v, vec3 n, float eta, vec3 dest) { + float ndi, eni, k; + + ndi = glm_vec3_dot(n, v); + eni = eta * ndi; + k = 1.0f - eta * eta + eni * eni; + + if (k < 0.0f) { + glm_vec3_zero(dest); + return false; + } + + glm_vec3_scale(v, eta, dest); + glm_vec3_mulsubs(n, eni + sqrtf(k), dest); + return true; +} + +#endif /* cglm_vec3_h */ diff --git a/cglm/include/cglm/vec4-ext.h b/cglm/include/cglm/vec4-ext.h new file mode 100644 index 0000000..193a5e9 --- /dev/null +++ b/cglm/include/cglm/vec4-ext.h @@ -0,0 +1,400 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/*! + * @brief SIMD like functions + */ + +/* + Functions: + CGLM_INLINE void glm_vec4_broadcast(float val, vec4 d); + CGLM_INLINE void glm_vec4_fill(vec4 v, float val); + CGLM_INLINE bool glm_vec4_eq(vec4 v, float val); + CGLM_INLINE bool glm_vec4_eq_eps(vec4 v, float val); + CGLM_INLINE bool glm_vec4_eq_all(vec4 v); + CGLM_INLINE bool glm_vec4_eqv(vec4 a, vec4 b); + CGLM_INLINE bool glm_vec4_eqv_eps(vec4 a, vec4 b); + CGLM_INLINE float glm_vec4_max(vec4 v); + CGLM_INLINE float glm_vec4_min(vec4 v); + CGLM_INLINE bool glm_vec4_isnan(vec4 v); + CGLM_INLINE bool glm_vec4_isinf(vec4 v); + CGLM_INLINE bool glm_vec4_isvalid(vec4 v); + CGLM_INLINE void glm_vec4_sign(vec4 v, vec4 dest); + CGLM_INLINE void glm_vec4_abs(vec4 v, vec4 dest); + CGLM_INLINE void glm_vec4_fract(vec4 v, vec4 dest); + CGLM_INLINE void glm_vec4_floor(vec4 v, vec4 dest); + CGLM_INLINE float glm_vec4_mods(vec4 v, float s, vec4 dest); + CGLM_INLINE float glm_vec4_steps(float edge, vec4 v, vec4 dest); + CGLM_INLINE void glm_vec4_stepr(vec4 edge, float v, vec4 dest); + CGLM_INLINE float glm_vec4_hadd(vec4 v); + CGLM_INLINE void glm_vec4_sqrt(vec4 v, vec4 dest); + */ + +#ifndef cglm_vec4_ext_h +#define cglm_vec4_ext_h + +#include "common.h" +#include "vec3-ext.h" + +/*! + * @brief fill a vector with specified value + * + * @param val value + * @param d dest + */ +CGLM_INLINE +void +glm_vec4_broadcast(float val, vec4 d) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(d, wasm_f32x4_splat(val)); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(d, glmm_set1(val)); +#else + d[0] = d[1] = d[2] = d[3] = val; +#endif +} + +/*! + * @brief fill a vector with specified value + * + * @param v dest + * @param val value + */ +CGLM_INLINE +void +glm_vec4_fill(vec4 v, float val) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(v, wasm_f32x4_splat(val)); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(v, glmm_set1(val)); +#else + v[0] = v[1] = v[2] = v[3] = val; +#endif +} + +/*! + * @brief check if vector is equal to value (without epsilon) + * + * @param v vector + * @param val value + */ +CGLM_INLINE +bool +glm_vec4_eq(vec4 v, float val) { + return v[0] == val + && v[0] == v[1] + && v[0] == v[2] + && v[0] == v[3]; +} + +/*! + * @brief check if vector is equal to value (with epsilon) + * + * @param v vector + * @param val value + */ +CGLM_INLINE +bool +glm_vec4_eq_eps(vec4 v, float val) { + return fabsf(v[0] - val) <= GLM_FLT_EPSILON + && fabsf(v[1] - val) <= GLM_FLT_EPSILON + && fabsf(v[2] - val) <= GLM_FLT_EPSILON + && fabsf(v[3] - val) <= GLM_FLT_EPSILON; +} + +/*! + * @brief check if vector members are equal (without epsilon) + * + * @param v vector + */ +CGLM_INLINE +bool +glm_vec4_eq_all(vec4 v) { + return glm_vec4_eq_eps(v, v[0]); +} + +/*! + * @brief check if vector is equal to another (without epsilon) + * + * @param a vector + * @param b vector + */ +CGLM_INLINE +bool +glm_vec4_eqv(vec4 a, vec4 b) { + return a[0] == b[0] + && a[1] == b[1] + && a[2] == b[2] + && a[3] == b[3]; +} + +/*! + * @brief check if vector is equal to another (with epsilon) + * + * @param a vector + * @param b vector + */ +CGLM_INLINE +bool +glm_vec4_eqv_eps(vec4 a, vec4 b) { + return fabsf(a[0] - b[0]) <= GLM_FLT_EPSILON + && fabsf(a[1] - b[1]) <= GLM_FLT_EPSILON + && fabsf(a[2] - b[2]) <= GLM_FLT_EPSILON + && fabsf(a[3] - b[3]) <= GLM_FLT_EPSILON; +} + +/*! + * @brief max value of vector + * + * @param v vector + */ +CGLM_INLINE +float +glm_vec4_max(vec4 v) { + float max; + + max = glm_vec3_max(v); + if (v[3] > max) + max = v[3]; + + return max; +} + +/*! + * @brief min value of vector + * + * @param v vector + */ +CGLM_INLINE +float +glm_vec4_min(vec4 v) { + float min; + + min = glm_vec3_min(v); + if (v[3] < min) + min = v[3]; + + return min; +} + +/*! + * @brief check if one of items is NaN (not a number) + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glm_vec4_isnan(vec4 v) { +#ifndef CGLM_FAST_MATH + return isnan(v[0]) || isnan(v[1]) || isnan(v[2]) || isnan(v[3]); +#else + return false; +#endif +} + +/*! + * @brief check if one of items is INFINITY + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glm_vec4_isinf(vec4 v) { +#ifndef CGLM_FAST_MATH + return isinf(v[0]) || isinf(v[1]) || isinf(v[2]) || isinf(v[3]); +#else + return false; +#endif +} + +/*! + * @brief check if all items are valid number + * you should only use this in DEBUG mode or very critical asserts + * + * @param[in] v vector + */ +CGLM_INLINE +bool +glm_vec4_isvalid(vec4 v) { + return !glm_vec4_isnan(v) && !glm_vec4_isinf(v); +} + +/*! + * @brief get sign of 32 bit float as +1, -1, 0 + * + * Important: It returns 0 for zero/NaN input + * + * @param v vector + */ +CGLM_INLINE +void +glm_vec4_sign(vec4 v, vec4 dest) { +#if defined( __SSE__ ) || defined( __SSE2__ ) + __m128 x0, x1, x2, x3, x4; + + x0 = glmm_load(v); + x1 = _mm_set_ps(0.0f, 0.0f, 1.0f, -1.0f); + x2 = glmm_splat(x1, 2); + + x3 = _mm_and_ps(_mm_cmpgt_ps(x0, x2), glmm_splat(x1, 1)); + x4 = _mm_and_ps(_mm_cmplt_ps(x0, x2), glmm_splat(x1, 0)); + + glmm_store(dest, _mm_or_ps(x3, x4)); +#else + dest[0] = glm_signf(v[0]); + dest[1] = glm_signf(v[1]); + dest[2] = glm_signf(v[2]); + dest[3] = glm_signf(v[3]); +#endif +} + +/*! + * @brief absolute value of each vector item + * + * @param[in] v vector + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec4_abs(vec4 v, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, glmm_abs(glmm_load(v))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, glmm_abs(glmm_load(v))); +#elif defined(CGLM_NEON_FP) + vst1q_f32(dest, vabsq_f32(vld1q_f32(v))); +#else + dest[0] = fabsf(v[0]); + dest[1] = fabsf(v[1]); + dest[2] = fabsf(v[2]); + dest[3] = fabsf(v[3]); +#endif +} + +/*! + * @brief fractional part of each vector item + * + * @param[in] v vector + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec4_fract(vec4 v, vec4 dest) { + dest[0] = fminf(v[0] - floorf(v[0]), 0.999999940395355224609375f); + dest[1] = fminf(v[1] - floorf(v[1]), 0.999999940395355224609375f); + dest[2] = fminf(v[2] - floorf(v[2]), 0.999999940395355224609375f); + dest[3] = fminf(v[3] - floorf(v[3]), 0.999999940395355224609375f); +} + +/*! + * @brief floor of each vector item + * + * @param[in] v vector + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec4_floor(vec4 v, vec4 dest) { + dest[0] = floorf(v[0]); + dest[1] = floorf(v[1]); + dest[2] = floorf(v[2]); + dest[3] = floorf(v[3]); +} + +/*! + * @brief mod of each vector item, result is written to dest (dest = v % s) + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec4_mods(vec4 v, float s, vec4 dest) { + dest[0] = fmodf(v[0], s); + dest[1] = fmodf(v[1], s); + dest[2] = fmodf(v[2], s); + dest[3] = fmodf(v[3], s); +} + +/*! + * @brief threshold each vector item with scalar + * condition is: (x[i] < edge) ? 0.0 : 1.0 + * + * @param[in] edge threshold + * @param[in] x vector to test against threshold + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_steps(float edge, vec4 x, vec4 dest) { + dest[0] = glm_step(edge, x[0]); + dest[1] = glm_step(edge, x[1]); + dest[2] = glm_step(edge, x[2]); + dest[3] = glm_step(edge, x[3]); +} + +/*! + * @brief threshold a value with *vector* as the threshold + * condition is: (x < edge[i]) ? 0.0 : 1.0 + * + * @param[in] edge threshold vector + * @param[in] x value to test against threshold + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_stepr(vec4 edge, float x, vec4 dest) { + dest[0] = glm_step(edge[0], x); + dest[1] = glm_step(edge[1], x); + dest[2] = glm_step(edge[2], x); + dest[3] = glm_step(edge[3], x); +} + +/*! + * @brief vector reduction by summation + * @warning could overflow + * + * @param[in] v vector + * @return sum of all vector's elements + */ +CGLM_INLINE +float +glm_vec4_hadd(vec4 v) { +#if defined(__wasm__) && defined(__wasm_simd128__) + return glmm_hadd(glmm_load(v)); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + return glmm_hadd(glmm_load(v)); +#else + return v[0] + v[1] + v[2] + v[3]; +#endif +} + +/*! + * @brief square root of each vector item + * + * @param[in] v vector + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec4_sqrt(vec4 v, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_sqrt(glmm_load(v))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_sqrt_ps(glmm_load(v))); +#else + dest[0] = sqrtf(v[0]); + dest[1] = sqrtf(v[1]); + dest[2] = sqrtf(v[2]); + dest[3] = sqrtf(v[3]); +#endif +} + +#endif /* cglm_vec4_ext_h */ diff --git a/cglm/include/cglm/vec4.h b/cglm/include/cglm/vec4.h new file mode 100644 index 0000000..ded09c9 --- /dev/null +++ b/cglm/include/cglm/vec4.h @@ -0,0 +1,1348 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* + Macros: + GLM_VEC4_ONE_INIT + GLM_VEC4_BLACK_INIT + GLM_VEC4_ZERO_INIT + GLM_VEC4_ONE + GLM_VEC4_BLACK + GLM_VEC4_ZERO + + Functions: + CGLM_INLINE void glm_vec4(vec3 v3, float last, vec4 dest); + CGLM_INLINE void glm_vec4_copy3(vec4 a, vec3 dest); + CGLM_INLINE void glm_vec4_copy(vec4 v, vec4 dest); + CGLM_INLINE void glm_vec4_ucopy(vec4 v, vec4 dest); + CGLM_INLINE float glm_vec4_dot(vec4 a, vec4 b); + CGLM_INLINE float glm_vec4_norm2(vec4 v); + CGLM_INLINE float glm_vec4_norm(vec4 v); + CGLM_INLINE float glm_vec4_norm_one(vec4 v); + CGLM_INLINE float glm_vec4_norm_inf(vec4 v); + CGLM_INLINE void glm_vec4_add(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_adds(vec4 v, float s, vec4 dest); + CGLM_INLINE void glm_vec4_sub(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_subs(vec4 v, float s, vec4 dest); + CGLM_INLINE void glm_vec4_mul(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_scale(vec4 v, float s, vec4 dest); + CGLM_INLINE void glm_vec4_scale_as(vec4 v, float s, vec4 dest); + CGLM_INLINE void glm_vec4_div(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_divs(vec4 v, float s, vec4 dest); + CGLM_INLINE void glm_vec4_addadd(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_subadd(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_muladd(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_muladds(vec4 a, float s, vec4 dest); + CGLM_INLINE void glm_vec4_maxadd(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_minadd(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_subsub(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_addsub(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_mulsub(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_mulsubs(vec4 a, float s, vec4 dest); + CGLM_INLINE void glm_vec4_maxsub(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_minsub(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_negate(vec4 v); + CGLM_INLINE void glm_vec4_inv(vec4 v); + CGLM_INLINE void glm_vec4_inv_to(vec4 v, vec4 dest); + CGLM_INLINE void glm_vec4_normalize(vec4 v); + CGLM_INLINE void glm_vec4_normalize_to(vec4 vec, vec4 dest); + CGLM_INLINE float glm_vec4_distance(vec4 a, vec4 b); + CGLM_INLINE float glm_vec4_distance2(vec4 a, vec4 b); + CGLM_INLINE void glm_vec4_maxv(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_minv(vec4 a, vec4 b, vec4 dest); + CGLM_INLINE void glm_vec4_clamp(vec4 v, float minVal, float maxVal); + CGLM_INLINE void glm_vec4_lerp(vec4 from, vec4 to, float t, vec4 dest); + CGLM_INLINE void glm_vec4_lerpc(vec4 from, vec4 to, float t, vec4 dest); + CGLM_INLINE void glm_vec4_step(vec4 edge, vec4 x, vec4 dest); + CGLM_INLINE void glm_vec4_smoothstep_uni(float edge0, float edge1, vec4 x, vec4 dest); + CGLM_INLINE void glm_vec4_smoothstep(vec4 edge0, vec4 edge1, vec4 x, vec4 dest); + CGLM_INLINE void glm_vec4_smoothinterp(vec4 from, vec4 to, float t, vec4 dest); + CGLM_INLINE void glm_vec4_smoothinterpc(vec4 from, vec4 to, float t, vec4 dest); + CGLM_INLINE void glm_vec4_swizzle(vec4 v, int mask, vec4 dest); + CGLM_INLINE void glm_vec4_make(float * restrict src, vec4 dest); + CGLM_INLINE void glm_vec4_reflect(vec4 v, vec4 n, vec4 dest); + CGLM_INLINE void glm_vec4_refract(vec4 v, vec4 n, float eta, vec4 dest); + + DEPRECATED: + glm_vec4_dup + glm_vec4_flipsign + glm_vec4_flipsign_to + glm_vec4_inv + glm_vec4_inv_to + glm_vec4_mulv + glm_vec4_step_uni --> use glm_vec4_steps + */ + +#ifndef cglm_vec4_h +#define cglm_vec4_h + +#include "common.h" +#include "vec4-ext.h" +#include "util.h" + +/* DEPRECATED! functions */ +#define glm_vec4_dup3(v, dest) glm_vec4_copy3(v, dest) +#define glm_vec4_dup(v, dest) glm_vec4_copy(v, dest) +#define glm_vec4_flipsign(v) glm_vec4_negate(v) +#define glm_vec4_flipsign_to(v, dest) glm_vec4_negate_to(v, dest) +#define glm_vec4_inv(v) glm_vec4_negate(v) +#define glm_vec4_inv_to(v, dest) glm_vec4_negate_to(v, dest) +#define glm_vec4_mulv(a, b, d) glm_vec4_mul(a, b, d) +#define glm_vec4_step_uni(edge, x, dest) glm_vec4_steps(edge, x, dest) + +#define GLM_VEC4_ONE_INIT {1.0f, 1.0f, 1.0f, 1.0f} +#define GLM_VEC4_BLACK_INIT {0.0f, 0.0f, 0.0f, 1.0f} +#define GLM_VEC4_ZERO_INIT {0.0f, 0.0f, 0.0f, 0.0f} + +#define GLM_VEC4_ONE ((vec4)GLM_VEC4_ONE_INIT) +#define GLM_VEC4_BLACK ((vec4)GLM_VEC4_BLACK_INIT) +#define GLM_VEC4_ZERO ((vec4)GLM_VEC4_ZERO_INIT) + +#define GLM_XXXX GLM_SHUFFLE4(0, 0, 0, 0) +#define GLM_YYYY GLM_SHUFFLE4(1, 1, 1, 1) +#define GLM_ZZZZ GLM_SHUFFLE4(2, 2, 2, 2) +#define GLM_WWWW GLM_SHUFFLE4(3, 3, 3, 3) +#define GLM_WZYX GLM_SHUFFLE4(0, 1, 2, 3) + +/*! + * @brief init vec4 using vec3 + * + * @param[in] v3 vector3 + * @param[in] last last item + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4(vec3 v3, float last, vec4 dest) { + dest[0] = v3[0]; + dest[1] = v3[1]; + dest[2] = v3[2]; + dest[3] = last; +} + +/*! + * @brief copy first 3 members of [a] to [dest] + * + * @param[in] a source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_copy3(vec4 a, vec3 dest) { + dest[0] = a[0]; + dest[1] = a[1]; + dest[2] = a[2]; +} + +/*! + * @brief copy all members of [a] to [dest] + * + * @param[in] v source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_copy(vec4 v, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, glmm_load(v)); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, glmm_load(v)); +#elif defined(CGLM_NEON_FP) + vst1q_f32(dest, vld1q_f32(v)); +#else + dest[0] = v[0]; + dest[1] = v[1]; + dest[2] = v[2]; + dest[3] = v[3]; +#endif +} + +/*! + * @brief copy all members of [a] to [dest] + * + * alignment is not required + * + * @param[in] v source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_ucopy(vec4 v, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + /* note here wasm v128.load/v128.store support unaligned loads and stores */ + wasm_v128_store(dest, wasm_v128_load(v)); +#else + dest[0] = v[0]; + dest[1] = v[1]; + dest[2] = v[2]; + dest[3] = v[3]; +#endif +} + +/*! + * @brief make vector zero + * + * @param[in, out] v vector + */ +CGLM_INLINE +void +glm_vec4_zero(vec4 v) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(v, wasm_f32x4_const_splat(0.f)); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(v, _mm_setzero_ps()); +#elif defined(CGLM_NEON_FP) + vst1q_f32(v, vdupq_n_f32(0.0f)); +#else + v[0] = 0.0f; + v[1] = 0.0f; + v[2] = 0.0f; + v[3] = 0.0f; +#endif +} + +/*! + * @brief make vector one + * + * @param[in, out] v vector + */ +CGLM_INLINE +void +glm_vec4_one(vec4 v) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(v, wasm_f32x4_const_splat(1.0f)); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(v, glmm_set1_rval(1.0f)); +#elif defined(CGLM_NEON_FP) + vst1q_f32(v, vdupq_n_f32(1.0f)); +#else + v[0] = 1.0f; + v[1] = 1.0f; + v[2] = 1.0f; + v[3] = 1.0f; +#endif +} + +/*! + * @brief vec4 dot product + * + * @param[in] a vector1 + * @param[in] b vector2 + * + * @return dot product + */ +CGLM_INLINE +float +glm_vec4_dot(vec4 a, vec4 b) { +#if defined(CGLM_SIMD) + return glmm_dot(glmm_load(a), glmm_load(b)); +#else + return a[0] * b[0] + a[1] * b[1] + a[2] * b[2] + a[3] * b[3]; +#endif +} + +/*! + * @brief norm * norm (magnitude) of vec + * + * we can use this func instead of calling norm * norm, because it would call + * sqrtf function twice but with this func we can avoid func call, maybe this is + * not good name for this func + * + * @param[in] v vec4 + * + * @return norm * norm + */ +CGLM_INLINE +float +glm_vec4_norm2(vec4 v) { + return glm_vec4_dot(v, v); +} + +/*! + * @brief euclidean norm (magnitude), also called L2 norm + * this will give magnitude of vector in euclidean space + * + * @param[in] v vector + * + * @return norm + */ +CGLM_INLINE +float +glm_vec4_norm(vec4 v) { +#if defined(CGLM_SIMD) + return glmm_norm(glmm_load(v)); +#else + return sqrtf(glm_vec4_dot(v, v)); +#endif +} + +/*! + * @brief L1 norm of vec4 + * Also known as Manhattan Distance or Taxicab norm. + * L1 Norm is the sum of the magnitudes of the vectors in a space. + * It is calculated as the sum of the absolute values of the vector components. + * In this norm, all the components of the vector are weighted equally. + * + * This computes: + * L1 norm = |v[0]| + |v[1]| + |v[2]| + |v[3]| + * + * @param[in] v vector + * + * @return L1 norm + */ +CGLM_INLINE +float +glm_vec4_norm_one(vec4 v) { +#if defined(CGLM_SIMD) + return glmm_norm_one(glmm_load(v)); +#else + vec4 t; + glm_vec4_abs(v, t); + return glm_vec4_hadd(t); +#endif +} + +/*! + * @brief infinity norm of vec4 + * Also known as Maximum norm. + * Infinity Norm is the largest magnitude among each element of a vector. + * It is calculated as the maximum of the absolute values of the vector components. + * + * This computes: + * inf norm = max(|v[0]|, |v[1]|, |v[2]|, |v[3]|) + * + * @param[in] v vector + * + * @return infinity norm + */ +CGLM_INLINE +float +glm_vec4_norm_inf(vec4 v) { +#if defined(CGLM_SIMD) + return glmm_norm_inf(glmm_load(v)); +#else + vec4 t; + glm_vec4_abs(v, t); + return glm_vec4_max(t); +#endif +} + +/*! + * @brief add b vector to a vector store result in dest + * + * @param[in] a vector1 + * @param[in] b vector2 + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec4_add(vec4 a, vec4 b, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_add(glmm_load(a), glmm_load(b))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_add_ps(glmm_load(a), glmm_load(b))); +#elif defined(CGLM_NEON_FP) + vst1q_f32(dest, vaddq_f32(vld1q_f32(a), vld1q_f32(b))); +#else + dest[0] = a[0] + b[0]; + dest[1] = a[1] + b[1]; + dest[2] = a[2] + b[2]; + dest[3] = a[3] + b[3]; +#endif +} + +/*! + * @brief add scalar to v vector store result in dest (d = v + vec(s)) + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec4_adds(vec4 v, float s, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_add(glmm_load(v), wasm_f32x4_splat(s))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_add_ps(glmm_load(v), glmm_set1(s))); +#elif defined(CGLM_NEON_FP) + vst1q_f32(dest, vaddq_f32(vld1q_f32(v), vdupq_n_f32(s))); +#else + dest[0] = v[0] + s; + dest[1] = v[1] + s; + dest[2] = v[2] + s; + dest[3] = v[3] + s; +#endif +} + +/*! + * @brief subtract b vector from a vector store result in dest (d = a - b) + * + * @param[in] a vector1 + * @param[in] b vector2 + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec4_sub(vec4 a, vec4 b, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_sub(glmm_load(a), glmm_load(b))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_sub_ps(glmm_load(a), glmm_load(b))); +#elif defined(CGLM_NEON_FP) + vst1q_f32(dest, vsubq_f32(vld1q_f32(a), vld1q_f32(b))); +#else + dest[0] = a[0] - b[0]; + dest[1] = a[1] - b[1]; + dest[2] = a[2] - b[2]; + dest[3] = a[3] - b[3]; +#endif +} + +/*! + * @brief subtract scalar from v vector store result in dest (d = v - vec(s)) + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec4_subs(vec4 v, float s, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_sub(glmm_load(v), wasm_f32x4_splat(s))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_sub_ps(glmm_load(v), glmm_set1(s))); +#elif defined(CGLM_NEON_FP) + vst1q_f32(dest, vsubq_f32(vld1q_f32(v), vdupq_n_f32(s))); +#else + dest[0] = v[0] - s; + dest[1] = v[1] - s; + dest[2] = v[2] - s; + dest[3] = v[3] - s; +#endif +} + +/*! + * @brief multiply two vectors (component-wise multiplication) + * + * @param a vector1 + * @param b vector2 + * @param dest dest = (a[0] * b[0], a[1] * b[1], a[2] * b[2], a[3] * b[3]) + */ +CGLM_INLINE +void +glm_vec4_mul(vec4 a, vec4 b, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_mul(glmm_load(a), glmm_load(b))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_mul_ps(glmm_load(a), glmm_load(b))); +#elif defined(CGLM_NEON_FP) + vst1q_f32(dest, vmulq_f32(vld1q_f32(a), vld1q_f32(b))); +#else + dest[0] = a[0] * b[0]; + dest[1] = a[1] * b[1]; + dest[2] = a[2] * b[2]; + dest[3] = a[3] * b[3]; +#endif +} + +/*! + * @brief multiply/scale vec4 vector with scalar: result = v * s + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec4_scale(vec4 v, float s, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_mul(glmm_load(v), wasm_f32x4_splat(s))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_mul_ps(glmm_load(v), glmm_set1(s))); +#elif defined(CGLM_NEON_FP) + vst1q_f32(dest, vmulq_f32(vld1q_f32(v), vdupq_n_f32(s))); +#else + dest[0] = v[0] * s; + dest[1] = v[1] * s; + dest[2] = v[2] * s; + dest[3] = v[3] * s; +#endif +} + +/*! + * @brief make vec4 vector scale as specified: result = unit(v) * s + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec4_scale_as(vec4 v, float s, vec4 dest) { + float norm; + norm = glm_vec4_norm(v); + + if (CGLM_UNLIKELY(norm < FLT_EPSILON)) { + glm_vec4_zero(dest); + return; + } + + glm_vec4_scale(v, s / norm, dest); +} + +/*! + * @brief div vector with another component-wise division: d = a / b + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest result = (a[0]/b[0], a[1]/b[1], a[2]/b[2], a[3]/b[3]) + */ +CGLM_INLINE +void +glm_vec4_div(vec4 a, vec4 b, vec4 dest) { +#if defined(CGLM_SIMD) + glmm_store(dest, glmm_div(glmm_load(a), glmm_load(b))); +#else + dest[0] = a[0] / b[0]; + dest[1] = a[1] / b[1]; + dest[2] = a[2] / b[2]; + dest[3] = a[3] / b[3]; +#endif +} + +/*! + * @brief div vec4 vector with scalar: d = v / s + * + * @param[in] v vector + * @param[in] s scalar + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec4_divs(vec4 v, float s, vec4 dest) { +#if defined(CGLM_SIMD) + glmm_store(dest, glmm_div(glmm_load(v), glmm_set1(s))); +#else + glm_vec4_scale(v, 1.0f / s, dest); +#endif +} + +/*! + * @brief add two vectors and add result to sum + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest += (a + b) + */ +CGLM_INLINE +void +glm_vec4_addadd(vec4 a, vec4 b, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_add( + glmm_load(dest), + wasm_f32x4_add(glmm_load(a), glmm_load(b)))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_add_ps(glmm_load(dest), + _mm_add_ps(glmm_load(a), + glmm_load(b)))); +#elif defined(CGLM_NEON_FP) + vst1q_f32(dest, vaddq_f32(vld1q_f32(dest), + vaddq_f32(vld1q_f32(a), + vld1q_f32(b)))); +#else + dest[0] += a[0] + b[0]; + dest[1] += a[1] + b[1]; + dest[2] += a[2] + b[2]; + dest[3] += a[3] + b[3]; +#endif +} + +/*! + * @brief sub two vectors and add result to dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest += (a - b) + */ +CGLM_INLINE +void +glm_vec4_subadd(vec4 a, vec4 b, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_add( + glmm_load(dest), + wasm_f32x4_sub(glmm_load(a), glmm_load(b)))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_add_ps(glmm_load(dest), + _mm_sub_ps(glmm_load(a), + glmm_load(b)))); +#elif defined(CGLM_NEON_FP) + vst1q_f32(dest, vaddq_f32(vld1q_f32(dest), + vsubq_f32(vld1q_f32(a), + vld1q_f32(b)))); +#else + dest[0] += a[0] - b[0]; + dest[1] += a[1] - b[1]; + dest[2] += a[2] - b[2]; + dest[3] += a[3] - b[3]; +#endif +} + +/*! + * @brief mul two vectors and add result to dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest += (a * b) + */ +CGLM_INLINE +void +glm_vec4_muladd(vec4 a, vec4 b, vec4 dest) { +#if defined(CGLM_SIMD) + glmm_store(dest, glmm_fmadd(glmm_load(a), glmm_load(b), glmm_load(dest))); +#else + dest[0] += a[0] * b[0]; + dest[1] += a[1] * b[1]; + dest[2] += a[2] * b[2]; + dest[3] += a[3] * b[3]; +#endif +} + +/*! + * @brief mul vector with scalar and add result to sum + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest += (a * b) + */ +CGLM_INLINE +void +glm_vec4_muladds(vec4 a, float s, vec4 dest) { +#if defined(CGLM_SIMD) + glmm_store(dest, glmm_fmadd(glmm_load(a), glmm_set1(s), glmm_load(dest))); +#else + dest[0] += a[0] * s; + dest[1] += a[1] * s; + dest[2] += a[2] * s; + dest[3] += a[3] * s; +#endif +} + +/*! + * @brief add max of two vectors to result/dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest += max(a, b) + */ +CGLM_INLINE +void +glm_vec4_maxadd(vec4 a, vec4 b, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_add(glmm_load(dest), + glmm_max(glmm_load(a), glmm_load(b)))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_add_ps(glmm_load(dest), + glmm_max(glmm_load(a), glmm_load(b)))); +#elif defined(CGLM_NEON_FP) + glmm_store(dest, vaddq_f32(glmm_load(dest), + glmm_max(glmm_load(a), glmm_load(b)))); +#else + dest[0] += glm_max(a[0], b[0]); + dest[1] += glm_max(a[1], b[1]); + dest[2] += glm_max(a[2], b[2]); + dest[3] += glm_max(a[3], b[3]); +#endif +} + +/*! + * @brief add min of two vectors to result/dest + * + * it applies += operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest += min(a, b) + */ +CGLM_INLINE +void +glm_vec4_minadd(vec4 a, vec4 b, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_add(glmm_load(dest), + glmm_min(glmm_load(a), glmm_load(b)))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_add_ps(glmm_load(dest), + glmm_min(glmm_load(a), glmm_load(b)))); +#elif defined(CGLM_NEON_FP) + glmm_store(dest, vaddq_f32(glmm_load(dest), + glmm_min(glmm_load(a), glmm_load(b)))); +#else + dest[0] += glm_min(a[0], b[0]); + dest[1] += glm_min(a[1], b[1]); + dest[2] += glm_min(a[2], b[2]); + dest[3] += glm_min(a[3], b[3]); +#endif +} + +/*! + * @brief sub two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest -= (a - b) + */ +CGLM_INLINE +void +glm_vec4_subsub(vec4 a, vec4 b, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_sub( + glmm_load(dest), + wasm_f32x4_sub(glmm_load(a), glmm_load(b)))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_sub_ps(glmm_load(dest), + _mm_sub_ps(glmm_load(a), + glmm_load(b)))); +#elif defined(CGLM_NEON_FP) + vst1q_f32(dest, vsubq_f32(vld1q_f32(dest), + vsubq_f32(vld1q_f32(a), + vld1q_f32(b)))); +#else + dest[0] -= a[0] - b[0]; + dest[1] -= a[1] - b[1]; + dest[2] -= a[2] - b[2]; + dest[3] -= a[3] - b[3]; +#endif +} + +/*! + * @brief add two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest -= (a + b) + */ +CGLM_INLINE +void +glm_vec4_addsub(vec4 a, vec4 b, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_sub( + glmm_load(dest), + wasm_f32x4_add(glmm_load(a), glmm_load(b)))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_sub_ps(glmm_load(dest), + _mm_add_ps(glmm_load(a), + glmm_load(b)))); +#elif defined(CGLM_NEON_FP) + vst1q_f32(dest, vsubq_f32(vld1q_f32(dest), + vaddq_f32(vld1q_f32(a), + vld1q_f32(b)))); +#else + dest[0] -= a[0] + b[0]; + dest[1] -= a[1] + b[1]; + dest[2] -= a[2] + b[2]; + dest[3] -= a[3] + b[3]; +#endif +} + +/*! + * @brief mul two vectors and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest -= (a * b) + */ +CGLM_INLINE +void +glm_vec4_mulsub(vec4 a, vec4 b, vec4 dest) { +#if defined(CGLM_SIMD) + glmm_store(dest, glmm_fnmadd(glmm_load(a), glmm_load(b), glmm_load(dest))); +#else + dest[0] -= a[0] * b[0]; + dest[1] -= a[1] * b[1]; + dest[2] -= a[2] * b[2]; + dest[3] -= a[3] * b[3]; +#endif +} + +/*! + * @brief mul vector with scalar and sub result to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector + * @param[in] s scalar + * @param[out] dest dest -= (a * b) + */ +CGLM_INLINE +void +glm_vec4_mulsubs(vec4 a, float s, vec4 dest) { +#if defined(CGLM_SIMD) + glmm_store(dest, glmm_fnmadd(glmm_load(a), glmm_set1(s), glmm_load(dest))); +#else + dest[0] -= a[0] * s; + dest[1] -= a[1] * s; + dest[2] -= a[2] * s; + dest[3] -= a[3] * s; +#endif +} + +/*! + * @brief sub max of two vectors to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest -= max(a, b) + */ +CGLM_INLINE +void +glm_vec4_maxsub(vec4 a, vec4 b, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_sub(glmm_load(dest), + glmm_max(glmm_load(a), glmm_load(b)))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_sub_ps(glmm_load(dest), + glmm_max(glmm_load(a), glmm_load(b)))); +#elif defined(CGLM_NEON_FP) + glmm_store(dest, vsubq_f32(glmm_load(dest), + glmm_max(glmm_load(a), glmm_load(b)))); +#else + dest[0] -= glm_max(a[0], b[0]); + dest[1] -= glm_max(a[1], b[1]); + dest[2] -= glm_max(a[2], b[2]); + dest[3] -= glm_max(a[3], b[3]); +#endif +} + +/*! + * @brief sub min of two vectors to dest + * + * it applies -= operator so dest must be initialized + * + * @param[in] a vector 1 + * @param[in] b vector 2 + * @param[out] dest dest -= min(a, b) + */ +CGLM_INLINE +void +glm_vec4_minsub(vec4 a, vec4 b, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_sub(glmm_load(dest), + glmm_min(glmm_load(a), glmm_load(b)))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_sub_ps(glmm_load(dest), + glmm_min(glmm_load(a), glmm_load(b)))); +#elif defined(CGLM_NEON_FP) + glmm_store(dest, vsubq_f32(vld1q_f32(dest), + glmm_min(glmm_load(a), glmm_load(b)))); +#else + dest[0] -= glm_min(a[0], b[0]); + dest[1] -= glm_min(a[1], b[1]); + dest[2] -= glm_min(a[2], b[2]); + dest[3] -= glm_min(a[3], b[3]); +#endif +} + +/*! + * @brief negate vector components and store result in dest + * + * @param[in] v vector + * @param[out] dest result vector + */ +CGLM_INLINE +void +glm_vec4_negate_to(vec4 v, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(dest, wasm_f32x4_neg(glmm_load(v))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(dest, _mm_xor_ps(glmm_load(v), glmm_float32x4_SIGNMASK_NEG)); +#elif defined(CGLM_NEON_FP) + vst1q_f32(dest, vnegq_f32(vld1q_f32(v))); +#else + dest[0] = -v[0]; + dest[1] = -v[1]; + dest[2] = -v[2]; + dest[3] = -v[3]; +#endif +} + +/*! + * @brief flip sign of all vec4 members + * + * @param[in, out] v vector + */ +CGLM_INLINE +void +glm_vec4_negate(vec4 v) { + glm_vec4_negate_to(v, v); +} + +/*! + * @brief normalize vec4 to dest + * + * @param[in] v source + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_normalize_to(vec4 v, vec4 dest) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_128 xdot, x0; + float dot; + + x0 = glmm_load(v); + xdot = glmm_vdot(x0, x0); + /* dot = _mm_cvtss_f32(xdot); */ + dot = wasm_f32x4_extract_lane(xdot, 0); + + if (CGLM_UNLIKELY(dot < FLT_EPSILON)) { + glmm_store(dest, wasm_f32x4_const_splat(0.f)); + return; + } + + glmm_store(dest, glmm_div(x0, wasm_f32x4_sqrt(xdot))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + __m128 xdot, x0; + float dot; + + x0 = glmm_load(v); + xdot = glmm_vdot(x0, x0); + dot = _mm_cvtss_f32(xdot); + + if (CGLM_UNLIKELY(dot < FLT_EPSILON)) { + glmm_store(dest, _mm_setzero_ps()); + return; + } + + glmm_store(dest, glmm_div(x0, _mm_sqrt_ps(xdot))); +#else + float norm; + + norm = glm_vec4_norm(v); + + if (CGLM_UNLIKELY(norm < FLT_EPSILON)) { + glm_vec4_zero(dest); + return; + } + + glm_vec4_scale(v, 1.0f / norm, dest); +#endif +} + +/*! + * @brief normalize vec4 and store result in same vec + * + * @param[in, out] v vector + */ +CGLM_INLINE +void +glm_vec4_normalize(vec4 v) { + glm_vec4_normalize_to(v, v); +} + +/** + * @brief distance between two vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @return returns distance + */ +CGLM_INLINE +float +glm_vec4_distance(vec4 a, vec4 b) { +#if defined(__wasm__) && defined(__wasm_simd128__) + return glmm_norm(wasm_f32x4_sub(glmm_load(a), glmm_load(b))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + return glmm_norm(_mm_sub_ps(glmm_load(a), glmm_load(b))); +#elif defined(CGLM_NEON_FP) + return glmm_norm(vsubq_f32(glmm_load(a), glmm_load(b))); +#else + return sqrtf(glm_pow2(a[0] - b[0]) + + glm_pow2(a[1] - b[1]) + + glm_pow2(a[2] - b[2]) + + glm_pow2(a[3] - b[3])); +#endif +} + +/** + * @brief squared distance between two vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @return returns squared distance + */ +CGLM_INLINE +float +glm_vec4_distance2(vec4 a, vec4 b) { +#if defined(__wasm__) && defined(__wasm_simd128__) + return glmm_norm2(wasm_f32x4_sub(glmm_load(a), glmm_load(b))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + return glmm_norm2(_mm_sub_ps(glmm_load(a), glmm_load(b))); +#elif defined(CGLM_NEON_FP) + return glmm_norm2(vsubq_f32(glmm_load(a), glmm_load(b))); +#else + return glm_pow2(a[0] - b[0]) + + glm_pow2(a[1] - b[1]) + + glm_pow2(a[2] - b[2]) + + glm_pow2(a[3] - b[3]); +#endif +} + +/*! + * @brief max values of vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_maxv(vec4 a, vec4 b, vec4 dest) { +#if defined(CGLM_SIMD) + glmm_store(dest, glmm_max(glmm_load(a), glmm_load(b))); +#else + dest[0] = glm_max(a[0], b[0]); + dest[1] = glm_max(a[1], b[1]); + dest[2] = glm_max(a[2], b[2]); + dest[3] = glm_max(a[3], b[3]); +#endif +} + +/*! + * @brief min values of vectors + * + * @param[in] a vector1 + * @param[in] b vector2 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_minv(vec4 a, vec4 b, vec4 dest) { +#if defined(CGLM_SIMD) + glmm_store(dest, glmm_min(glmm_load(a), glmm_load(b))); +#else + dest[0] = glm_min(a[0], b[0]); + dest[1] = glm_min(a[1], b[1]); + dest[2] = glm_min(a[2], b[2]); + dest[3] = glm_min(a[3], b[3]); +#endif +} + +/*! + * @brief clamp vector's individual members between min and max values + * + * @param[in, out] v vector + * @param[in] minVal minimum value + * @param[in] maxVal maximum value + */ +CGLM_INLINE +void +glm_vec4_clamp(vec4 v, float minVal, float maxVal) { +#if defined(__wasm__) && defined(__wasm_simd128__) + glmm_store(v, glmm_min(glmm_max(glmm_load(v), wasm_f32x4_splat(minVal)), + wasm_f32x4_splat(maxVal))); +#elif defined( __SSE__ ) || defined( __SSE2__ ) + glmm_store(v, glmm_min(glmm_max(glmm_load(v), glmm_set1(minVal)), + glmm_set1(maxVal))); +#elif defined(CGLM_NEON_FP) + glmm_store(v, glmm_min(glmm_max(vld1q_f32(v), vdupq_n_f32(minVal)), + vdupq_n_f32(maxVal))); +#else + v[0] = glm_clamp(v[0], minVal, maxVal); + v[1] = glm_clamp(v[1], minVal, maxVal); + v[2] = glm_clamp(v[2], minVal, maxVal); + v[3] = glm_clamp(v[3], minVal, maxVal); +#endif +} + +/*! + * @brief linear interpolation between two vectors + * + * formula: from + t * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_lerp(vec4 from, vec4 to, float t, vec4 dest) { + vec4 s, v; + + /* from + s * (to - from) */ + glm_vec4_broadcast(t, s); + glm_vec4_sub(to, from, v); + glm_vec4_mul(s, v, v); + glm_vec4_add(from, v, dest); +} + +/*! + * @brief linear interpolation between two vectors (clamped) + * + * formula: from + t * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) clamped between 0 and 1 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_lerpc(vec4 from, vec4 to, float t, vec4 dest) { + glm_vec4_lerp(from, to, glm_clamp_zo(t), dest); +} + +/*! + * @brief linear interpolation between two vectors + * + * formula: from + t * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_mix(vec4 from, vec4 to, float t, vec4 dest) { + glm_vec4_lerp(from, to, t, dest); +} + +/*! + * @brief linear interpolation between two vectors (clamped) + * + * formula: from + t * (to - from) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) clamped between 0 and 1 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_mixc(vec4 from, vec4 to, float t, vec4 dest) { + glm_vec4_lerpc(from, to, t, dest); +} + +/*! + * @brief threshold function + * + * @param[in] edge threshold + * @param[in] x value to test against threshold + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_step(vec4 edge, vec4 x, vec4 dest) { + dest[0] = glm_step(edge[0], x[0]); + dest[1] = glm_step(edge[1], x[1]); + dest[2] = glm_step(edge[2], x[2]); + dest[3] = glm_step(edge[3], x[3]); +} + +/*! + * @brief threshold function with a smooth transition (unidimensional) + * + * @param[in] edge0 low threshold + * @param[in] edge1 high threshold + * @param[in] x value to test against threshold + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_smoothstep_uni(float edge0, float edge1, vec4 x, vec4 dest) { + dest[0] = glm_smoothstep(edge0, edge1, x[0]); + dest[1] = glm_smoothstep(edge0, edge1, x[1]); + dest[2] = glm_smoothstep(edge0, edge1, x[2]); + dest[3] = glm_smoothstep(edge0, edge1, x[3]); +} + +/*! + * @brief threshold function with a smooth transition + * + * @param[in] edge0 low threshold + * @param[in] edge1 high threshold + * @param[in] x value to test against threshold + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_smoothstep(vec4 edge0, vec4 edge1, vec4 x, vec4 dest) { + dest[0] = glm_smoothstep(edge0[0], edge1[0], x[0]); + dest[1] = glm_smoothstep(edge0[1], edge1[1], x[1]); + dest[2] = glm_smoothstep(edge0[2], edge1[2], x[2]); + dest[3] = glm_smoothstep(edge0[3], edge1[3], x[3]); +} + +/*! + * @brief smooth Hermite interpolation between two vectors + * + * formula: t^2 * (3 - 2*t) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_smoothinterp(vec4 from, vec4 to, float t, vec4 dest) { + vec4 s, v; + + /* from + smoothstep * (to - from) */ + glm_vec4_broadcast(glm_smooth(t), s); + glm_vec4_sub(to, from, v); + glm_vec4_mul(s, v, v); + glm_vec4_add(from, v, dest); +} + +/*! + * @brief smooth Hermite interpolation between two vectors (clamped) + * + * formula: t^2 * (3 - 2*t) + * + * @param[in] from from value + * @param[in] to to value + * @param[in] t interpolant (amount) clamped between 0 and 1 + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_smoothinterpc(vec4 from, vec4 to, float t, vec4 dest) { + glm_vec4_smoothinterp(from, to, glm_clamp_zo(t), dest); +} + +/*! + * @brief helper to fill vec4 as [S^3, S^2, S, 1] + * + * @param[in] s parameter + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_cubic(float s, vec4 dest) { + float ss; + + ss = s * s; + + dest[0] = ss * s; + dest[1] = ss; + dest[2] = s; + dest[3] = 1.0f; +} + +/*! + * @brief swizzle vector components + * + * you can use existing masks e.g. GLM_XXXX, GLM_WZYX + * + * @param[in] v source + * @param[in] mask mask + * @param[out] dest destination + */ +CGLM_INLINE +void +glm_vec4_swizzle(vec4 v, int mask, vec4 dest) { + vec4 t; + + t[0] = v[(mask & (3 << 0))]; + t[1] = v[(mask & (3 << 2)) >> 2]; + t[2] = v[(mask & (3 << 4)) >> 4]; + t[3] = v[(mask & (3 << 6)) >> 6]; + + glm_vec4_copy(t, dest); +} + +/*! + * @brief Create four dimensional vector from pointer + * + * @param[in] src pointer to an array of floats + * @param[out] dest destination vector + */ +CGLM_INLINE +void +glm_vec4_make(const float * __restrict src, vec4 dest) { + dest[0] = src[0]; dest[1] = src[1]; + dest[2] = src[2]; dest[3] = src[3]; +} + +/*! + * @brief reflection vector using an incident ray and a surface normal + * + * @param[in] v incident vector + * @param[in] n normalized normal vector + * @param[out] dest destination vector for the reflection result + */ +CGLM_INLINE +void +glm_vec4_reflect(vec4 v, vec4 n, vec4 dest) { + vec4 temp; + + /* TODO: direct simd touch */ + glm_vec4_scale(n, 2.0f * glm_vec4_dot(v, n), temp); + glm_vec4_sub(v, temp, dest); + + dest[3] = v[3]; +} + +/*! + * @brief computes refraction vector for an incident vector and a surface normal. + * + * calculates the refraction vector based on Snell's law. If total internal reflection + * occurs (angle too great given eta), dest is set to zero and returns false. + * Otherwise, computes refraction vector, stores it in dest, and returns true. + * + * this implementation does not explicitly preserve the 'w' component of the + * incident vector 'I' in the output 'dest', users requiring the preservation of + * the 'w' component should manually adjust 'dest' after calling this function. + * + * @param[in] v normalized incident vector + * @param[in] n normalized normal vector + * @param[in] eta ratio of indices of refraction (incident/transmitted) + * @param[out] dest refraction vector if refraction occurs; zero vector otherwise + * + * @returns true if refraction occurs; false if total internal reflection occurs. + */ +CGLM_INLINE +bool +glm_vec4_refract(vec4 v, vec4 n, float eta, vec4 dest) { + float ndi, eni, k; + + ndi = glm_vec4_dot(n, v); + eni = eta * ndi; + k = 1.0f - eta * eta + eni * eni; + + if (k < 0.0f) { + glm_vec4_zero(dest); + return false; + } + + glm_vec4_scale(v, eta, dest); + glm_vec4_mulsubs(n, eni + sqrtf(k), dest); + return true; +} + +#endif /* cglm_vec4_h */ diff --git a/cglm/include/cglm/version.h b/cglm/include/cglm/version.h new file mode 100644 index 0000000..9e815d4 --- /dev/null +++ b/cglm/include/cglm/version.h @@ -0,0 +1,15 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm_version_h +#define cglm_version_h + +#define CGLM_VERSION_MAJOR 0 +#define CGLM_VERSION_MINOR 9 +#define CGLM_VERSION_PATCH 6 + +#endif /* cglm_version_h */ diff --git a/cglm/include/module.modulemap b/cglm/include/module.modulemap new file mode 100644 index 0000000..40b8b9f --- /dev/null +++ b/cglm/include/module.modulemap @@ -0,0 +1,14 @@ +module cglm { + header "cglm/cglm.h" + header "cglm/struct.h" + + export * +} + +module cglmc { + header "cglm/cglm.h" + header "cglm/struct.h" + header "cglm/call.h" + + export * +} diff --git a/cglm/meson.build b/cglm/meson.build new file mode 100644 index 0000000..653c8d3 --- /dev/null +++ b/cglm/meson.build @@ -0,0 +1,142 @@ +project('cglm', 'c', + version : '0.9.6', + license : 'mit', + default_options : [ + 'c_std=c11', + 'warning_level=2', + 'buildtype=release' + ] +) + +cc = meson.get_compiler('c') + +cglm_install = get_option('install') +cglm_deps = cc.find_library('m', required : false) + +cglm_args = [] +build_args = [] + +if get_option('default_library') == 'static' + cglm_args += '-DCGLM_STATIC' +endif + +if cc.compiles( + 'int *test(char *p) { return (int*)__builtin_assume_aligned(p, 4); }', + name : '__builtin_assume_aligned test') + cglm_args += '-DCGLM_HAVE_BUILTIN_ASSUME_ALIGNED=1' +else + cglm_args += '-DCGLM_HAVE_BUILTIN_ASSUME_ALIGNED=0' +endif + +if host_machine.system() == 'windows' + build_args += '-DCGLM_EXPORTS' +endif + +cglm_inc = include_directories('include') + +cglm_src = files( + 'src/euler.c', + 'src/affine.c', + 'src/io.c', + 'src/quat.c', + 'src/cam.c', + 'src/vec2.c', + 'src/ivec2.c', + 'src/vec3.c', + 'src/ivec3.c', + 'src/vec4.c', + 'src/ivec4.c', + 'src/mat2.c', + 'src/mat2x3.c', + 'src/mat2x4.c', + 'src/mat3.c', + 'src/mat3x2.c', + 'src/mat3x4.c', + 'src/mat4.c', + 'src/mat4x2.c', + 'src/mat4x3.c', + 'src/plane.c', + 'src/noise.c', + 'src/frustum.c', + 'src/box.c', + 'src/aabb2d.c', + 'src/project.c', + 'src/sphere.c', + 'src/ease.c', + 'src/curve.c', + 'src/bezier.c', + 'src/ray.c', + 'src/affine2d.c', + 'src/clipspace/ortho_lh_no.c', + 'src/clipspace/ortho_lh_zo.c', + 'src/clipspace/ortho_rh_no.c', + 'src/clipspace/ortho_rh_zo.c', + 'src/clipspace/persp_lh_no.c', + 'src/clipspace/persp_lh_zo.c', + 'src/clipspace/persp_rh_no.c', + 'src/clipspace/persp_rh_zo.c', + 'src/clipspace/view_lh_no.c', + 'src/clipspace/view_lh_zo.c', + 'src/clipspace/view_rh_no.c', + 'src/clipspace/view_rh_zo.c', + 'src/clipspace/project_no.c', + 'src/clipspace/project_zo.c' +) + +cglm_lib = library('cglm', + cglm_src, + install : cglm_install, + dependencies : cglm_deps, + c_args : [ build_args, cglm_args ], + version : meson.project_version(), + soversion : '0', + build_by_default: not meson.is_subproject() +) + +cglm_dep = declare_dependency( + link_with : cglm_lib, + dependencies : cglm_deps, + compile_args : cglm_args, + include_directories : cglm_inc, + version : meson.project_version() +) + +if meson.version().version_compare('>= 0.54.0') + meson.override_dependency('cglm', cglm_dep) +endif + +if cglm_install + install_subdir('include/cglm', install_dir : get_option('includedir')) + + pkg = import('pkgconfig') + pkg.generate( + name : 'cglm', + libraries : cglm_lib, + extra_cflags : cglm_args, + version : meson.project_version(), + url : 'https://github.com/recp/cglm', + description : 'OpenGL Mathematics (glm) for C' + ) +endif + +if get_option('build_tests') == true + +test_src = files( + 'test/runner.c', + 'test/src/test_bezier.c', + 'test/src/test_clamp.c', + 'test/src/test_common.c', + 'test/src/test_euler.c', + 'test/src/tests.c', + 'test/src/test_struct.c', +) + +test_exe = executable('tests', + test_src, + dependencies : cglm_dep, + c_args : '-DGLM_TESTS_NO_COLORFUL_OUTPUT' +) + +test('cglm.tests', test_exe) + +endif diff --git a/cglm/meson_options.txt b/cglm/meson_options.txt new file mode 100644 index 0000000..fb3ced8 --- /dev/null +++ b/cglm/meson_options.txt @@ -0,0 +1,2 @@ +option('build_tests', type : 'boolean', value : false, description : 'Build tests') +option('install', type : 'boolean', value : true, description : 'Include the library, headers, and pkg-config file in the install target') diff --git a/cglm/src/aabb2d.c b/cglm/src/aabb2d.c new file mode 100644 index 0000000..2bace8f --- /dev/null +++ b/cglm/src/aabb2d.c @@ -0,0 +1,108 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_aabb2d_zero(vec2 aabb[2]) { + glm_aabb2d_zero(aabb); +} + +CGLM_EXPORT +void +glmc_aabb2d_copy(vec2 aabb[2], vec2 dest[2]) { + glm_aabb2d_copy(aabb, dest); +} + +CGLM_EXPORT +void +glmc_aabb2d_transform(vec2 aabb[2], mat3 m, vec2 dest[2]) { + glm_aabb2d_transform(aabb, m, dest); +} + +CGLM_EXPORT +void +glmc_aabb2d_merge(vec2 aabb1[2], vec2 aabb2[2], vec2 dest[2]) { + glm_aabb2d_merge(aabb1, aabb2, dest); +} + +CGLM_EXPORT +void +glmc_aabb2d_crop(vec2 aabb[2], vec2 cropAabb[2], vec2 dest[2]) { + glm_aabb2d_crop(aabb, cropAabb, dest); +} + +CGLM_EXPORT +void +glmc_aabb2d_crop_until(vec2 aabb[2], + vec2 cropAabb[2], + vec2 clampAabb[2], + vec2 dest[2]) { + glm_aabb2d_crop_until(aabb, cropAabb, clampAabb, dest); +} + +CGLM_EXPORT +void +glmc_aabb2d_invalidate(vec2 aabb[2]) { + glm_aabb2d_invalidate(aabb); +} + +CGLM_EXPORT +bool +glmc_aabb2d_isvalid(vec2 aabb[2]) { + return glm_aabb2d_isvalid(aabb); +} + +CGLM_EXPORT +float +glmc_aabb2d_diag(vec2 aabb[2]) { + return glm_aabb2d_diag(aabb); +} + +CGLM_EXPORT +void +glmc_aabb2d_sizev(vec2 aabb[2], vec2 dest) { + glm_aabb2d_sizev(aabb, dest); +} + +CGLM_EXPORT +float +glmc_aabb2d_radius(vec2 aabb[2]) { + return glm_aabb2d_radius(aabb); +} + +CGLM_EXPORT +void +glmc_aabb2d_center(vec2 aabb[2], vec2 dest) { + glm_aabb2d_center(aabb, dest); +} + +CGLM_EXPORT +bool +glmc_aabb2d_aabb(vec2 aabb[2], vec2 other[2]) { + return glm_aabb2d_aabb(aabb, other); +} + +CGLM_EXPORT +bool +glmc_aabb2d_point(vec2 aabb[2], vec2 point) { + return glm_aabb2d_point(aabb, point); +} + +CGLM_EXPORT +bool +glmc_aabb2d_contains(vec2 aabb[2], vec2 other[2]) { + return glm_aabb2d_contains(aabb, other); +} + +CGLM_EXPORT +bool +glmc_aabb2d_circle(vec2 aabb[2], vec3 s) { + return glm_aabb2d_circle(aabb, s); +} diff --git a/cglm/src/affine.c b/cglm/src/affine.c new file mode 100644 index 0000000..6bba523 --- /dev/null +++ b/cglm/src/affine.c @@ -0,0 +1,225 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_translate_make(mat4 m, vec3 v) { + glm_translate_make(m, v); +} + +CGLM_EXPORT +void +glmc_translate_to(mat4 m, vec3 v, mat4 dest) { + glm_translate_to(m, v, dest); +} + +CGLM_EXPORT +void +glmc_translate(mat4 m, vec3 v) { + glm_translate(m, v); +} + +CGLM_EXPORT +void +glmc_translate_x(mat4 m, float to) { + glm_translate_x(m, to); +} + +CGLM_EXPORT +void +glmc_translate_y(mat4 m, float to) { + glm_translate_y(m, to); +} + +CGLM_EXPORT +void +glmc_translate_z(mat4 m, float to) { + glm_translate_z(m, to); +} + +CGLM_EXPORT +void +glmc_scale_make(mat4 m, vec3 v) { + glm_scale_make(m, v); +} + +CGLM_EXPORT +void +glmc_scale_to(mat4 m, vec3 v, mat4 dest) { + glm_scale_to(m, v, dest); +} + +CGLM_EXPORT +void +glmc_scale(mat4 m, vec3 v) { + glm_scale(m, v); +} + +CGLM_EXPORT +void +glmc_scale_uni(mat4 m, float s) { + glm_scale_uni(m, s); +} + +CGLM_EXPORT +void +glmc_rotate_x(mat4 m, float rad, mat4 dest) { + glm_rotate_x(m, rad, dest); +} + +CGLM_EXPORT +void +glmc_rotate_y(mat4 m, float rad, mat4 dest) { + glm_rotate_y(m, rad, dest); +} + +CGLM_EXPORT +void +glmc_rotate_z(mat4 m, float rad, mat4 dest) { + glm_rotate_z(m, rad, dest); +} + +CGLM_EXPORT +void +glmc_rotate_make(mat4 m, float angle, vec3 axis) { + glm_rotate_make(m, angle, axis); +} + +CGLM_EXPORT +void +glmc_rotate(mat4 m, float angle, vec3 axis) { + glm_rotate(m, angle, axis); +} + +CGLM_EXPORT +void +glmc_rotate_at(mat4 m, vec3 pivot, float angle, vec3 axis) { + glm_rotate_at(m, pivot, angle, axis); +} + +CGLM_EXPORT +void +glmc_rotate_atm(mat4 m, vec3 pivot, float angle, vec3 axis) { + glm_rotate_atm(m, pivot, angle, axis); +} + +CGLM_EXPORT +void +glmc_spin(mat4 m, float angle, vec3 axis) { + glm_spin(m, angle, axis); +} + +CGLM_EXPORT +void +glmc_decompose_scalev(mat4 m, vec3 s) { + glm_decompose_scalev(m, s); +} + +CGLM_EXPORT +bool +glmc_uniscaled(mat4 m) { + return glm_uniscaled(m); +} + +CGLM_EXPORT +void +glmc_decompose_rs(mat4 m, mat4 r, vec3 s) { + glm_decompose_rs(m, r, s); +} + +CGLM_EXPORT +void +glmc_translated(mat4 m, vec3 v) { + glm_translated(m, v); +} + +CGLM_EXPORT +void +glmc_translated_to(mat4 m, vec3 v, mat4 dest) { + glm_translated_to(m, v, dest); +} + +CGLM_EXPORT +void +glmc_translated_x(mat4 m, float x) { + glm_translated_x(m, x); +} + +CGLM_EXPORT +void +glmc_translated_y(mat4 m, float y) { + glm_translated_y(m, y); +} + +CGLM_EXPORT +void +glmc_translated_z(mat4 m, float z) { + glm_translated_z(m, z); +} + +CGLM_EXPORT +void +glmc_rotated_x(mat4 m, float angle, mat4 dest) { + glm_rotated_x(m, angle, dest); +} + +CGLM_EXPORT +void +glmc_rotated_y(mat4 m, float angle, mat4 dest) { + glm_rotated_y(m, angle, dest); +} + +CGLM_EXPORT +void +glmc_rotated_z(mat4 m, float angle, mat4 dest) { + glm_rotated_z(m, angle, dest); +} + +CGLM_EXPORT +void +glmc_rotated(mat4 m, float angle, vec3 axis) { + glm_rotated(m, angle, axis); +} + +CGLM_EXPORT +void +glmc_rotated_at(mat4 m, vec3 pivot, float angle, vec3 axis) { + glm_rotated_at(m, pivot, angle, axis); +} + +CGLM_EXPORT +void +glmc_spinned(mat4 m, float angle, vec3 axis) { + glm_spinned(m, angle, axis); +} + +CGLM_EXPORT +void +glmc_decompose(mat4 m, vec4 t, mat4 r, vec3 s) { + glm_decompose(m, t, r, s); +} + +CGLM_EXPORT +void +glmc_mul(mat4 m1, mat4 m2, mat4 dest) { + glm_mul(m1, m2, dest); +} + +CGLM_EXPORT +void +glmc_mul_rot(mat4 m1, mat4 m2, mat4 dest) { + glm_mul_rot(m1, m2, dest); +} + +CGLM_EXPORT +void +glmc_inv_tr(mat4 mat) { + glm_inv_tr(mat); +} diff --git a/cglm/src/affine2d.c b/cglm/src/affine2d.c new file mode 100644 index 0000000..6e5913e --- /dev/null +++ b/cglm/src/affine2d.c @@ -0,0 +1,81 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_translate2d_make(mat3 m, vec2 v) { + glm_translate2d_make(m, v); +} + +CGLM_EXPORT +void +glmc_translate2d_to(mat3 m, vec2 v, mat3 dest) { + glm_translate2d_to(m, v, dest); +} + +CGLM_EXPORT +void +glmc_translate2d(mat3 m, vec2 v) { + glm_translate2d(m, v); +} + +CGLM_EXPORT +void +glmc_translate2d_x(mat3 m, float to) { + glm_translate2d_x(m, to); +} + +CGLM_EXPORT +void +glmc_translate2d_y(mat3 m, float to) { + glm_translate2d_y(m, to); +} + +CGLM_EXPORT +void +glmc_scale2d_to(mat3 m, vec2 v, mat3 dest) { + glm_scale2d_to(m, v, dest); +} + +CGLM_EXPORT +void +glmc_scale2d_make(mat3 m, vec2 v) { + glm_scale2d_make(m, v); +} + +CGLM_EXPORT +void +glmc_scale2d(mat3 m, vec2 v) { + glm_scale2d(m, v); +} + +CGLM_EXPORT +void +glmc_scale2d_uni(mat3 m, float s) { + glm_scale2d_uni(m, s); +} + +CGLM_EXPORT +void +glmc_rotate2d_make(mat3 m, float angle) { + glm_rotate2d_make(m, angle); +} + +CGLM_EXPORT +void +glmc_rotate2d(mat3 m, float angle) { + glm_rotate2d(m, angle); +} + +CGLM_EXPORT +void +glmc_rotate2d_to(mat3 m, float angle, mat3 dest) { + glm_rotate2d_to(m, angle, dest); +} diff --git a/cglm/src/bezier.c b/cglm/src/bezier.c new file mode 100644 index 0000000..21e6495 --- /dev/null +++ b/cglm/src/bezier.c @@ -0,0 +1,27 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +float +glmc_bezier(float s, float p0, float c0, float c1, float p1) { + return glm_bezier(s, p0, c0, c1, p1); +} + +CGLM_EXPORT +float +glmc_hermite(float s, float p0, float t0, float t1, float p1) { + return glm_hermite(s, p0, t0, t1, p1); +} + +CGLM_EXPORT +float +glmc_decasteljau(float prm, float p0, float c0, float c1, float p1) { + return glm_decasteljau(prm, p0, c0, c1, p1); +} diff --git a/cglm/src/box.c b/cglm/src/box.c new file mode 100644 index 0000000..fd639ea --- /dev/null +++ b/cglm/src/box.c @@ -0,0 +1,96 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_aabb_transform(vec3 box[2], mat4 m, vec3 dest[2]) { + glm_aabb_transform(box, m, dest); +} + +CGLM_EXPORT +void +glmc_aabb_merge(vec3 box1[2], vec3 box2[2], vec3 dest[2]) { + glm_aabb_merge(box1, box2, dest); +} + +CGLM_EXPORT +void +glmc_aabb_crop(vec3 box[2], vec3 cropBox[2], vec3 dest[2]) { + glm_aabb_crop(box, cropBox, dest); +} + +CGLM_EXPORT +void +glmc_aabb_crop_until(vec3 box[2], + vec3 cropBox[2], + vec3 clampBox[2], + vec3 dest[2]) { + glm_aabb_crop_until(box, cropBox, clampBox, dest); +} + +CGLM_EXPORT +bool +glmc_aabb_frustum(vec3 box[2], vec4 planes[6]) { + return glm_aabb_frustum(box, planes); +} + +CGLM_EXPORT +void +glmc_aabb_invalidate(vec3 box[2]) { + glm_aabb_invalidate(box); +} + +CGLM_EXPORT +bool +glmc_aabb_isvalid(vec3 box[2]) { + return glm_aabb_isvalid(box); +} + +CGLM_EXPORT +float +glmc_aabb_size(vec3 box[2]) { + return glm_aabb_size(box); +} + +CGLM_EXPORT +float +glmc_aabb_radius(vec3 box[2]) { + return glm_aabb_radius(box); +} + +CGLM_EXPORT +void +glmc_aabb_center(vec3 box[2], vec3 dest) { + glm_aabb_center(box, dest); +} + +CGLM_EXPORT +bool +glmc_aabb_aabb(vec3 box[2], vec3 other[2]) { + return glm_aabb_aabb(box, other); +} + +CGLM_EXPORT +bool +glmc_aabb_point(vec3 box[2], vec3 point) { + return glm_aabb_point(box, point); +} + +CGLM_EXPORT +bool +glmc_aabb_contains(vec3 box[2], vec3 other[2]) { + return glm_aabb_contains(box, other); +} + +CGLM_EXPORT +bool +glmc_aabb_sphere(vec3 box[2], vec4 s) { + return glm_aabb_sphere(box, s); +} diff --git a/cglm/src/cam.c b/cglm/src/cam.c new file mode 100644 index 0000000..40db351 --- /dev/null +++ b/cglm/src/cam.c @@ -0,0 +1,171 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_frustum(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + glm_frustum(left, right, bottom, top, nearZ, farZ, dest); +} + +CGLM_EXPORT +void +glmc_ortho(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + glm_ortho(left, right, bottom, top, nearZ, farZ, dest); +} + +CGLM_EXPORT +void +glmc_ortho_aabb(vec3 box[2], mat4 dest) { + glm_ortho_aabb(box, dest); +} + +CGLM_EXPORT +void +glmc_ortho_aabb_p(vec3 box[2], float padding, mat4 dest) { + glm_ortho_aabb_p(box, padding, dest); +} + +CGLM_EXPORT +void +glmc_ortho_aabb_pz(vec3 box[2], float padding, mat4 dest) { + glm_ortho_aabb_pz(box, padding, dest); +} + +CGLM_EXPORT +void +glmc_ortho_default(float aspect, mat4 dest) { + glm_ortho_default(aspect, dest); +} + +CGLM_EXPORT +void +glmc_ortho_default_s(float aspect, float size, mat4 dest) { + glm_ortho_default_s(aspect, size, dest); +} + +CGLM_EXPORT +void +glmc_perspective(float fovy, float aspect, float nearZ, float farZ, mat4 dest) { + glm_perspective(fovy, aspect, nearZ, farZ, dest); +} + +CGLM_EXPORT +void +glmc_persp_move_far(mat4 proj, float deltaFar) { + glm_persp_move_far(proj, deltaFar); +} + +CGLM_EXPORT +void +glmc_perspective_default(float aspect, mat4 dest) { + glm_perspective_default(aspect, dest); +} + +CGLM_EXPORT +void +glmc_perspective_resize(float aspect, mat4 proj) { + glm_perspective_resize(aspect, proj); +} + +CGLM_EXPORT +void +glmc_lookat(vec3 eye, vec3 center, vec3 up, mat4 dest) { + glm_lookat(eye, center, up, dest); +} + +CGLM_EXPORT +void +glmc_look(vec3 eye, vec3 dir, vec3 up, mat4 dest) { + glm_look(eye, dir, up, dest); +} + +CGLM_EXPORT +void +glmc_look_anyup(vec3 eye, vec3 dir, mat4 dest) { + glm_look_anyup(eye, dir, dest); +} + +CGLM_EXPORT +void +glmc_persp_decomp(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ, + float * __restrict top, + float * __restrict bottom, + float * __restrict left, + float * __restrict right) { + glm_persp_decomp(proj, nearZ, farZ, top, bottom, left, right); +} + +CGLM_EXPORT +void +glmc_persp_decompv(mat4 proj, float dest[6]) { + glm_persp_decompv(proj, dest); +} + +CGLM_EXPORT +void +glmc_persp_decomp_x(mat4 proj, + float * __restrict left, + float * __restrict right) { + glm_persp_decomp_x(proj, left, right); +} + +CGLM_EXPORT +void +glmc_persp_decomp_y(mat4 proj, + float * __restrict top, + float * __restrict bottom) { + glm_persp_decomp_y(proj, top, bottom); +} + +CGLM_EXPORT +void +glmc_persp_decomp_z(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ) { + glm_persp_decomp_z(proj, nearZ, farZ); +} + +CGLM_EXPORT +void +glmc_persp_decomp_far(mat4 proj, float * __restrict farZ) { + glm_persp_decomp_far(proj, farZ); +} + +CGLM_EXPORT +void +glmc_persp_decomp_near(mat4 proj, float * __restrict nearZ) { + glm_persp_decomp_near(proj, nearZ); +} + +CGLM_EXPORT +float +glmc_persp_fovy(mat4 proj) { + return glm_persp_fovy(proj); +} + +CGLM_EXPORT +float +glmc_persp_aspect(mat4 proj) { + return glm_persp_aspect(proj); +} + +CGLM_EXPORT +void +glmc_persp_sizes(mat4 proj, float fovy, vec4 dest) { + glm_persp_sizes(proj, fovy, dest); +} diff --git a/cglm/src/clipspace/ortho_lh_no.c b/cglm/src/clipspace/ortho_lh_no.c new file mode 100644 index 0000000..839926a --- /dev/null +++ b/cglm/src/clipspace/ortho_lh_no.c @@ -0,0 +1,51 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../../include/cglm/clipspace/ortho_lh_no.h" +#include "../../include/cglm/call/clipspace/ortho_lh_no.h" + +CGLM_EXPORT +void +glmc_ortho_lh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + glm_ortho_lh_no(left, right, + bottom, top, + nearZ, farZ, + dest); +} + +CGLM_EXPORT +void +glmc_ortho_aabb_lh_no(vec3 box[2], mat4 dest) { + glm_ortho_aabb_lh_no(box, dest); +} + +CGLM_EXPORT +void +glmc_ortho_aabb_p_lh_no(vec3 box[2], float padding, mat4 dest) { + glm_ortho_aabb_p_lh_no(box, padding, dest); +} + +CGLM_EXPORT +void +glmc_ortho_aabb_pz_lh_no(vec3 box[2], float padding, mat4 dest) { + glm_ortho_aabb_pz_lh_no(box, padding, dest); +} + +CGLM_EXPORT +void +glmc_ortho_default_lh_no(float aspect, mat4 dest) { + glm_ortho_default_lh_no(aspect, dest); +} + +CGLM_EXPORT +void +glmc_ortho_default_s_lh_no(float aspect, float size, mat4 dest) { + glm_ortho_default_s_lh_no(aspect, size, dest); +} diff --git a/cglm/src/clipspace/ortho_lh_zo.c b/cglm/src/clipspace/ortho_lh_zo.c new file mode 100644 index 0000000..88291ac --- /dev/null +++ b/cglm/src/clipspace/ortho_lh_zo.c @@ -0,0 +1,51 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../../include/cglm/clipspace/ortho_lh_zo.h" +#include "../../include/cglm/call/clipspace/ortho_lh_zo.h" + +CGLM_EXPORT +void +glmc_ortho_lh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + glm_ortho_lh_zo(left, right, + bottom, top, + nearZ, farZ, + dest); +} + +CGLM_EXPORT +void +glmc_ortho_aabb_lh_zo(vec3 box[2], mat4 dest) { + glm_ortho_aabb_lh_zo(box, dest); +} + +CGLM_EXPORT +void +glmc_ortho_aabb_p_lh_zo(vec3 box[2], float padding, mat4 dest) { + glm_ortho_aabb_p_lh_zo(box, padding, dest); +} + +CGLM_EXPORT +void +glmc_ortho_aabb_pz_lh_zo(vec3 box[2], float padding, mat4 dest) { + glm_ortho_aabb_pz_lh_zo(box, padding, dest); +} + +CGLM_EXPORT +void +glmc_ortho_default_lh_zo(float aspect, mat4 dest) { + glm_ortho_default_lh_zo(aspect, dest); +} + +CGLM_EXPORT +void +glmc_ortho_default_s_lh_zo(float aspect, float size, mat4 dest) { + glm_ortho_default_s_lh_zo(aspect, size, dest); +} diff --git a/cglm/src/clipspace/ortho_rh_no.c b/cglm/src/clipspace/ortho_rh_no.c new file mode 100644 index 0000000..ca30e82 --- /dev/null +++ b/cglm/src/clipspace/ortho_rh_no.c @@ -0,0 +1,51 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../../include/cglm/clipspace/ortho_rh_no.h" +#include "../../include/cglm/call/clipspace/ortho_rh_no.h" + +CGLM_EXPORT +void +glmc_ortho_rh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + glm_ortho_rh_no(left, right, + bottom, top, + nearZ, farZ, + dest); +} + +CGLM_EXPORT +void +glmc_ortho_aabb_rh_no(vec3 box[2], mat4 dest) { + glm_ortho_aabb_rh_no(box, dest); +} + +CGLM_EXPORT +void +glmc_ortho_aabb_p_rh_no(vec3 box[2], float padding, mat4 dest) { + glm_ortho_aabb_p_rh_no(box, padding, dest); +} + +CGLM_EXPORT +void +glmc_ortho_aabb_pz_rh_no(vec3 box[2], float padding, mat4 dest) { + glm_ortho_aabb_pz_rh_no(box, padding, dest); +} + +CGLM_EXPORT +void +glmc_ortho_default_rh_no(float aspect, mat4 dest) { + glm_ortho_default_rh_no(aspect, dest); +} + +CGLM_EXPORT +void +glmc_ortho_default_s_rh_no(float aspect, float size, mat4 dest) { + glm_ortho_default_s_rh_no(aspect, size, dest); +} diff --git a/cglm/src/clipspace/ortho_rh_zo.c b/cglm/src/clipspace/ortho_rh_zo.c new file mode 100644 index 0000000..06d119b --- /dev/null +++ b/cglm/src/clipspace/ortho_rh_zo.c @@ -0,0 +1,51 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../../include/cglm/clipspace/ortho_rh_zo.h" +#include "../../include/cglm/call/clipspace/ortho_rh_zo.h" + +CGLM_EXPORT +void +glmc_ortho_rh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + glm_ortho_rh_zo(left, right, + bottom, top, + nearZ, farZ, + dest); +} + +CGLM_EXPORT +void +glmc_ortho_aabb_rh_zo(vec3 box[2], mat4 dest) { + glm_ortho_aabb_rh_zo(box, dest); +} + +CGLM_EXPORT +void +glmc_ortho_aabb_p_rh_zo(vec3 box[2], float padding, mat4 dest) { + glm_ortho_aabb_p_rh_zo(box, padding, dest); +} + +CGLM_EXPORT +void +glmc_ortho_aabb_pz_rh_zo(vec3 box[2], float padding, mat4 dest) { + glm_ortho_aabb_pz_rh_zo(box, padding, dest); +} + +CGLM_EXPORT +void +glmc_ortho_default_rh_zo(float aspect, mat4 dest) { + glm_ortho_default_rh_zo(aspect, dest); +} + +CGLM_EXPORT +void +glmc_ortho_default_s_rh_zo(float aspect, float size, mat4 dest) { + glm_ortho_default_s_rh_zo(aspect, size, dest); +} diff --git a/cglm/src/clipspace/persp_lh_no.c b/cglm/src/clipspace/persp_lh_no.c new file mode 100644 index 0000000..8d6db76 --- /dev/null +++ b/cglm/src/clipspace/persp_lh_no.c @@ -0,0 +1,110 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../../include/cglm/clipspace/persp_lh_no.h" +#include "../../include/cglm/call/clipspace/persp_lh_no.h" + +CGLM_EXPORT +void +glmc_frustum_lh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + glm_frustum_lh_no(left, right, + bottom, top, + nearZ, farZ, + dest); +} + +CGLM_EXPORT +void +glmc_perspective_lh_no(float fovy, + float aspect, + float nearVal, + float farVal, + mat4 dest) { + glm_perspective_lh_no(fovy, + aspect, + nearVal, + farVal, + dest); +} + +CGLM_EXPORT +void +glmc_persp_move_far_lh_no(mat4 proj, float deltaFar) { + glm_persp_move_far_lh_no(proj, deltaFar); +} + +CGLM_EXPORT +void +glmc_persp_decomp_lh_no(mat4 proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right) { + glm_persp_decomp_lh_no(proj, nearZ, farZ, top, bottom, left, right); +} + +CGLM_EXPORT +void +glmc_persp_decompv_lh_no(mat4 proj, float dest[6]) { + glm_persp_decompv_lh_no(proj, dest); +} + +CGLM_EXPORT +void +glmc_persp_decomp_x_lh_no(mat4 proj, + float * __restrict left, + float * __restrict right) { + glm_persp_decomp_x_lh_no(proj, left, right); +} + +CGLM_EXPORT +void +glmc_persp_decomp_y_lh_no(mat4 proj, + float * __restrict top, + float * __restrict bottom) { + glm_persp_decomp_y_lh_no(proj, top, bottom); +} + +CGLM_EXPORT +void +glmc_persp_decomp_z_lh_no(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ) { + glm_persp_decomp_z_lh_no(proj, nearZ, farZ); +} + +CGLM_EXPORT +void +glmc_persp_decomp_far_lh_no(mat4 proj, float * __restrict farZ) { + glm_persp_decomp_far_lh_no(proj, farZ); +} + +CGLM_EXPORT +void +glmc_persp_decomp_near_lh_no(mat4 proj, float * __restrict nearZ) { + glm_persp_decomp_near_lh_no(proj, nearZ); +} + +CGLM_EXPORT +void +glmc_persp_sizes_lh_no(mat4 proj, float fovy, vec4 dest) { + glm_persp_sizes_lh_no(proj, fovy, dest); +} + +CGLM_EXPORT +float +glmc_persp_fovy_lh_no(mat4 proj) { + return glm_persp_fovy_lh_no(proj); +} + +CGLM_EXPORT +float +glmc_persp_aspect_lh_no(mat4 proj) { + return glm_persp_aspect_lh_no(proj); +} diff --git a/cglm/src/clipspace/persp_lh_zo.c b/cglm/src/clipspace/persp_lh_zo.c new file mode 100644 index 0000000..d9fec0c --- /dev/null +++ b/cglm/src/clipspace/persp_lh_zo.c @@ -0,0 +1,110 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../../include/cglm/clipspace/persp_lh_zo.h" +#include "../../include/cglm/call/clipspace/persp_lh_zo.h" + +CGLM_EXPORT +void +glmc_frustum_lh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + glm_frustum_lh_zo(left, right, + bottom, top, + nearZ, farZ, + dest); +} + +CGLM_EXPORT +void +glmc_perspective_lh_zo(float fovy, + float aspect, + float nearVal, + float farVal, + mat4 dest) { + glm_perspective_lh_zo(fovy, + aspect, + nearVal, + farVal, + dest); +} + +CGLM_EXPORT +void +glmc_persp_move_far_lh_zo(mat4 proj, float deltaFar) { + glm_persp_move_far_lh_zo(proj, deltaFar); +} + +CGLM_EXPORT +void +glmc_persp_decomp_lh_zo(mat4 proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right) { + glm_persp_decomp_lh_zo(proj, nearZ, farZ, top, bottom, left, right); +} + +CGLM_EXPORT +void +glmc_persp_decompv_lh_zo(mat4 proj, float dest[6]) { + glm_persp_decompv_lh_zo(proj, dest); +} + +CGLM_EXPORT +void +glmc_persp_decomp_x_lh_zo(mat4 proj, + float * __restrict left, + float * __restrict right) { + glm_persp_decomp_x_lh_zo(proj, left, right); +} + +CGLM_EXPORT +void +glmc_persp_decomp_y_lh_zo(mat4 proj, + float * __restrict top, + float * __restrict bottom) { + glm_persp_decomp_y_lh_zo(proj, top, bottom); +} + +CGLM_EXPORT +void +glmc_persp_decomp_z_lh_zo(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ) { + glm_persp_decomp_z_lh_zo(proj, nearZ, farZ); +} + +CGLM_EXPORT +void +glmc_persp_decomp_far_lh_zo(mat4 proj, float * __restrict farZ) { + glm_persp_decomp_far_lh_zo(proj, farZ); +} + +CGLM_EXPORT +void +glmc_persp_decomp_near_lh_zo(mat4 proj, float * __restrict nearZ) { + glm_persp_decomp_near_lh_zo(proj, nearZ); +} + +CGLM_EXPORT +void +glmc_persp_sizes_lh_zo(mat4 proj, float fovy, vec4 dest) { + glm_persp_sizes_lh_zo(proj, fovy, dest); +} + +CGLM_EXPORT +float +glmc_persp_fovy_lh_zo(mat4 proj) { + return glm_persp_fovy_lh_zo(proj); +} + +CGLM_EXPORT +float +glmc_persp_aspect_lh_zo(mat4 proj) { + return glm_persp_aspect_lh_zo(proj); +} diff --git a/cglm/src/clipspace/persp_rh_no.c b/cglm/src/clipspace/persp_rh_no.c new file mode 100644 index 0000000..8fc7735 --- /dev/null +++ b/cglm/src/clipspace/persp_rh_no.c @@ -0,0 +1,110 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../../include/cglm/clipspace/persp_rh_no.h" +#include "../../include/cglm/call/clipspace/persp_rh_no.h" + +CGLM_EXPORT +void +glmc_frustum_rh_no(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + glm_frustum_rh_no(left, right, + bottom, top, + nearZ, farZ, + dest); +} + +CGLM_EXPORT +void +glmc_perspective_rh_no(float fovy, + float aspect, + float nearVal, + float farVal, + mat4 dest) { + glm_perspective_rh_no(fovy, + aspect, + nearVal, + farVal, + dest); +} + +CGLM_EXPORT +void +glmc_persp_move_far_rh_no(mat4 proj, float deltaFar) { + glm_persp_move_far_rh_no(proj, deltaFar); +} + +CGLM_EXPORT +void +glmc_persp_decomp_rh_no(mat4 proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right) { + glm_persp_decomp_rh_no(proj, nearZ, farZ, top, bottom, left, right); +} + +CGLM_EXPORT +void +glmc_persp_decompv_rh_no(mat4 proj, float dest[6]) { + glm_persp_decompv_rh_no(proj, dest); +} + +CGLM_EXPORT +void +glmc_persp_decomp_x_rh_no(mat4 proj, + float * __restrict left, + float * __restrict right) { + glm_persp_decomp_x_rh_no(proj, left, right); +} + +CGLM_EXPORT +void +glmc_persp_decomp_y_rh_no(mat4 proj, + float * __restrict top, + float * __restrict bottom) { + glm_persp_decomp_y_rh_no(proj, top, bottom); +} + +CGLM_EXPORT +void +glmc_persp_decomp_z_rh_no(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ) { + glm_persp_decomp_z_rh_no(proj, nearZ, farZ); +} + +CGLM_EXPORT +void +glmc_persp_decomp_far_rh_no(mat4 proj, float * __restrict farZ) { + glm_persp_decomp_far_rh_no(proj, farZ); +} + +CGLM_EXPORT +void +glmc_persp_decomp_near_rh_no(mat4 proj, float * __restrict nearZ) { + glm_persp_decomp_near_rh_no(proj, nearZ); +} + +CGLM_EXPORT +void +glmc_persp_sizes_rh_no(mat4 proj, float fovy, vec4 dest) { + glm_persp_sizes_rh_no(proj, fovy, dest); +} + +CGLM_EXPORT +float +glmc_persp_fovy_rh_no(mat4 proj) { + return glm_persp_fovy_rh_no(proj); +} + +CGLM_EXPORT +float +glmc_persp_aspect_rh_no(mat4 proj) { + return glm_persp_aspect_rh_no(proj); +} diff --git a/cglm/src/clipspace/persp_rh_zo.c b/cglm/src/clipspace/persp_rh_zo.c new file mode 100644 index 0000000..50190f2 --- /dev/null +++ b/cglm/src/clipspace/persp_rh_zo.c @@ -0,0 +1,110 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../../include/cglm/clipspace/persp_rh_zo.h" +#include "../../include/cglm/call/clipspace/persp_rh_zo.h" + +CGLM_EXPORT +void +glmc_frustum_rh_zo(float left, float right, + float bottom, float top, + float nearZ, float farZ, + mat4 dest) { + glm_frustum_rh_zo(left, right, + bottom, top, + nearZ, farZ, + dest); +} + +CGLM_EXPORT +void +glmc_perspective_rh_zo(float fovy, + float aspect, + float nearVal, + float farVal, + mat4 dest) { + glm_perspective_rh_zo(fovy, + aspect, + nearVal, + farVal, + dest); +} + +CGLM_EXPORT +void +glmc_persp_move_far_rh_zo(mat4 proj, float deltaFar) { + glm_persp_move_far_rh_zo(proj, deltaFar); +} + +CGLM_EXPORT +void +glmc_persp_decomp_rh_zo(mat4 proj, + float * __restrict nearZ, float * __restrict farZ, + float * __restrict top, float * __restrict bottom, + float * __restrict left, float * __restrict right) { + glm_persp_decomp_rh_zo(proj, nearZ, farZ, top, bottom, left, right); +} + +CGLM_EXPORT +void +glmc_persp_decompv_rh_zo(mat4 proj, float dest[6]) { + glm_persp_decompv_rh_zo(proj, dest); +} + +CGLM_EXPORT +void +glmc_persp_decomp_x_rh_zo(mat4 proj, + float * __restrict left, + float * __restrict right) { + glm_persp_decomp_x_rh_zo(proj, left, right); +} + +CGLM_EXPORT +void +glmc_persp_decomp_y_rh_zo(mat4 proj, + float * __restrict top, + float * __restrict bottom) { + glm_persp_decomp_y_rh_zo(proj, top, bottom); +} + +CGLM_EXPORT +void +glmc_persp_decomp_z_rh_zo(mat4 proj, + float * __restrict nearZ, + float * __restrict farZ) { + glm_persp_decomp_z_rh_zo(proj, nearZ, farZ); +} + +CGLM_EXPORT +void +glmc_persp_decomp_far_rh_zo(mat4 proj, float * __restrict farZ) { + glm_persp_decomp_far_rh_zo(proj, farZ); +} + +CGLM_EXPORT +void +glmc_persp_decomp_near_rh_zo(mat4 proj, float * __restrict nearZ) { + glm_persp_decomp_near_rh_zo(proj, nearZ); +} + +CGLM_EXPORT +void +glmc_persp_sizes_rh_zo(mat4 proj, float fovy, vec4 dest) { + glm_persp_sizes_rh_zo(proj, fovy, dest); +} + +CGLM_EXPORT +float +glmc_persp_fovy_rh_zo(mat4 proj) { + return glm_persp_fovy_rh_zo(proj); +} + +CGLM_EXPORT +float +glmc_persp_aspect_rh_zo(mat4 proj) { + return glm_persp_aspect_rh_zo(proj); +} diff --git a/cglm/src/clipspace/project_no.c b/cglm/src/clipspace/project_no.c new file mode 100644 index 0000000..93b1453 --- /dev/null +++ b/cglm/src/clipspace/project_no.c @@ -0,0 +1,27 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../../include/cglm/clipspace/project_no.h" +#include "../../include/cglm/call/clipspace/project_no.h" + +CGLM_EXPORT +void +glmc_unprojecti_no(vec3 pos, mat4 invMat, vec4 vp, vec3 dest) { + glm_unprojecti_no(pos, invMat, vp, dest); +} + +CGLM_EXPORT +void +glmc_project_no(vec3 pos, mat4 m, vec4 vp, vec3 dest) { + glm_project_no(pos, m, vp, dest); +} + +CGLM_EXPORT +float +glmc_project_z_no(vec3 pos, mat4 m) { + return glm_project_z_no(pos, m); +} diff --git a/cglm/src/clipspace/project_zo.c b/cglm/src/clipspace/project_zo.c new file mode 100644 index 0000000..6699be9 --- /dev/null +++ b/cglm/src/clipspace/project_zo.c @@ -0,0 +1,27 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../../include/cglm/clipspace/project_zo.h" +#include "../../include/cglm/call/clipspace/project_zo.h" + +CGLM_EXPORT +void +glmc_unprojecti_zo(vec3 pos, mat4 invMat, vec4 vp, vec3 dest) { + glm_unprojecti_zo(pos, invMat, vp, dest); +} + +CGLM_EXPORT +void +glmc_project_zo(vec3 pos, mat4 m, vec4 vp, vec3 dest) { + glm_project_zo(pos, m, vp, dest); +} + +CGLM_EXPORT +float +glmc_project_z_zo(vec3 pos, mat4 m) { + return glm_project_z_zo(pos, m); +} diff --git a/cglm/src/clipspace/view_lh_no.c b/cglm/src/clipspace/view_lh_no.c new file mode 100644 index 0000000..39f2a9d --- /dev/null +++ b/cglm/src/clipspace/view_lh_no.c @@ -0,0 +1,27 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../../include/cglm/clipspace/view_lh_no.h" +#include "../../include/cglm/call/clipspace/view_lh_no.h" + +CGLM_EXPORT +void +glmc_lookat_lh_no(vec3 eye, vec3 center, vec3 up, mat4 dest) { + glm_lookat_lh_no(eye, center, up, dest); +} + +CGLM_EXPORT +void +glmc_look_lh_no(vec3 eye, vec3 dir, vec3 up, mat4 dest) { + glm_look_lh_no(eye, dir, up, dest); +} + +CGLM_EXPORT +void +glmc_look_anyup_lh_no(vec3 eye, vec3 dir, mat4 dest) { + glm_look_anyup_lh_no(eye, dir, dest); +} diff --git a/cglm/src/clipspace/view_lh_zo.c b/cglm/src/clipspace/view_lh_zo.c new file mode 100644 index 0000000..a8680d9 --- /dev/null +++ b/cglm/src/clipspace/view_lh_zo.c @@ -0,0 +1,27 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../../include/cglm/clipspace/view_lh_zo.h" +#include "../../include/cglm/call/clipspace/view_lh_zo.h" + +CGLM_EXPORT +void +glmc_lookat_lh_zo(vec3 eye, vec3 center, vec3 up, mat4 dest) { + glm_lookat_lh_zo(eye, center, up, dest); +} + +CGLM_EXPORT +void +glmc_look_lh_zo(vec3 eye, vec3 dir, vec3 up, mat4 dest) { + glm_look_lh_zo(eye, dir, up, dest); +} + +CGLM_EXPORT +void +glmc_look_anyup_lh_zo(vec3 eye, vec3 dir, mat4 dest) { + glm_look_anyup_lh_zo(eye, dir, dest); +} diff --git a/cglm/src/clipspace/view_rh_no.c b/cglm/src/clipspace/view_rh_no.c new file mode 100644 index 0000000..6d60c08 --- /dev/null +++ b/cglm/src/clipspace/view_rh_no.c @@ -0,0 +1,27 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../../include/cglm/clipspace/view_rh_no.h" +#include "../../include/cglm/call/clipspace/view_rh_no.h" + +CGLM_EXPORT +void +glmc_lookat_rh_no(vec3 eye, vec3 center, vec3 up, mat4 dest) { + glm_lookat_rh_no(eye, center, up, dest); +} + +CGLM_EXPORT +void +glmc_look_rh_no(vec3 eye, vec3 dir, vec3 up, mat4 dest) { + glm_look_rh_no(eye, dir, up, dest); +} + +CGLM_EXPORT +void +glmc_look_anyup_rh_no(vec3 eye, vec3 dir, mat4 dest) { + glm_look_anyup_rh_no(eye, dir, dest); +} diff --git a/cglm/src/clipspace/view_rh_zo.c b/cglm/src/clipspace/view_rh_zo.c new file mode 100644 index 0000000..5133fda --- /dev/null +++ b/cglm/src/clipspace/view_rh_zo.c @@ -0,0 +1,27 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../../include/cglm/clipspace/view_rh_zo.h" +#include "../../include/cglm/call/clipspace/view_rh_zo.h" + +CGLM_EXPORT +void +glmc_lookat_rh_zo(vec3 eye, vec3 center, vec3 up, mat4 dest) { + glm_lookat_rh_zo(eye, center, up, dest); +} + +CGLM_EXPORT +void +glmc_look_rh_zo(vec3 eye, vec3 dir, vec3 up, mat4 dest) { + glm_look_rh_zo(eye, dir, up, dest); +} + +CGLM_EXPORT +void +glmc_look_anyup_rh_zo(vec3 eye, vec3 dir, mat4 dest) { + glm_look_anyup_rh_zo(eye, dir, dest); +} diff --git a/cglm/src/config.h b/cglm/src/config.h new file mode 100644 index 0000000..ddec761 --- /dev/null +++ b/cglm/src/config.h @@ -0,0 +1,22 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef cglm__config__h_ +#define cglm__config__h_ + +#if defined(_WIN32) || defined(WIN32) + +/* Exclude rarely-used stuff from Windows headers */ +# define WIN32_LEAN_AND_MEAN +# include + +/* Windows Header Files: */ +# include + +#endif + +#endif /* cglm__config__h_ */ diff --git a/cglm/src/curve.c b/cglm/src/curve.c new file mode 100644 index 0000000..74d4702 --- /dev/null +++ b/cglm/src/curve.c @@ -0,0 +1,15 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +float +glmc_smc(float s, mat4 m, vec4 c) { + return glm_smc(s, m, c); +} diff --git a/cglm/src/ease.c b/cglm/src/ease.c new file mode 100644 index 0000000..702dfce --- /dev/null +++ b/cglm/src/ease.c @@ -0,0 +1,195 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +float +glmc_ease_linear(float t) { + return glm_ease_linear(t); +} + +CGLM_EXPORT +float +glmc_ease_sine_in(float t) { + return glm_ease_sine_in(t); +} + +CGLM_EXPORT +float +glmc_ease_sine_out(float t) { + return glm_ease_sine_out(t); +} + +CGLM_EXPORT +float +glmc_ease_sine_inout(float t) { + return glm_ease_sine_inout(t); +} + +CGLM_EXPORT +float +glmc_ease_quad_in(float t) { + return glm_ease_quad_in(t); +} + +CGLM_EXPORT +float +glmc_ease_quad_out(float t) { + return glm_ease_quad_out(t); +} + +CGLM_EXPORT +float +glmc_ease_quad_inout(float t) { + return glm_ease_quad_inout(t); +} + +CGLM_EXPORT +float +glmc_ease_cubic_in(float t) { + return glm_ease_cubic_in(t); +} + +CGLM_EXPORT +float +glmc_ease_cubic_out(float t) { + return glm_ease_cubic_out(t); +} + +CGLM_EXPORT +float +glmc_ease_cubic_inout(float t) { + return glm_ease_cubic_inout(t); +} + +CGLM_EXPORT +float +glmc_ease_quart_in(float t) { + return glm_ease_quart_in(t); +} + +CGLM_EXPORT +float +glmc_ease_quart_out(float t) { + return glm_ease_quart_out(t); +} + +CGLM_EXPORT +float +glmc_ease_quart_inout(float t) { + return glm_ease_quart_inout(t); +} + +CGLM_EXPORT +float +glmc_ease_quint_in(float t) { + return glm_ease_quint_in(t); +} + +CGLM_EXPORT +float +glmc_ease_quint_out(float t) { + return glm_ease_quint_out(t); +} + +CGLM_EXPORT +float +glmc_ease_quint_inout(float t) { + return glm_ease_quint_inout(t); +} + +CGLM_EXPORT +float +glmc_ease_exp_in(float t) { + return glm_ease_exp_in(t); +} + +CGLM_EXPORT +float +glmc_ease_exp_out(float t) { + return glm_ease_exp_out(t); +} + +CGLM_EXPORT +float +glmc_ease_exp_inout(float t) { + return glm_ease_exp_inout(t); +} + +CGLM_EXPORT +float +glmc_ease_circ_in(float t) { + return glm_ease_circ_in(t); +} + +CGLM_EXPORT +float +glmc_ease_circ_out(float t) { + return glm_ease_circ_out(t); +} + +CGLM_EXPORT +float +glmc_ease_circ_inout(float t) { + return glm_ease_circ_inout(t); +} + +CGLM_EXPORT +float +glmc_ease_back_in(float t) { + return glm_ease_back_in(t); +} + +CGLM_EXPORT +float +glmc_ease_back_out(float t) { + return glm_ease_back_out(t); +} + +CGLM_EXPORT +float +glmc_ease_back_inout(float t) { + return glm_ease_back_inout(t); +} + +CGLM_EXPORT +float +glmc_ease_elast_in(float t) { + return glm_ease_elast_in(t); +} + +CGLM_EXPORT +float +glmc_ease_elast_out(float t) { + return glm_ease_elast_out(t); +} + +CGLM_EXPORT +float +glmc_ease_elast_inout(float t) { + return glm_ease_elast_inout(t); +} + +CGLM_EXPORT +float +glmc_ease_bounce_out(float t) { + return glm_ease_bounce_out(t); +} + +CGLM_EXPORT +float +glmc_ease_bounce_in(float t) { + return glm_ease_bounce_in(t); +} + +CGLM_EXPORT +float +glmc_ease_bounce_inout(float t) { + return glm_ease_bounce_inout(t); +} diff --git a/cglm/src/euler.c b/cglm/src/euler.c new file mode 100644 index 0000000..8749ba5 --- /dev/null +++ b/cglm/src/euler.c @@ -0,0 +1,99 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_euler_angles(mat4 m, vec3 dest) { + glm_euler_angles(m, dest); +} + +CGLM_EXPORT +void +glmc_euler(vec3 angles, mat4 dest) { + glm_euler(angles, dest); +} + +CGLM_EXPORT +void +glmc_euler_xyz(vec3 angles, mat4 dest) { + glm_euler_xyz(angles, dest); +} + +CGLM_EXPORT +void +glmc_euler_zyx(vec3 angles, mat4 dest) { + glm_euler_zyx(angles, dest); +} + +CGLM_EXPORT +void +glmc_euler_zxy(vec3 angles, mat4 dest) { + glm_euler_zxy(angles, dest); +} + +CGLM_EXPORT +void +glmc_euler_xzy(vec3 angles, mat4 dest) { + glm_euler_xzy(angles, dest); +} + +CGLM_EXPORT +void +glmc_euler_yzx(vec3 angles, mat4 dest) { + glm_euler_yzx(angles, dest); +} + +CGLM_EXPORT +void +glmc_euler_yxz(vec3 angles, mat4 dest) { + glm_euler_yxz(angles, dest); +} + +CGLM_EXPORT +void +glmc_euler_by_order(vec3 angles, glm_euler_seq axis, mat4 dest) { + glm_euler_by_order(angles, axis, dest); +} + +CGLM_EXPORT +void +glmc_euler_xyz_quat(vec3 angles, versor dest) { + glm_euler_xyz_quat(angles, dest); +} + +CGLM_EXPORT +void +glmc_euler_xzy_quat(vec3 angles, versor dest) { + glm_euler_xzy_quat(angles, dest); +} + +CGLM_EXPORT +void +glmc_euler_yxz_quat(vec3 angles, versor dest) { + glm_euler_yxz_quat(angles, dest); +} + +CGLM_EXPORT +void +glmc_euler_yzx_quat(vec3 angles, versor dest) { + glm_euler_yzx_quat(angles, dest); +} + +CGLM_EXPORT +void +glmc_euler_zxy_quat(vec3 angles, versor dest) { + glm_euler_zxy_quat(angles, dest); +} + +CGLM_EXPORT +void +glmc_euler_zyx_quat(vec3 angles, versor dest) { + glm_euler_zyx_quat(angles, dest); +} diff --git a/cglm/src/frustum.c b/cglm/src/frustum.c new file mode 100644 index 0000000..312c3d3 --- /dev/null +++ b/cglm/src/frustum.c @@ -0,0 +1,42 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_frustum_planes(mat4 m, vec4 dest[6]) { + glm_frustum_planes(m, dest); +} + +CGLM_EXPORT +void +glmc_frustum_corners(mat4 invMat, vec4 dest[8]) { + glm_frustum_corners(invMat, dest); +} + +CGLM_EXPORT +void +glmc_frustum_center(vec4 corners[8], vec4 dest) { + glm_frustum_center(corners, dest); +} + +CGLM_EXPORT +void +glmc_frustum_box(vec4 corners[8], mat4 m, vec3 box[2]) { + glm_frustum_box(corners, m, box); +} + +CGLM_EXPORT +void +glmc_frustum_corners_at(vec4 corners[8], + float splitDist, + float farDist, + vec4 planeCorners[4]) { + glm_frustum_corners_at(corners, splitDist, farDist, planeCorners); +} diff --git a/cglm/src/io.c b/cglm/src/io.c new file mode 100644 index 0000000..fd81dac --- /dev/null +++ b/cglm/src/io.c @@ -0,0 +1,46 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#define CGLM_LIB_SRC + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_mat4_print(mat4 matrix, + FILE * __restrict ostream) { + glm_mat4_print(matrix, ostream); +} + +CGLM_EXPORT +void +glmc_mat3_print(mat3 matrix, + FILE * __restrict ostream) { + glm_mat3_print(matrix, ostream); +} + +CGLM_EXPORT +void +glmc_vec4_print(vec4 vec, + FILE * __restrict ostream) { + glm_vec4_print(vec, ostream); +} + +CGLM_EXPORT +void +glmc_vec3_print(vec3 vec, + FILE * __restrict ostream) { + glm_vec3_print(vec, ostream); +} + +CGLM_EXPORT +void +glmc_versor_print(versor vec, + FILE * __restrict ostream) { + glm_versor_print(vec, ostream); +} diff --git a/cglm/src/ivec2.c b/cglm/src/ivec2.c new file mode 100644 index 0000000..2fd0993 --- /dev/null +++ b/cglm/src/ivec2.c @@ -0,0 +1,249 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_ivec2(int * __restrict v, ivec2 dest) { + glm_ivec2(v, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_copy(ivec2 a, ivec2 dest) { + glm_ivec2_copy(a, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_zero(ivec2 v) { + glm_ivec2_zero(v); +} + +CGLM_EXPORT +void +glmc_ivec2_one(ivec2 v) { + glm_ivec2_one(v); +} + +CGLM_EXPORT +int +glmc_ivec2_dot(ivec2 a, ivec2 b) { + return glm_ivec2_dot(a, b); +} + +CGLM_EXPORT +int +glmc_ivec2_cross(ivec2 a, ivec2 b) { + return glm_ivec2_cross(a, b); +} + +CGLM_EXPORT +void +glmc_ivec2_add(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_add(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_adds(ivec2 v, int s, ivec2 dest) { + glm_ivec2_adds(v, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_sub(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_sub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_subs(ivec2 v, int s, ivec2 dest) { + glm_ivec2_subs(v, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_mul(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_mul(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_scale(ivec2 v, int s, ivec2 dest) { + glm_ivec2_scale(v, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_div(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_div(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_divs(ivec2 v, int s, ivec2 dest) { + glm_ivec2_divs(v, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_mod(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_mod(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_addadd(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_addadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_addadds(ivec2 a, int s, ivec2 dest) { + glm_ivec2_addadds(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_subadd(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_subadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_subadds(ivec2 a, int s, ivec2 dest) { + glm_ivec2_subadds(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_muladd(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_muladd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_muladds(ivec2 a, int s, ivec2 dest) { + glm_ivec2_muladds(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_maxadd(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_maxadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_minadd(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_minadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_subsub(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_subsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_subsubs(ivec2 a, int s, ivec2 dest) { + glm_ivec2_subsubs(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_addsub(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_addsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_addsubs(ivec2 a, int s, ivec2 dest) { + glm_ivec2_addsubs(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_mulsub(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_mulsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_mulsubs(ivec2 a, int s, ivec2 dest) { + glm_ivec2_mulsubs(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_maxsub(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_maxsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_minsub(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_minsub(a, b, dest); +} + +CGLM_EXPORT +int +glmc_ivec2_distance2(ivec2 a, ivec2 b) { + return glm_ivec2_distance2(a, b); +} + +CGLM_EXPORT +float +glmc_ivec2_distance(ivec2 a, ivec2 b) { + return glm_ivec2_distance(a, b); +} + +CGLM_EXPORT +void +glmc_ivec2_fill(ivec2 v, int val) { + glm_ivec2_fill(v, val); +} + +CGLM_EXPORT +bool +glmc_ivec2_eq(ivec2 v, int val) { + return glm_ivec2_eq(v, val); +} + +CGLM_EXPORT +bool +glmc_ivec2_eqv(ivec2 a, ivec2 b) { + return glm_ivec2_eqv(a, b); +} + +CGLM_EXPORT +void +glmc_ivec2_maxv(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_maxv(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_minv(ivec2 a, ivec2 b, ivec2 dest) { + glm_ivec2_minv(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec2_clamp(ivec2 v, int minVal, int maxVal) { + glm_ivec2_clamp(v, minVal, maxVal); +} + +CGLM_EXPORT +void +glmc_ivec2_abs(ivec2 v, ivec2 dest) { + glm_ivec2_abs(v, dest); +} diff --git a/cglm/src/ivec3.c b/cglm/src/ivec3.c new file mode 100644 index 0000000..4a2ea2f --- /dev/null +++ b/cglm/src/ivec3.c @@ -0,0 +1,255 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_ivec3(ivec4 v4, ivec3 dest) { + glm_ivec3(v4, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_copy(ivec3 a, ivec3 dest) { + glm_ivec3_copy(a, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_zero(ivec3 v) { + glm_ivec3_zero(v); +} + +CGLM_EXPORT +void +glmc_ivec3_one(ivec3 v) { + glm_ivec3_one(v); +} + +CGLM_EXPORT +int +glmc_ivec3_dot(ivec3 a, ivec3 b) { + return glm_ivec3_dot(a, b); +} + +CGLM_EXPORT +int +glmc_ivec3_norm2(ivec3 v) { + return glm_ivec3_norm2(v); +} + +CGLM_EXPORT +int +glmc_ivec3_norm(ivec3 v) { + return glm_ivec3_norm(v); +} + +CGLM_EXPORT +void +glmc_ivec3_add(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_add(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_adds(ivec3 v, int s, ivec3 dest) { + glm_ivec3_adds(v, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_sub(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_sub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_subs(ivec3 v, int s, ivec3 dest) { + glm_ivec3_subs(v, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_mul(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_mul(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_scale(ivec3 v, int s, ivec3 dest) { + glm_ivec3_scale(v, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_div(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_div(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_divs(ivec3 v, int s, ivec3 dest) { + glm_ivec3_divs(v, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_mod(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_mod(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_addadd(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_addadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_addadds(ivec3 a, int s, ivec3 dest) { + glm_ivec3_addadds(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_subadd(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_subadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_subadds(ivec3 a, int s, ivec3 dest) { + glm_ivec3_subadds(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_muladd(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_muladd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_muladds(ivec3 a, int s, ivec3 dest) { + glm_ivec3_muladds(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_maxadd(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_maxadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_minadd(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_minadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_subsub(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_subsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_subsubs(ivec3 a, int s, ivec3 dest) { + glm_ivec3_subsubs(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_addsub(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_addsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_addsubs(ivec3 a, int s, ivec3 dest) { + glm_ivec3_addsubs(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_mulsub(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_mulsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_mulsubs(ivec3 a, int s, ivec3 dest) { + glm_ivec3_mulsubs(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_maxsub(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_maxsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_minsub(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_minsub(a, b, dest); +} + +CGLM_EXPORT +int +glmc_ivec3_distance2(ivec3 a, ivec3 b) { + return glm_ivec3_distance2(a, b); +} + +CGLM_EXPORT +float +glmc_ivec3_distance(ivec3 a, ivec3 b) { + return glm_ivec3_distance(a, b); +} + +CGLM_EXPORT +void +glmc_ivec3_fill(ivec3 v, int val) { + glm_ivec3_fill(v, val); +} + +CGLM_EXPORT +bool +glmc_ivec3_eq(ivec3 v, int val) { + return glm_ivec3_eq(v, val); +} + +CGLM_EXPORT +bool +glmc_ivec3_eqv(ivec3 a, ivec3 b) { + return glm_ivec3_eqv(a, b); +} + +CGLM_EXPORT +void +glmc_ivec3_maxv(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_maxv(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_minv(ivec3 a, ivec3 b, ivec3 dest) { + glm_ivec3_minv(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec3_clamp(ivec3 v, int minVal, int maxVal) { + glm_ivec3_clamp(v, minVal, maxVal); +} + +CGLM_EXPORT +void +glmc_ivec3_abs(ivec3 v, ivec3 dest) { + glm_ivec3_abs(v, dest); +} diff --git a/cglm/src/ivec4.c b/cglm/src/ivec4.c new file mode 100644 index 0000000..1e976a6 --- /dev/null +++ b/cglm/src/ivec4.c @@ -0,0 +1,201 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_ivec4(ivec3 v3, int last, ivec4 dest) { + glm_ivec4(v3, last, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_copy(ivec4 a, ivec4 dest) { + glm_ivec4_copy(a, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_zero(ivec4 v) { + glm_ivec4_zero(v); +} + +CGLM_EXPORT +void +glmc_ivec4_one(ivec4 v) { + glm_ivec4_one(v); +} + +CGLM_EXPORT +void +glmc_ivec4_add(ivec4 a, ivec4 b, ivec4 dest) { + glm_ivec4_add(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_adds(ivec4 v, int s, ivec4 dest) { + glm_ivec4_adds(v, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_sub(ivec4 a, ivec4 b, ivec4 dest) { + glm_ivec4_sub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_subs(ivec4 v, int s, ivec4 dest) { + glm_ivec4_subs(v, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_mul(ivec4 a, ivec4 b, ivec4 dest) { + glm_ivec4_mul(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_scale(ivec4 v, int s, ivec4 dest) { + glm_ivec4_scale(v, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_addadd(ivec4 a, ivec4 b, ivec4 dest) { + glm_ivec4_addadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_addadds(ivec4 a, int s, ivec4 dest) { + glm_ivec4_addadds(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_subadd(ivec4 a, ivec4 b, ivec4 dest) { + glm_ivec4_subadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_subadds(ivec4 a, int s, ivec4 dest) { + glm_ivec4_subadds(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_muladd(ivec4 a, ivec4 b, ivec4 dest) { + glm_ivec4_muladd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_muladds(ivec4 a, int s, ivec4 dest) { + glm_ivec4_muladds(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_maxadd(ivec4 a, ivec4 b, ivec4 dest) { + glm_ivec4_maxadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_minadd(ivec4 a, ivec4 b, ivec4 dest) { + glm_ivec4_minadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_subsub(ivec4 a, ivec4 b, ivec4 dest) { + glm_ivec4_subsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_subsubs(ivec4 a, int s, ivec4 dest) { + glm_ivec4_subsubs(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_addsub(ivec4 a, ivec4 b, ivec4 dest) { + glm_ivec4_addsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_addsubs(ivec4 a, int s, ivec4 dest) { + glm_ivec4_addsubs(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_mulsub(ivec4 a, ivec4 b, ivec4 dest) { + glm_ivec4_mulsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_mulsubs(ivec4 a, int s, ivec4 dest) { + glm_ivec4_mulsubs(a, s, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_maxsub(ivec4 a, ivec4 b, ivec4 dest) { + glm_ivec4_maxsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_minsub(ivec4 a, ivec4 b, ivec4 dest) { + glm_ivec4_minsub(a, b, dest); +} + +CGLM_EXPORT +int +glmc_ivec4_distance2(ivec4 a, ivec4 b) { + return glm_ivec4_distance2(a, b); +} + +CGLM_EXPORT +float +glmc_ivec4_distance(ivec4 a, ivec4 b) { + return glm_ivec4_distance(a, b); +} + +CGLM_EXPORT +void +glmc_ivec4_maxv(ivec4 a, ivec4 b, ivec4 dest) { + glm_ivec4_maxv(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_minv(ivec4 a, ivec4 b, ivec4 dest) { + glm_ivec4_minv(a, b, dest); +} + +CGLM_EXPORT +void +glmc_ivec4_clamp(ivec4 v, int minVal, int maxVal) { + glm_ivec4_clamp(v, minVal, maxVal); +} + +CGLM_EXPORT +void +glmc_ivec4_abs(ivec4 v, ivec4 dest) { + glm_ivec4_abs(v, dest); +} diff --git a/cglm/src/mat2.c b/cglm/src/mat2.c new file mode 100644 index 0000000..3f0750c --- /dev/null +++ b/cglm/src/mat2.c @@ -0,0 +1,105 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_mat2_copy(mat2 mat, mat2 dest) { + glm_mat2_copy(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat2_identity(mat2 mat) { + glm_mat2_identity(mat); +} + +CGLM_EXPORT +void +glmc_mat2_identity_array(mat2 * __restrict mat, size_t count) { + glm_mat2_identity_array(mat, count); +} + +CGLM_EXPORT +void +glmc_mat2_zero(mat2 mat) { + glm_mat2_zero(mat); +} + +CGLM_EXPORT +void +glmc_mat2_mul(mat2 m1, mat2 m2, mat2 dest) { + glm_mat2_mul(m1, m2, dest); +} + +CGLM_EXPORT +void +glmc_mat2_transpose_to(mat2 m, mat2 dest) { + glm_mat2_transpose_to(m, dest); +} + +CGLM_EXPORT +void +glmc_mat2_transpose(mat2 m) { + glm_mat2_transpose(m); +} + +CGLM_EXPORT +void +glmc_mat2_mulv(mat2 m, vec2 v, vec2 dest) { + glm_mat2_mulv(m, v, dest); +} + +CGLM_EXPORT +float +glmc_mat2_trace(mat2 m) { + return glm_mat2_trace(m); +} + +CGLM_EXPORT +void +glmc_mat2_scale(mat2 m, float s) { + glm_mat2_scale(m, s); +} + +CGLM_EXPORT +float +glmc_mat2_det(mat2 mat) { + return glm_mat2_det(mat); +} + +CGLM_EXPORT +void +glmc_mat2_inv(mat2 mat, mat2 dest) { + glm_mat2_inv(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat2_swap_col(mat2 mat, int col1, int col2) { + glm_mat2_swap_col(mat, col1, col2); +} + +CGLM_EXPORT +void +glmc_mat2_swap_row(mat2 mat, int row1, int row2) { + glm_mat2_swap_row(mat, row1, row2); +} + +CGLM_EXPORT +float +glmc_mat2_rmc(vec2 r, mat2 m, vec2 c) { + return glm_mat2_rmc(r, m, c); +} + +CGLM_EXPORT +void +glmc_mat2_make(const float * __restrict src, mat2 dest) { + glm_mat2_make(src, dest); +} diff --git a/cglm/src/mat2x3.c b/cglm/src/mat2x3.c new file mode 100644 index 0000000..3397bf2 --- /dev/null +++ b/cglm/src/mat2x3.c @@ -0,0 +1,51 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_mat2x3_copy(mat2x3 mat, mat2x3 dest) { + glm_mat2x3_copy(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat2x3_zero(mat2x3 mat) { + glm_mat2x3_zero(mat); +} + +CGLM_EXPORT +void +glmc_mat2x3_make(const float * __restrict src, mat2x3 dest) { + glm_mat2x3_make(src, dest); +} + +CGLM_EXPORT +void +glmc_mat2x3_mul(mat2x3 m1, mat3x2 m2, mat3 dest) { + glm_mat2x3_mul(m1, m2, dest); +} + +CGLM_EXPORT +void +glmc_mat2x3_mulv(mat2x3 m, vec2 v, vec3 dest) { + glm_mat2x3_mulv(m, v, dest); +} + +CGLM_EXPORT +void +glmc_mat2x3_transpose(mat2x3 m, mat3x2 dest) { + glm_mat2x3_transpose(m, dest); +} + +CGLM_EXPORT +void +glmc_mat2x3_scale(mat2x3 m, float s) { + glm_mat2x3_scale(m, s); +} diff --git a/cglm/src/mat2x4.c b/cglm/src/mat2x4.c new file mode 100644 index 0000000..a221ed3 --- /dev/null +++ b/cglm/src/mat2x4.c @@ -0,0 +1,51 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_mat2x4_copy(mat2x4 mat, mat2x4 dest) { + glm_mat2x4_copy(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat2x4_zero(mat2x4 mat) { + glm_mat2x4_zero(mat); +} + +CGLM_EXPORT +void +glmc_mat2x4_make(const float * __restrict src, mat2x4 dest) { + glm_mat2x4_make(src, dest); +} + +CGLM_EXPORT +void +glmc_mat2x4_mul(mat2x4 m1, mat4x2 m2, mat4 dest) { + glm_mat2x4_mul(m1, m2, dest); +} + +CGLM_EXPORT +void +glmc_mat2x4_mulv(mat2x4 m, vec2 v, vec4 dest) { + glm_mat2x4_mulv(m, v, dest); +} + +CGLM_EXPORT +void +glmc_mat2x4_transpose(mat2x4 m, mat4x2 dest) { + glm_mat2x4_transpose(m, dest); +} + +CGLM_EXPORT +void +glmc_mat2x4_scale(mat2x4 m, float s) { + glm_mat2x4_scale(m, s); +} diff --git a/cglm/src/mat3.c b/cglm/src/mat3.c new file mode 100644 index 0000000..a325a4c --- /dev/null +++ b/cglm/src/mat3.c @@ -0,0 +1,111 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_mat3_copy(mat3 mat, mat3 dest) { + glm_mat3_copy(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat3_identity(mat3 mat) { + glm_mat3_identity(mat); +} + +CGLM_EXPORT +void +glmc_mat3_zero(mat3 mat) { + glm_mat3_zero(mat); +} + +CGLM_EXPORT +void +glmc_mat3_identity_array(mat3 * __restrict mat, size_t count) { + glm_mat3_identity_array(mat, count); +} + +CGLM_EXPORT +void +glmc_mat3_mul(mat3 m1, mat3 m2, mat3 dest) { + glm_mat3_mul(m1, m2, dest); +} + +CGLM_EXPORT +void +glmc_mat3_transpose_to(mat3 m, mat3 dest) { + glm_mat3_transpose_to(m, dest); +} + +CGLM_EXPORT +void +glmc_mat3_transpose(mat3 m) { + glm_mat3_transpose(m); +} + +CGLM_EXPORT +void +glmc_mat3_mulv(mat3 m, vec3 v, vec3 dest) { + glm_mat3_mulv(m, v, dest); +} + +CGLM_EXPORT +float +glmc_mat3_trace(mat3 m) { + return glm_mat3_trace(m); +} + +CGLM_EXPORT +void +glmc_mat3_quat(mat3 m, versor dest) { + glm_mat3_quat(m, dest); +} + +CGLM_EXPORT +void +glmc_mat3_scale(mat3 m, float s) { + glm_mat3_scale(m, s); +} + +CGLM_EXPORT +float +glmc_mat3_det(mat3 mat) { + return glm_mat3_det(mat); +} + +CGLM_EXPORT +void +glmc_mat3_inv(mat3 mat, mat3 dest) { + glm_mat3_inv(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat3_swap_col(mat3 mat, int col1, int col2) { + glm_mat3_swap_col(mat, col1, col2); +} + +CGLM_EXPORT +void +glmc_mat3_swap_row(mat3 mat, int row1, int row2) { + glm_mat3_swap_row(mat, row1, row2); +} + +CGLM_EXPORT +float +glmc_mat3_rmc(vec3 r, mat3 m, vec3 c) { + return glm_mat3_rmc(r, m, c); +} + +CGLM_EXPORT +void +glmc_mat3_make(const float * __restrict src, mat3 dest) { + glm_mat3_make(src, dest); +} diff --git a/cglm/src/mat3x2.c b/cglm/src/mat3x2.c new file mode 100644 index 0000000..5ade303 --- /dev/null +++ b/cglm/src/mat3x2.c @@ -0,0 +1,51 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_mat3x2_copy(mat3x2 mat, mat3x2 dest) { + glm_mat3x2_copy(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat3x2_zero(mat3x2 mat) { + glm_mat3x2_zero(mat); +} + +CGLM_EXPORT +void +glmc_mat3x2_make(const float * __restrict src, mat3x2 dest) { + glm_mat3x2_make(src, dest); +} + +CGLM_EXPORT +void +glmc_mat3x2_mul(mat3x2 m1, mat2x3 m2, mat2 dest) { + glm_mat3x2_mul(m1, m2, dest); +} + +CGLM_EXPORT +void +glmc_mat3x2_mulv(mat3x2 m, vec3 v, vec2 dest) { + glm_mat3x2_mulv(m, v, dest); +} + +CGLM_EXPORT +void +glmc_mat3x2_transpose(mat3x2 m, mat2x3 dest) { + glm_mat3x2_transpose(m, dest); +} + +CGLM_EXPORT +void +glmc_mat3x2_scale(mat3x2 m, float s) { + glm_mat3x2_scale(m, s); +} diff --git a/cglm/src/mat3x4.c b/cglm/src/mat3x4.c new file mode 100644 index 0000000..75963c9 --- /dev/null +++ b/cglm/src/mat3x4.c @@ -0,0 +1,51 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_mat3x4_copy(mat3x4 mat, mat3x4 dest) { + glm_mat3x4_copy(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat3x4_zero(mat3x4 mat) { + glm_mat3x4_zero(mat); +} + +CGLM_EXPORT +void +glmc_mat3x4_make(const float * __restrict src, mat3x4 dest) { + glm_mat3x4_make(src, dest); +} + +CGLM_EXPORT +void +glmc_mat3x4_mul(mat3x4 m1, mat4x3 m2, mat4 dest) { + glm_mat3x4_mul(m1, m2, dest); +} + +CGLM_EXPORT +void +glmc_mat3x4_mulv(mat3x4 m, vec3 v, vec4 dest) { + glm_mat3x4_mulv(m, v, dest); +} + +CGLM_EXPORT +void +glmc_mat3x4_transpose(mat3x4 m, mat4x3 dest) { + glm_mat3x4_transpose(m, dest); +} + +CGLM_EXPORT +void +glmc_mat3x4_scale(mat3x4 m, float s) { + glm_mat3x4_scale(m, s); +} diff --git a/cglm/src/mat4.c b/cglm/src/mat4.c new file mode 100644 index 0000000..c4d1111 --- /dev/null +++ b/cglm/src/mat4.c @@ -0,0 +1,171 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_mat4_ucopy(mat4 mat, mat4 dest) { + glm_mat4_copy(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat4_copy(mat4 mat, mat4 dest) { + glm_mat4_copy(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat4_identity(mat4 mat) { + glm_mat4_identity(mat); +} + +CGLM_EXPORT +void +glmc_mat4_identity_array(mat4 * __restrict mat, size_t count) { + glm_mat4_identity_array(mat, count); +} + +CGLM_EXPORT +void +glmc_mat4_zero(mat4 mat) { + glm_mat4_zero(mat); +} + +CGLM_EXPORT +void +glmc_mat4_pick3(mat4 mat, mat3 dest) { + glm_mat4_pick3(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat4_pick3t(mat4 mat, mat3 dest) { + glm_mat4_pick3t(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat4_ins3(mat3 mat, mat4 dest) { + glm_mat4_ins3(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat4_mul(mat4 m1, mat4 m2, mat4 dest) { + glm_mat4_mul(m1, m2, dest); +} + +CGLM_EXPORT +void +glmc_mat4_mulN(mat4 * __restrict matrices[], uint32_t len, mat4 dest) { + glm_mat4_mulN(matrices, len, dest); +} + +CGLM_EXPORT +void +glmc_mat4_mulv(mat4 m, vec4 v, vec4 dest) { + glm_mat4_mulv(m, v, dest); +} + +CGLM_EXPORT +void +glmc_mat4_mulv3(mat4 m, vec3 v, float last, vec3 dest) { + glm_mat4_mulv3(m, v, last, dest); +} + +CGLM_EXPORT +float +glmc_mat4_trace(mat4 m) { + return glm_mat4_trace(m); +} + +CGLM_EXPORT +float +glmc_mat4_trace3(mat4 m) { + return glm_mat4_trace3(m); +} + +CGLM_EXPORT +void +glmc_mat4_quat(mat4 m, versor dest) { + glm_mat4_quat(m, dest); +} + +CGLM_EXPORT +void +glmc_mat4_transpose_to(mat4 m, mat4 dest) { + glm_mat4_transpose_to(m, dest); +} + +CGLM_EXPORT +void +glmc_mat4_transpose(mat4 m) { + glm_mat4_transpose(m); +} + +CGLM_EXPORT +void +glmc_mat4_scale_p(mat4 m, float s) { + glm_mat4_scale_p(m, s); +} + +CGLM_EXPORT +void +glmc_mat4_scale(mat4 m, float s) { + glm_mat4_scale(m, s); +} + +CGLM_EXPORT +float +glmc_mat4_det(mat4 mat) { + return glm_mat4_det(mat); +} + +CGLM_EXPORT +void +glmc_mat4_inv(mat4 mat, mat4 dest) { + glm_mat4_inv(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat4_inv_precise(mat4 mat, mat4 dest) { + glm_mat4_inv_precise(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat4_inv_fast(mat4 mat, mat4 dest) { + glm_mat4_inv_fast(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat4_swap_col(mat4 mat, int col1, int col2) { + glm_mat4_swap_col(mat, col1, col2); +} + +CGLM_EXPORT +void +glmc_mat4_swap_row(mat4 mat, int row1, int row2) { + glm_mat4_swap_row(mat, row1, row2); +} + +CGLM_EXPORT +float +glmc_mat4_rmc(vec4 r, mat4 m, vec4 c) { + return glm_mat4_rmc(r, m, c); +} + +CGLM_EXPORT +void +glmc_mat4_make(const float * __restrict src, mat4 dest) { + glm_mat4_make(src, dest); +} diff --git a/cglm/src/mat4x2.c b/cglm/src/mat4x2.c new file mode 100644 index 0000000..32015f2 --- /dev/null +++ b/cglm/src/mat4x2.c @@ -0,0 +1,51 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_mat4x2_copy(mat4x2 mat, mat4x2 dest) { + glm_mat4x2_copy(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat4x2_zero(mat4x2 mat) { + glm_mat4x2_zero(mat); +} + +CGLM_EXPORT +void +glmc_mat4x2_make(const float * __restrict src, mat4x2 dest) { + glm_mat4x2_make(src, dest); +} + +CGLM_EXPORT +void +glmc_mat4x2_mul(mat4x2 m1, mat2x4 m2, mat2 dest) { + glm_mat4x2_mul(m1, m2, dest); +} + +CGLM_EXPORT +void +glmc_mat4x2_mulv(mat4x2 m, vec4 v, vec2 dest) { + glm_mat4x2_mulv(m, v, dest); +} + +CGLM_EXPORT +void +glmc_mat4x2_transpose(mat4x2 m, mat2x4 dest) { + glm_mat4x2_transpose(m, dest); +} + +CGLM_EXPORT +void +glmc_mat4x2_scale(mat4x2 m, float s) { + glm_mat4x2_scale(m, s); +} diff --git a/cglm/src/mat4x3.c b/cglm/src/mat4x3.c new file mode 100644 index 0000000..8285424 --- /dev/null +++ b/cglm/src/mat4x3.c @@ -0,0 +1,51 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_mat4x3_copy(mat4x3 mat, mat4x3 dest) { + glm_mat4x3_copy(mat, dest); +} + +CGLM_EXPORT +void +glmc_mat4x3_zero(mat4x3 mat) { + glm_mat4x3_zero(mat); +} + +CGLM_EXPORT +void +glmc_mat4x3_make(const float * __restrict src, mat4x3 dest) { + glm_mat4x3_make(src, dest); +} + +CGLM_EXPORT +void +glmc_mat4x3_mul(mat4x3 m1, mat3x4 m2, mat3 dest) { + glm_mat4x3_mul(m1, m2, dest); +} + +CGLM_EXPORT +void +glmc_mat4x3_mulv(mat4x3 m, vec4 v, vec3 dest) { + glm_mat4x3_mulv(m, v, dest); +} + +CGLM_EXPORT +void +glmc_mat4x3_transpose(mat4x3 m, mat3x4 dest) { + glm_mat4x3_transpose(m, dest); +} + +CGLM_EXPORT +void +glmc_mat4x3_scale(mat4x3 m, float s) { + glm_mat4x3_scale(m, s); +} diff --git a/cglm/src/noise.c b/cglm/src/noise.c new file mode 100644 index 0000000..f464202 --- /dev/null +++ b/cglm/src/noise.c @@ -0,0 +1,27 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +float +glmc_perlin_vec4(vec4 p) { + return glm_perlin_vec4(p); +} + +CGLM_EXPORT +float +glmc_perlin_vec3(vec3 p) { + return glm_perlin_vec3(p); +} + +CGLM_EXPORT +float +glmc_perlin_vec2(vec2 p) { + return glm_perlin_vec2(p); +} \ No newline at end of file diff --git a/cglm/src/plane.c b/cglm/src/plane.c new file mode 100644 index 0000000..7ee0c0f --- /dev/null +++ b/cglm/src/plane.c @@ -0,0 +1,15 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_plane_normalize(vec4 plane) { + glm_plane_normalize(plane); +} diff --git a/cglm/src/project.c b/cglm/src/project.c new file mode 100644 index 0000000..4d22337 --- /dev/null +++ b/cglm/src/project.c @@ -0,0 +1,39 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_unprojecti(vec3 pos, mat4 invMat, vec4 vp, vec3 dest) { + glm_unprojecti(pos, invMat, vp, dest); +} + +CGLM_EXPORT +void +glmc_unproject(vec3 pos, mat4 m, vec4 vp, vec3 dest) { + glm_unproject(pos, m, vp, dest); +} + +CGLM_EXPORT +void +glmc_project(vec3 pos, mat4 m, vec4 vp, vec3 dest) { + glm_project(pos, m, vp, dest); +} + +CGLM_EXPORT +float +glmc_project_z(vec3 pos, mat4 m) { + return glm_project_z(pos, m); +} + +CGLM_EXPORT +void +glmc_pickmatrix(vec2 center, vec2 size, vec4 vp, mat4 dest) { + glm_pickmatrix(center, size, vp, dest); +} diff --git a/cglm/src/quat.c b/cglm/src/quat.c new file mode 100644 index 0000000..3f41286 --- /dev/null +++ b/cglm/src/quat.c @@ -0,0 +1,243 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_quat_identity(versor q) { + glm_quat_identity(q); +} + +CGLM_EXPORT +void +glmc_quat_identity_array(versor * __restrict q, size_t count) { + glm_quat_identity_array(q, count); +} + +CGLM_EXPORT +void +glmc_quat_init(versor q, float x, float y, float z, float w) { + glm_quat_init(q, x, y, z, w); +} + +CGLM_EXPORT +void +glmc_quat(versor q, float angle, float x, float y, float z) { + glm_quat(q, angle, x, y, z); +} + +CGLM_EXPORT +void +glmc_quatv(versor q, float angle, vec3 axis) { + glm_quatv(q, angle, axis); +} + +CGLM_EXPORT +void +glmc_quat_copy(versor q, versor dest) { + glm_quat_copy(q, dest); +} + +CGLM_EXPORT +void +glmc_quat_from_vecs(vec3 a, vec3 b, versor dest) { + glm_quat_from_vecs(a, b, dest); +} + +CGLM_EXPORT +float +glmc_quat_norm(versor q) { + return glm_quat_norm(q); +} + +CGLM_EXPORT +void +glmc_quat_normalize_to(versor q, versor dest) { + glm_quat_normalize_to(q, dest); +} + +CGLM_EXPORT +void +glmc_quat_normalize(versor q) { + glm_quat_normalize(q); +} + +CGLM_EXPORT +float +glmc_quat_dot(versor p, versor q) { + return glm_quat_dot(p, q); +} + +CGLM_EXPORT +void +glmc_quat_conjugate(versor q, versor dest) { + glm_quat_conjugate(q, dest); +} + +CGLM_EXPORT +void +glmc_quat_inv(versor q, versor dest) { + glm_quat_inv(q, dest); +} + +CGLM_EXPORT +void +glmc_quat_add(versor p, versor q, versor dest) { + glm_quat_add(p, q, dest); +} + +CGLM_EXPORT +void +glmc_quat_sub(versor p, versor q, versor dest) { + glm_quat_sub(p, q, dest); +} + +CGLM_EXPORT +float +glmc_quat_real(versor q) { + return glm_quat_real(q); +} + +CGLM_EXPORT +void +glmc_quat_imag(versor q, vec3 dest) { + glm_quat_imag(q, dest); +} + +CGLM_EXPORT +void +glmc_quat_imagn(versor q, vec3 dest) { + glm_quat_imagn(q, dest); +} + +CGLM_EXPORT +float +glmc_quat_imaglen(versor q) { + return glm_quat_imaglen(q); +} + +CGLM_EXPORT +float +glmc_quat_angle(versor q) { + return glm_quat_angle(q); +} + +CGLM_EXPORT +void +glmc_quat_axis(versor q, vec3 dest) { + glm_quat_axis(q, dest); +} + +CGLM_EXPORT +void +glmc_quat_mul(versor p, versor q, versor dest) { + glm_quat_mul(p, q, dest); +} + +CGLM_EXPORT +void +glmc_quat_mat4(versor q, mat4 dest) { + glm_quat_mat4(q, dest); +} + +CGLM_EXPORT +void +glmc_quat_mat4t(versor q, mat4 dest) { + glm_quat_mat4t(q, dest); +} + +CGLM_EXPORT +void +glmc_quat_mat3(versor q, mat3 dest) { + glm_quat_mat3(q, dest); +} + +CGLM_EXPORT +void +glmc_quat_mat3t(versor q, mat3 dest) { + glm_quat_mat3t(q, dest); +} + +CGLM_EXPORT +void +glmc_quat_lerp(versor from, versor to, float t, versor dest) { + glm_quat_lerp(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_quat_lerpc(versor from, versor to, float t, versor dest) { + glm_quat_lerpc(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_quat_nlerp(versor from, versor to, float t, versor dest) { + glm_quat_nlerp(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_quat_slerp(versor from, versor to, float t, versor dest) { + glm_quat_slerp(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_quat_slerp_longest(versor from, versor to, float t, versor dest) { + glm_quat_slerp_longest(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_quat_look(vec3 eye, versor ori, mat4 dest) { + glm_quat_look(eye, ori, dest); +} + +CGLM_EXPORT +void +glmc_quat_for(vec3 dir, vec3 up, versor dest) { + glm_quat_for(dir, up, dest); +} + +CGLM_EXPORT +void +glmc_quat_forp(vec3 from, vec3 to, vec3 up, versor dest) { + glm_quat_forp(from, to, up, dest); +} + +CGLM_EXPORT +void +glmc_quat_rotatev(versor q, vec3 v, vec3 dest) { + glm_quat_rotatev(q, v, dest); +} + +CGLM_EXPORT +void +glmc_quat_rotate(mat4 m, versor q, mat4 dest) { + glm_quat_rotate(m, q, dest); +} + +CGLM_EXPORT +void +glmc_quat_rotate_at(mat4 model, versor q, vec3 pivot) { + glm_quat_rotate_at(model, q, pivot); +} + +CGLM_EXPORT +void +glmc_quat_rotate_atm(mat4 m, versor q, vec3 pivot) { + glm_quat_rotate_atm(m, q, pivot); +} + +CGLM_EXPORT +void +glmc_quat_make(const float * __restrict src, versor dest) { + glm_quat_make(src, dest); +} diff --git a/cglm/src/ray.c b/cglm/src/ray.c new file mode 100644 index 0000000..9592b83 --- /dev/null +++ b/cglm/src/ray.c @@ -0,0 +1,29 @@ +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +bool +glmc_ray_triangle(vec3 origin, + vec3 direction, + vec3 v0, + vec3 v1, + vec3 v2, + float *d) { + return glm_ray_triangle(origin, direction, v0, v1, v2, d); +} + +CGLM_EXPORT +bool +glmc_ray_sphere(vec3 origin, + vec3 dir, + vec4 s, + float * __restrict t1, + float * __restrict t2) { + return glm_ray_sphere(origin, dir, s, t1, t2); +} + +CGLM_EXPORT +void +glmc_ray_at(vec3 orig, vec3 dir, float t, vec3 point) { + glm_ray_at(orig, dir, t, point); +} diff --git a/cglm/src/sphere.c b/cglm/src/sphere.c new file mode 100644 index 0000000..003ef87 --- /dev/null +++ b/cglm/src/sphere.c @@ -0,0 +1,39 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +float +glmc_sphere_radii(vec4 s) { + return glm_sphere_radii(s); +} + +CGLM_EXPORT +void +glmc_sphere_transform(vec4 s, mat4 m, vec4 dest) { + glm_sphere_transform(s, m, dest); +} + +CGLM_EXPORT +void +glmc_sphere_merge(vec4 s1, vec4 s2, vec4 dest) { + glm_sphere_merge(s1, s2, dest); +} + +CGLM_EXPORT +bool +glmc_sphere_sphere(vec4 s1, vec4 s2) { + return glm_sphere_sphere(s1, s2); +} + +CGLM_EXPORT +bool +glmc_sphere_point(vec4 s, vec3 point) { + return glm_sphere_point(s, point); +} diff --git a/cglm/src/swift/empty.c b/cglm/src/swift/empty.c new file mode 100644 index 0000000..7c27d40 --- /dev/null +++ b/cglm/src/swift/empty.c @@ -0,0 +1 @@ +// This empty file is needed to trick swiftpm to build the header-only version of cglm as swiftpm itself does not support C targets that have no source code files \ No newline at end of file diff --git a/cglm/src/vec2.c b/cglm/src/vec2.c new file mode 100644 index 0000000..479c41e --- /dev/null +++ b/cglm/src/vec2.c @@ -0,0 +1,358 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_vec2(float * __restrict v, vec2 dest) { + glm_vec2(v, dest); +} + +CGLM_EXPORT +void +glmc_vec2_fill(vec2 v, float val) { + glm_vec2_fill(v, val); +} + +CGLM_EXPORT +bool +glmc_vec2_eq(vec2 v, float val) { + return glm_vec2_eq(v, val); +} + +CGLM_EXPORT +bool +glmc_vec2_eqv(vec2 a, vec2 b) { + return glm_vec2_eqv(a, b); +} + +CGLM_EXPORT +void +glmc_vec2_copy(vec2 a, vec2 dest) { + glm_vec2_copy(a, dest); +} + +CGLM_EXPORT +void +glmc_vec2_zero(vec2 v) { + glm_vec2_zero(v); +} + +CGLM_EXPORT +void +glmc_vec2_one(vec2 v) { + glm_vec2_one(v); +} + +CGLM_EXPORT +float +glmc_vec2_dot(vec2 a, vec2 b) { + return glm_vec2_dot(a, b); +} + +CGLM_EXPORT +float +glmc_vec2_cross(vec2 a, vec2 b) { + return glm_vec2_cross(a, b); +} + +CGLM_EXPORT +float +glmc_vec2_norm2(vec2 v) { + return glm_vec2_norm2(v); +} + +CGLM_EXPORT +float +glmc_vec2_norm(vec2 v) { + return glm_vec2_norm(v); +} + +CGLM_EXPORT +void +glmc_vec2_add(vec2 a, vec2 b, vec2 dest) { + glm_vec2_add(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_adds(vec2 v, float s, vec2 dest) { + glm_vec2_adds(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec2_sub(vec2 a, vec2 b, vec2 dest) { + glm_vec2_sub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_subs(vec2 v, float s, vec2 dest) { + glm_vec2_subs(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec2_mul(vec2 a, vec2 b, vec2 dest) { + glm_vec2_mul(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_scale(vec2 v, float s, vec2 dest) { + glm_vec2_scale(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec2_scale_as(vec2 v, float s, vec2 dest) { + glm_vec2_scale_as(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec2_div(vec2 a, vec2 b, vec2 dest) { + glm_vec2_div(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_divs(vec2 v, float s, vec2 dest) { + glm_vec2_divs(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec2_addadd(vec2 a, vec2 b, vec2 dest) { + glm_vec2_addadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_subadd(vec2 a, vec2 b, vec2 dest) { + glm_vec2_subadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_muladd(vec2 a, vec2 b, vec2 dest) { + glm_vec2_muladd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_muladds(vec2 a, float s, vec2 dest) { + glm_vec2_muladds(a, s, dest); +} + +CGLM_EXPORT +void +glmc_vec2_maxadd(vec2 a, vec2 b, vec2 dest) { + glm_vec2_maxadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_minadd(vec2 a, vec2 b, vec2 dest) { + glm_vec2_minadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_subsub(vec2 a, vec2 b, vec2 dest) { + glm_vec2_subsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_addsub(vec2 a, vec2 b, vec2 dest) { + glm_vec2_addsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_mulsub(vec2 a, vec2 b, vec2 dest) { + glm_vec2_mulsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_mulsubs(vec2 a, float s, vec2 dest) { + glm_vec2_mulsubs(a, s, dest); +} + +CGLM_EXPORT +void +glmc_vec2_maxsub(vec2 a, vec2 b, vec2 dest) { + glm_vec2_maxsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_minsub(vec2 a, vec2 b, vec2 dest) { + glm_vec2_minsub(a, b, dest); +} + + +CGLM_EXPORT +void +glmc_vec2_negate_to(vec2 v, vec2 dest) { + glm_vec2_negate_to(v, dest); +} + +CGLM_EXPORT +void +glmc_vec2_negate(vec2 v) { + glm_vec2_negate(v); +} + +CGLM_EXPORT +void +glmc_vec2_normalize(vec2 v) { + glm_vec2_normalize(v); +} + +CGLM_EXPORT +void +glmc_vec2_normalize_to(vec2 v, vec2 dest) { + glm_vec2_normalize_to(v, dest); +} + +CGLM_EXPORT +void +glmc_vec2_rotate(vec2 v, float angle, vec2 dest) { + glm_vec2_rotate(v, angle, dest); +} + +CGLM_EXPORT +void +glmc_vec2_center(vec2 a, vec2 b, vec2 dest) { + glm_vec2_center(a, b, dest); +} + +CGLM_EXPORT +float +glmc_vec2_distance2(vec2 a, vec2 b) { + return glm_vec2_distance2(a, b); +} + +CGLM_EXPORT +float +glmc_vec2_distance(vec2 a, vec2 b) { + return glm_vec2_distance(a, b); +} + +CGLM_EXPORT +void +glmc_vec2_maxv(vec2 a, vec2 b, vec2 dest) { + glm_vec2_maxv(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_minv(vec2 a, vec2 b, vec2 dest) { + glm_vec2_minv(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_clamp(vec2 v, float minval, float maxval) { + glm_vec2_clamp(v, minval, maxval); +} + +CGLM_EXPORT +void +glmc_vec2_abs(vec2 v, vec2 dest) { + glm_vec2_abs(v, dest); +} + +CGLM_EXPORT +void +glmc_vec2_fract(vec2 v, vec2 dest) { + glm_vec2_fract(v, dest); +} + +CGLM_EXPORT +void +glmc_vec2_floor(vec2 v, vec2 dest) { + glm_vec2_floor(v, dest); +} + +CGLM_EXPORT +void +glmc_vec2_mods(vec2 v, float s, vec2 dest) { + glm_vec2_mods(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec2_step(vec2 edge, vec2 v, vec2 dest) { + glm_vec2_step(edge, v, dest); +} + +CGLM_EXPORT +void +glmc_vec2_steps(float edge, vec2 v, vec2 dest) { + glm_vec2_steps(edge, v, dest); +} + +CGLM_EXPORT +void +glmc_vec2_stepr(vec2 edge, float v, vec2 dest) { + glm_vec2_stepr(edge, v, dest); +} + +CGLM_EXPORT +void +glmc_vec2_swizzle(vec2 v, int mask, vec2 dest) { + glm_vec2_swizzle(v, mask, dest); +} + +CGLM_EXPORT +void +glmc_vec2_lerp(vec2 from, vec2 to, float t, vec2 dest) { + glm_vec2_lerp(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_vec2_complex_mul(vec2 a, vec2 b, vec2 dest) { + glm_vec2_complex_mul(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_complex_div(vec2 a, vec2 b, vec2 dest) { + glm_vec2_complex_div(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec2_complex_conjugate(vec2 a, vec2 dest) { + glm_vec2_complex_conjugate(a, dest); +} + +CGLM_EXPORT +void +glmc_vec2_make(const float * __restrict src, vec2 dest) { + glm_vec2_make(src, dest); +} + +CGLM_EXPORT +void +glmc_vec2_reflect(vec2 v, vec2 n, vec2 dest) { + glm_vec2_reflect(v, n, dest); +} + +CGLM_EXPORT +bool +glmc_vec2_refract(vec2 v, vec2 n, float eta, vec2 dest) { + return glm_vec2_refract(v, n, eta, dest); +} diff --git a/cglm/src/vec3.c b/cglm/src/vec3.c new file mode 100644 index 0000000..a8c1083 --- /dev/null +++ b/cglm/src/vec3.c @@ -0,0 +1,503 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_vec3(vec4 v4, vec3 dest) { + glm_vec3(v4, dest); +} + +CGLM_EXPORT +void +glmc_vec3_copy(vec3 a, vec3 dest) { + glm_vec3_copy(a, dest); +} + +CGLM_EXPORT +void +glmc_vec3_zero(vec3 v) { + glm_vec3_zero(v); +} + +CGLM_EXPORT +void +glmc_vec3_one(vec3 v) { + glm_vec3_one(v); +} + +CGLM_EXPORT +float +glmc_vec3_dot(vec3 a, vec3 b) { + return glm_vec3_dot(a, b); +} + +CGLM_EXPORT +void +glmc_vec3_cross(vec3 a, vec3 b, vec3 dest) { + glm_vec3_cross(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_crossn(vec3 a, vec3 b, vec3 dest) { + glm_vec3_crossn(a, b, dest); +} + +CGLM_EXPORT +float +glmc_vec3_norm(vec3 v) { + return glm_vec3_norm(v); +} + +CGLM_EXPORT +void +glmc_vec3_normalize_to(vec3 v, vec3 dest) { + glm_vec3_normalize_to(v, dest); +} + +CGLM_EXPORT +void +glmc_vec3_normalize(vec3 v) { + glm_vec3_normalize(v); +} + +CGLM_EXPORT +float +glmc_vec3_norm2(vec3 v) { + return glm_vec3_norm2(v); +} + +CGLM_EXPORT +float +glmc_vec3_norm_one(vec3 v) { + return glm_vec3_norm_one(v); +} + +CGLM_EXPORT +float +glmc_vec3_norm_inf(vec3 v) { + return glm_vec3_norm_inf(v); +} + +CGLM_EXPORT +void +glmc_vec3_add(vec3 a, vec3 b, vec3 dest) { + glm_vec3_add(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_adds(vec3 v, float s, vec3 dest) { + glm_vec3_adds(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec3_sub(vec3 a, vec3 b, vec3 dest) { + glm_vec3_sub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_subs(vec3 v, float s, vec3 dest) { + glm_vec3_subs(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec3_mul(vec3 a, vec3 b, vec3 d) { + glm_vec3_mul(a, b, d); +} + +CGLM_EXPORT +void +glmc_vec3_scale(vec3 v, float s, vec3 dest) { + glm_vec3_scale(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec3_scale_as(vec3 v, float s, vec3 dest) { + glm_vec3_scale_as(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec3_div(vec3 a, vec3 b, vec3 dest) { + glm_vec3_div(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_divs(vec3 a, float s, vec3 dest) { + glm_vec3_divs(a, s, dest); +} + +CGLM_EXPORT +void +glmc_vec3_addadd(vec3 a, vec3 b, vec3 dest) { + glm_vec3_addadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_subadd(vec3 a, vec3 b, vec3 dest) { + glm_vec3_subadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_muladd(vec3 a, vec3 b, vec3 dest) { + glm_vec3_muladd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_muladds(vec3 a, float s, vec3 dest) { + glm_vec3_muladds(a, s, dest); +} + +CGLM_EXPORT +void +glmc_vec3_maxadd(vec3 a, vec3 b, vec3 dest) { + glm_vec3_maxadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_minadd(vec3 a, vec3 b, vec3 dest) { + glm_vec3_minadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_subsub(vec3 a, vec3 b, vec3 dest) { + glm_vec3_subsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_addsub(vec3 a, vec3 b, vec3 dest) { + glm_vec3_addsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_mulsub(vec3 a, vec3 b, vec3 dest) { + glm_vec3_mulsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_mulsubs(vec3 a, float s, vec3 dest) { + glm_vec3_mulsubs(a, s, dest); +} + +CGLM_EXPORT +void +glmc_vec3_maxsub(vec3 a, vec3 b, vec3 dest) { + glm_vec3_maxsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_minsub(vec3 a, vec3 b, vec3 dest) { + glm_vec3_minsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_negate(vec3 v) { + glm_vec3_negate(v); +} + +CGLM_EXPORT +void +glmc_vec3_negate_to(vec3 v, vec3 dest) { + glm_vec3_negate_to(v, dest); +} + +CGLM_EXPORT +float +glmc_vec3_angle(vec3 a, vec3 b) { + return glm_vec3_angle(a, b); +} + +CGLM_EXPORT +void +glmc_vec3_rotate(vec3 v, float angle, vec3 axis) { + glm_vec3_rotate(v, angle, axis); +} + +CGLM_EXPORT +void +glmc_vec3_rotate_m4(mat4 m, vec3 v, vec3 dest) { + glm_vec3_rotate_m4(m, v, dest); +} + +CGLM_EXPORT +void +glmc_vec3_rotate_m3(mat3 m, vec3 v, vec3 dest) { + glm_vec3_rotate_m3(m, v, dest); +} + +CGLM_EXPORT +void +glmc_vec3_proj(vec3 a, vec3 b, vec3 dest) { + glm_vec3_proj(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_center(vec3 a, vec3 b, vec3 dest) { + glm_vec3_center(a, b, dest); +} + +CGLM_EXPORT +float +glmc_vec3_distance(vec3 a, vec3 b) { + return glm_vec3_distance(a, b); +} + +CGLM_EXPORT +float +glmc_vec3_distance2(vec3 a, vec3 b) { + return glm_vec3_distance2(a, b); +} + +CGLM_EXPORT +void +glmc_vec3_maxv(vec3 a, vec3 b, vec3 dest) { + glm_vec3_maxv(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_minv(vec3 a, vec3 b, vec3 dest) { + glm_vec3_minv(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec3_clamp(vec3 v, float minVal, float maxVal) { + glm_vec3_clamp(v, minVal, maxVal); +} + +CGLM_EXPORT +void +glmc_vec3_ortho(vec3 v, vec3 dest) { + glm_vec3_ortho(v, dest); +} + +CGLM_EXPORT +void +glmc_vec3_lerp(vec3 from, vec3 to, float t, vec3 dest) { + glm_vec3_lerp(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_vec3_lerpc(vec3 from, vec3 to, float t, vec3 dest) { + glm_vec3_lerpc(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_vec3_step(vec3 edge, vec3 x, vec3 dest) { + glm_vec3_step(edge, x, dest); +} + +CGLM_EXPORT +void +glmc_vec3_smoothstep_uni(float edge0, float edge1, vec3 x, vec3 dest) { + glm_vec3_smoothstep_uni(edge0, edge1, x, dest); +} + +CGLM_EXPORT +void +glmc_vec3_smoothstep(vec3 edge0, vec3 edge1, vec3 x, vec3 dest) { + glm_vec3_smoothstep(edge0, edge1, x, dest); +} + +CGLM_EXPORT +void +glmc_vec3_smoothinterp(vec3 from, vec3 to, float t, vec3 dest) { + glm_vec3_smoothinterp(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_vec3_smoothinterpc(vec3 from, vec3 to, float t, vec3 dest) { + glm_vec3_smoothinterpc(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_vec3_swizzle(vec3 v, int mask, vec3 dest) { + glm_vec3_swizzle(v, mask, dest); +} + +/* ext */ + +CGLM_EXPORT +void +glmc_vec3_mulv(vec3 a, vec3 b, vec3 d) { + glm_vec3_mulv(a, b, d); +} + +CGLM_EXPORT +void +glmc_vec3_broadcast(float val, vec3 d) { + glm_vec3_broadcast(val, d); +} + +CGLM_EXPORT +void +glmc_vec3_fill(vec3 v, float val) { + glm_vec3_fill(v, val); +} + +CGLM_EXPORT +bool +glmc_vec3_eq(vec3 v, float val) { + return glm_vec3_eq(v, val); +} + +CGLM_EXPORT +bool +glmc_vec3_eq_eps(vec3 v, float val) { + return glm_vec3_eq_eps(v, val); +} + +CGLM_EXPORT +bool +glmc_vec3_eq_all(vec3 v) { + return glm_vec3_eq_all(v); +} + +CGLM_EXPORT +bool +glmc_vec3_eqv(vec3 a, vec3 b) { + return glm_vec3_eqv(a, b); +} + +CGLM_EXPORT +bool +glmc_vec3_eqv_eps(vec3 a, vec3 b) { + return glm_vec3_eqv_eps(a, b); +} + +CGLM_EXPORT +float +glmc_vec3_max(vec3 v) { + return glm_vec3_max(v); +} + +CGLM_EXPORT +float +glmc_vec3_min(vec3 v) { + return glm_vec3_min(v); +} + +CGLM_EXPORT +bool +glmc_vec3_isnan(vec3 v) { + return glm_vec3_isnan(v); +} + +CGLM_EXPORT +bool +glmc_vec3_isinf(vec3 v) { + return glm_vec3_isinf(v); +} + +CGLM_EXPORT +bool +glmc_vec3_isvalid(vec3 v) { + return glm_vec3_isvalid(v); +} + +CGLM_EXPORT +void +glmc_vec3_sign(vec3 v, vec3 dest) { + glm_vec3_sign(v, dest); +} + +CGLM_EXPORT +void +glmc_vec3_abs(vec3 v, vec3 dest) { + glm_vec3_abs(v, dest); +} + +CGLM_EXPORT +void +glmc_vec3_fract(vec3 v, vec3 dest) { + glm_vec3_fract(v, dest); +} + +CGLM_EXPORT +void +glmc_vec3_floor(vec3 v, vec3 dest) { + glm_vec3_floor(v, dest); +} + +CGLM_EXPORT +void +glmc_vec3_mods(vec3 v, float s, vec3 dest) { + glm_vec3_mods(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec3_steps(float edge, vec3 v, vec3 dest) { + glm_vec3_steps(edge, v, dest); +} + +CGLM_EXPORT +void +glmc_vec3_stepr(vec3 edge, float v, vec3 dest) { + glm_vec3_stepr(edge, v, dest); +} + +CGLM_EXPORT +float +glmc_vec3_hadd(vec3 v) { + return glm_vec3_hadd(v); +} + +CGLM_EXPORT +void +glmc_vec3_sqrt(vec3 v, vec3 dest) { + glm_vec3_sqrt(v, dest); +} + +CGLM_EXPORT +void +glmc_vec3_make(const float * __restrict src, vec3 dest) { + glm_vec3_make(src, dest); +} + +CGLM_EXPORT +void +glmc_vec3_faceforward(vec3 n, vec3 v, vec3 nref, vec3 dest) { + glm_vec3_faceforward(n, v, nref, dest); +} + +CGLM_EXPORT +void +glmc_vec3_reflect(vec3 v, vec3 n, vec3 dest) { + glm_vec3_reflect(v, n, dest); +} + +CGLM_EXPORT +bool +glmc_vec3_refract(vec3 v, vec3 n, float eta, vec3 dest) { + return glm_vec3_refract(v, n, eta, dest); +} diff --git a/cglm/src/vec4.c b/cglm/src/vec4.c new file mode 100644 index 0000000..4b08abf --- /dev/null +++ b/cglm/src/vec4.c @@ -0,0 +1,461 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "../include/cglm/cglm.h" +#include "../include/cglm/call.h" + +CGLM_EXPORT +void +glmc_vec4(vec3 v3, float last, vec4 dest) { + glm_vec4(v3, last, dest); +} + +CGLM_EXPORT +void +glmc_vec4_zero(vec4 v) { + glm_vec4_zero(v); +} + +CGLM_EXPORT +void +glmc_vec4_one(vec4 v) { + glm_vec4_one(v); +} + +CGLM_EXPORT +void +glmc_vec4_copy3(vec4 v, vec3 dest) { + glm_vec4_copy3(v, dest); +} + +CGLM_EXPORT +void +glmc_vec4_copy(vec4 v, vec4 dest) { + glm_vec4_copy(v, dest); +} + +CGLM_EXPORT +void +glmc_vec4_ucopy(vec4 v, vec4 dest) { + glm_vec4_ucopy(v, dest); +} + +CGLM_EXPORT +float +glmc_vec4_dot(vec4 a, vec4 b) { + return glm_vec4_dot(a, b); +} + +CGLM_EXPORT +float +glmc_vec4_norm(vec4 v) { + return glm_vec4_norm(v); +} + +CGLM_EXPORT +void +glmc_vec4_normalize_to(vec4 v, vec4 dest) { + glm_vec4_normalize_to(v, dest); +} + +CGLM_EXPORT +void +glmc_vec4_normalize(vec4 v) { + glm_vec4_normalize(v); +} + +CGLM_EXPORT +float +glmc_vec4_norm2(vec4 v) { + return glm_vec4_norm2(v); +} + +CGLM_EXPORT +float +glmc_vec4_norm_one(vec4 v) { + return glm_vec4_norm_one(v); +} + +CGLM_EXPORT +float +glmc_vec4_norm_inf(vec4 v) { + return glm_vec4_norm_inf(v); +} + +CGLM_EXPORT +void +glmc_vec4_add(vec4 a, vec4 b, vec4 dest) { + glm_vec4_add(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec4_adds(vec4 v, float s, vec4 dest) { + glm_vec4_adds(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec4_sub(vec4 a, vec4 b, vec4 dest) { + glm_vec4_sub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec4_subs(vec4 v, float s, vec4 dest) { + glm_vec4_subs(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec4_mul(vec4 a, vec4 b, vec4 d) { + glm_vec4_mul(a, b, d); +} + +CGLM_EXPORT +void +glmc_vec4_scale(vec4 v, float s, vec4 dest) { + glm_vec4_scale(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec4_scale_as(vec4 v, float s, vec4 dest) { + glm_vec4_scale_as(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec4_div(vec4 a, vec4 b, vec4 dest) { + glm_vec4_div(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec4_divs(vec4 v, float s, vec4 dest) { + glm_vec4_divs(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec4_addadd(vec4 a, vec4 b, vec4 dest) { + glm_vec4_addadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec4_subadd(vec4 a, vec4 b, vec4 dest) { + glm_vec4_subadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec4_muladd(vec4 a, vec4 b, vec4 dest) { + glm_vec4_muladd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec4_muladds(vec4 a, float s, vec4 dest) { + glm_vec4_muladds(a, s, dest); +} + +CGLM_EXPORT +void +glmc_vec4_maxadd(vec4 a, vec4 b, vec4 dest) { + glm_vec4_maxadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec4_minadd(vec4 a, vec4 b, vec4 dest) { + glm_vec4_minadd(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec4_subsub(vec4 a, vec4 b, vec4 dest) { + glm_vec4_subsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec4_addsub(vec4 a, vec4 b, vec4 dest) { + glm_vec4_addsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec4_mulsub(vec4 a, vec4 b, vec4 dest) { + glm_vec4_mulsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec4_mulsubs(vec4 a, float s, vec4 dest) { + glm_vec4_mulsubs(a, s, dest); +} + +CGLM_EXPORT +void +glmc_vec4_maxsub(vec4 a, vec4 b, vec4 dest) { + glm_vec4_maxsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec4_minsub(vec4 a, vec4 b, vec4 dest) { + glm_vec4_minsub(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec4_negate(vec4 v) { + glm_vec4_negate(v); +} + +CGLM_EXPORT +void +glmc_vec4_negate_to(vec4 v, vec4 dest) { + glm_vec4_negate_to(v, dest); +} + +CGLM_EXPORT +float +glmc_vec4_distance(vec4 a, vec4 b) { + return glm_vec4_distance(a, b); +} + +CGLM_EXPORT +float +glmc_vec4_distance2(vec4 a, vec4 b) { + return glm_vec4_distance2(a, b); +} + +CGLM_EXPORT +void +glmc_vec4_maxv(vec4 a, vec4 b, vec4 dest) { + glm_vec4_maxv(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec4_minv(vec4 a, vec4 b, vec4 dest) { + glm_vec4_minv(a, b, dest); +} + +CGLM_EXPORT +void +glmc_vec4_clamp(vec4 v, float minVal, float maxVal) { + glm_vec4_clamp(v, minVal, maxVal); +} + +CGLM_EXPORT +void +glmc_vec4_lerp(vec4 from, vec4 to, float t, vec4 dest) { + glm_vec4_lerp(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_vec4_lerpc(vec4 from, vec4 to, float t, vec4 dest) { + glm_vec4_lerpc(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_vec4_step(vec4 edge, vec4 x, vec4 dest) { + glm_vec4_step(edge, x, dest); +} + +CGLM_EXPORT +void +glmc_vec4_smoothstep_uni(float edge0, float edge1, vec4 x, vec4 dest) { + glm_vec4_smoothstep_uni(edge0, edge1, x, dest); +} + +CGLM_EXPORT +void +glmc_vec4_smoothstep(vec4 edge0, vec4 edge1, vec4 x, vec4 dest) { + glm_vec4_smoothstep(edge0, edge1, x, dest); +} + +CGLM_EXPORT +void +glmc_vec4_smoothinterp(vec4 from, vec4 to, float t, vec4 dest) { + glm_vec4_smoothinterp(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_vec4_smoothinterpc(vec4 from, vec4 to, float t, vec4 dest) { + glm_vec4_smoothinterpc(from, to, t, dest); +} + +CGLM_EXPORT +void +glmc_vec4_cubic(float s, vec4 dest) { + glm_vec4_cubic(s, dest); +} + +CGLM_EXPORT +void +glmc_vec4_swizzle(vec4 v, int mask, vec4 dest) { + glm_vec4_swizzle(v, mask, dest); +} + +/* ext */ + +CGLM_EXPORT +void +glmc_vec4_mulv(vec4 a, vec4 b, vec4 d) { + glm_vec4_mulv(a, b, d); +} + +CGLM_EXPORT +void +glmc_vec4_broadcast(float val, vec4 d) { + glm_vec4_broadcast(val, d); +} + +CGLM_EXPORT +void +glmc_vec4_fill(vec4 v, float val) { + glm_vec4_fill(v, val); +} + +CGLM_EXPORT +bool +glmc_vec4_eq(vec4 v, float val) { + return glm_vec4_eq(v, val); +} + +CGLM_EXPORT +bool +glmc_vec4_eq_eps(vec4 v, float val) { + return glm_vec4_eq_eps(v, val); +} + +CGLM_EXPORT +bool +glmc_vec4_eq_all(vec4 v) { + return glm_vec4_eq_all(v); +} + +CGLM_EXPORT +bool +glmc_vec4_eqv(vec4 a, vec4 b) { + return glm_vec4_eqv(a, b); +} + +CGLM_EXPORT +bool +glmc_vec4_eqv_eps(vec4 a, vec4 b) { + return glm_vec4_eqv_eps(a, b); +} + +CGLM_EXPORT +float +glmc_vec4_max(vec4 v) { + return glm_vec4_max(v); +} + +CGLM_EXPORT +float +glmc_vec4_min(vec4 v) { + return glm_vec4_min(v); +} + +CGLM_EXPORT +bool +glmc_vec4_isnan(vec4 v) { + return glm_vec4_isnan(v); +} + +CGLM_EXPORT +bool +glmc_vec4_isinf(vec4 v) { + return glm_vec4_isinf(v); +} + +CGLM_EXPORT +bool +glmc_vec4_isvalid(vec4 v) { + return glm_vec4_isvalid(v); +} + +CGLM_EXPORT +void +glmc_vec4_sign(vec4 v, vec4 dest) { + glm_vec4_sign(v, dest); +} + +CGLM_EXPORT +void +glmc_vec4_abs(vec4 v, vec4 dest) { + glm_vec4_abs(v, dest); +} + +CGLM_EXPORT +void +glmc_vec4_fract(vec4 v, vec4 dest) { + glm_vec4_fract(v, dest); +} + +CGLM_EXPORT +void +glmc_vec4_floor(vec4 v, vec4 dest) { + glm_vec4_floor(v, dest); +} + +CGLM_EXPORT +void +glmc_vec4_mods(vec4 v, float s, vec4 dest) { + glm_vec4_mods(v, s, dest); +} + +CGLM_EXPORT +void +glmc_vec4_steps(float edge, vec4 v, vec4 dest) { + glm_vec4_steps(edge, v, dest); +} + +CGLM_EXPORT +void +glmc_vec4_stepr(vec4 edge, float v, vec4 dest) { + glm_vec4_stepr(edge, v, dest); +} + +CGLM_EXPORT +float +glmc_vec4_hadd(vec4 v) { + return glm_vec4_hadd(v); +} + +CGLM_EXPORT +void +glmc_vec4_sqrt(vec4 v, vec4 dest) { + glm_vec4_sqrt(v, dest); +} + +CGLM_EXPORT +void +glmc_vec4_make(const float * __restrict src, vec4 dest) { + glm_vec4_make(src, dest); +} + +CGLM_EXPORT +void +glmc_vec4_reflect(vec4 v, vec4 n, vec4 dest) { + glm_vec4_reflect(v, n, dest); +} + +CGLM_EXPORT +bool +glmc_vec4_refract(vec4 v, vec4 n, float eta, vec4 dest) { + return glm_vec4_refract(v, n, eta, dest); +} diff --git a/cglm/test/CMakeLists.txt b/cglm/test/CMakeLists.txt new file mode 100644 index 0000000..b90437a --- /dev/null +++ b/cglm/test/CMakeLists.txt @@ -0,0 +1,48 @@ +cmake_minimum_required(VERSION 3.8.2) + +# List all files containing tests. (Change as needed) +set(TESTFILES + runner.c + src/test_euler.c + src/test_bezier.c + src/test_struct.c + src/test_clamp.c + src/test_common.c + src/tests.c + ) + +set(TEST_MAIN tests) +set(TEST_RUNNER_PARAMS "") + +add_executable(${TEST_MAIN} ${TESTFILES}) +target_compile_definitions(${TEST_MAIN} PRIVATE CGLM_DEFINE_PRINTS=1) + +if(CMAKE_SYSTEM_NAME STREQUAL WASI) + target_compile_definitions(${TEST_MAIN} PRIVATE _WASI_EMULATED_PROCESS_CLOCKS=1) + target_link_options(${TEST_MAIN} PRIVATE "-lwasi-emulated-process-clocks") +endif() + +if(NOT MSVC) + target_link_libraries(${TEST_MAIN} PRIVATE m) +endif() + +target_link_libraries(${TEST_MAIN} PRIVATE cglm) +target_include_directories(${TEST_MAIN} PRIVATE + ${CMAKE_CURRENT_LIST_DIR}/include + ${CMAKE_CURRENT_LIST_DIR}/src + ) + +set_target_properties(${TEST_MAIN} PROPERTIES RUNTIME_OUTPUT_DIRECTORY ${PROJECT_BINARY_DIR}) + +if(LDFLAGS) + target_compile_options(${TEST_MAIN} PRIVATE ${LDFLAGS}) +endif() + +add_test( + NAME cglm.${TEST_MAIN} + COMMAND ${TEST_MAIN} ${TEST_RUNNER_PARAMS}) + +add_custom_target(check + make + COMMAND ${CMAKE_CTEST_COMMAND} -V + DEPENDS cglm) diff --git a/cglm/test/include/common.h b/cglm/test/include/common.h new file mode 100644 index 0000000..1b03f34 --- /dev/null +++ b/cglm/test/include/common.h @@ -0,0 +1,159 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef tests_common_h +#define tests_common_h + +#ifndef _USE_MATH_DEFINES +# define _USE_MATH_DEFINES /* for windows */ +#endif + +#ifndef _CRT_SECURE_NO_WARNINGS +# define _CRT_SECURE_NO_WARNINGS /* for windows */ +#endif + +#ifndef _GNU_SOURCE +# define _GNU_SOURCE /* for drand48() */ +#endif + +#ifndef CGLM_CLIPSPACE_INCLUDE_ALL +# define CGLM_CLIPSPACE_INCLUDE_ALL +#endif + +#include +#include +#include + +#include +#include +#include + +typedef struct test_status_t { + const char *msg; + int status; +} test_status_t; + +typedef test_status_t (*fntest)(void); + +typedef struct test_entry_t { + char *name; + fntest entry; + int ret; + int show_output; +} test_entry_t; + +#ifndef GLM_TESTS_NO_COLORFUL_OUTPUT + +#define RESET "\033[0m" +#define BLACK "\033[30m" /* Black */ +#define RED "\033[31m" /* Red */ +#define GREEN "\033[32m" /* Green */ +#define YELLOW "\033[33m" /* Yellow */ +#define BLUE "\033[34m" /* Blue */ +#define MAGENTA "\033[35m" /* Magenta */ +#define CYAN "\033[36m" /* Cyan */ +#define WHITE "\033[37m" /* White */ +#define BOLDBLACK "\033[1m\033[30m" /* Bold Black */ +#define BOLDRED "\033[1m\033[31m" /* Bold Red */ +#define BOLDGREEN "\033[1m\033[32m" /* Bold Green */ +#define BOLDYELLOW "\033[1m\033[33m" /* Bold Yellow */ +#define BOLDBLUE "\033[1m\033[34m" /* Bold Blue */ +#define BOLDMAGENTA "\033[1m\033[35m" /* Bold Magenta */ +#define BOLDCYAN "\033[1m\033[36m" /* Bold Cyan */ +#define BOLDWHITE "\033[1m\033[37m" /* Bold White */ + +#else + +#define RESET +#define BLACK +#define RED +#define GREEN +#define YELLOW +#define BLUE +#define MAGENTA +#define CYAN +#define WHITE +#define BOLDBLACK +#define BOLDRED +#define BOLDGREEN +#define BOLDYELLOW +#define BOLDBLUE +#define BOLDMAGENTA +#define BOLDCYAN +#define BOLDWHITE + +#endif + +#define TEST_DECLARE(FUN) test_status_t test_ ## FUN(void); +#define TEST_ENTRY(FUN) { #FUN, test_ ## FUN, 0, 0 }, +#define TEST_LIST static test_entry_t tests[] = + +/* __VA_ARGS__ workaround for MSVC: https://stackoverflow.com/a/5134656 */ +#define EXPAND(x) x + +#define TEST_OK 1 +#define TEST_SUCCESS return (test_status_t){NULL, TEST_OK}; + +#define TEST_IMPL_ARG1(FUN) \ + test_status_t test_ ## FUN (void); \ + test_status_t test_ ## FUN() + +#define TEST_IMPL_ARG2(PREFIX, FUN) TEST_IMPL_ARG1(PREFIX ## FUN) +#define TEST_IMPL_ARG3(arg1, arg2, arg3, ...) arg3 + +#define TEST_IMPL_CHOOSER(...) \ + EXPAND(TEST_IMPL_ARG3(__VA_ARGS__, TEST_IMPL_ARG2, TEST_IMPL_ARG1,)) + +#define TEST_IMPL(...) EXPAND(TEST_IMPL_CHOOSER(__VA_ARGS__)(__VA_ARGS__)) + +#define ASSERT_EXT(expr, msg) \ + if (!(expr)) { \ + fprintf(stderr, \ + RED " assert fail" RESET \ + " in " BOLDCYAN "%s " RESET \ + "on " BOLDMAGENTA "line %d" RESET \ + " : " BOLDWHITE " ASSERT(%s)\n" RESET, \ + __FILE__, \ + __LINE__, \ + #expr); \ + return (test_status_t){msg, 0}; \ + } + +#define ASSERT_ARG1(expr) ASSERT_EXT(expr, NULL) +#define ASSERT_ARG2(expr, msg) ASSERT_EXT(expr, msg) +#define ASSERT_ARG3(arg1, arg2, arg3, ...) arg3 + +#define ASSERT_CHOOSER(...) ASSERT_ARG3(__VA_ARGS__, ASSERT_ARG2, ASSERT_ARG1,) +#define ASSERT(...) do { ASSERT_CHOOSER(__VA_ARGS__)(__VA_ARGS__) } while(0); +#define ASSERTIFY(expr) do { \ + test_status_t ts; \ + ts = expr; \ + if (ts.status != TEST_OK) { \ + fprintf(stderr, \ + RED " assert fail" RESET \ + " in " BOLDCYAN "%s " RESET \ + "on " BOLDMAGENTA "line %d" RESET \ + " : " BOLDWHITE " ASSERTIFY(%s)\n" RESET, \ + __FILE__, \ + __LINE__, \ + #expr); \ + return (test_status_t){ts.msg, 0}; \ + } \ + } while(0); + +#if defined(_WIN32) || defined(__MINGW32__) || defined(__MINGW64__) +# define drand48() ((float)(rand() / (RAND_MAX + 1.0))) +# define OK_TEXT "ok:" +# define FAIL_TEXT "fail:" +# define FINAL_TEXT "^_^" +#else +# define OK_TEXT "✔︎" +# define FAIL_TEXT "𐄂" +# define FINAL_TEXT "🎉" +#endif + +#endif /* common_h */ diff --git a/cglm/test/runner.c b/cglm/test/runner.c new file mode 100644 index 0000000..888831c --- /dev/null +++ b/cglm/test/runner.c @@ -0,0 +1,100 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "include/common.h" +#include "tests.h" + +#include +#include +#include + +int +main(int argc, const char * argv[]) { + test_entry_t *entry; + test_status_t st; + int32_t i, count, passed, failed, maxlen; + double start, end, elapsed, total; + (void)argc; + (void)argv; + + passed = failed = maxlen = 0; + total = 0.0; + count = sizeof(tests) / sizeof(tests[0]); + + fprintf(stderr, CYAN "\nWelcome to cglm tests ( " RESET); + glm_arch_print(stderr); + fprintf(stderr, CYAN " )\n\n" RESET); + + srand((unsigned int)time(NULL)); + + for (i = 0; i < count; i++) { + int32_t len; + + entry = tests + i; + len = (int32_t)strlen(entry->name); + + maxlen = GLM_MAX(maxlen, len); + } + + maxlen += 5; + + fprintf(stderr, + BOLDWHITE " %-*s %-*s\n", + maxlen, "Test Name", maxlen, "Elapsed Time"); + + for (i = 0; i < count; i++) { + entry = tests + i; + start = clock(); + st = entry->entry(); + end = clock(); + elapsed = (end - start) / CLOCKS_PER_SEC; + total += elapsed; + + if (!st.status) { + fprintf(stderr, + BOLDRED " " FAIL_TEXT BOLDWHITE " %s " RESET, entry->name); + if (st.msg) { + fprintf(stderr, + YELLOW "- %s" RESET, + st.msg); + } + + fprintf(stderr, "\n"); + + failed++; + } else { + fprintf(stderr, GREEN " " OK_TEXT RESET " %-*s ", maxlen, entry->name); + + if (elapsed > 0.01) + fprintf(stderr, YELLOW "%.2fs", elapsed); + else + fprintf(stderr, "0"); + + fprintf(stderr, "\n" RESET); + passed++; + } + } + + if (failed == 0) { + fprintf(stderr, + BOLDGREEN "\n All tests passed " FINAL_TEXT "\n" RESET); + } + + fprintf(stderr, + CYAN "\ncglm test results (%0.2fs):\n" RESET + "--------------------------\n" + + MAGENTA "%d" RESET " tests ran, " + GREEN "%d" RESET " passed, " + RED "%d" RESET " failed\n\n" RESET, + total, + count, + passed, + failed); + + return failed; +} diff --git a/cglm/test/src/test_aabb2d.h b/cglm/test/src/test_aabb2d.h new file mode 100644 index 0000000..d2ce9fc --- /dev/null +++ b/cglm/test/src/test_aabb2d.h @@ -0,0 +1,28 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +#ifndef CGLM_TEST_AABB2D_ONCE +#define CGLM_TEST_AABB2D_ONCE + +/* Macros */ +/* Deprecated */ + +#endif /* CGLM_TEST_VEC4_ONCE */ + +/* --- */ +TEST_IMPL(GLM_PREFIX, aabb2d_sizev) { + vec2 a[2] = {{10.0f, 10.0f}, {20.0f, 20.0f}}; + vec2 size = {0}; + + GLM(aabb2d_sizev)(a, size); + + ASSERTIFY(test_assert_vec2_eq(size, (vec2){10.0f, 10.0f})) + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_affine.h b/cglm/test/src/test_affine.h new file mode 100644 index 0000000..2199a43 --- /dev/null +++ b/cglm/test/src/test_affine.h @@ -0,0 +1,634 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +TEST_IMPL(GLM_PREFIX, translate) { + mat4 m1; + vec4 v1 = {1.0f, 2.0f, 3.0f, 1.0f}, v2; + + glm_mat4_identity(m1); + GLM(translate)(m1, (vec3){13.0f, 11.0f, 7.0f}); + glm_mat4_mulv(m1, v1, v2); + + ASSERT(test_eq(v2[0], 14.0f)) + ASSERT(test_eq(v2[1], 13.0f)) + ASSERT(test_eq(v2[2], 10.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + glm_mat4_identity(m1); + GLM(translate)(m1, (vec3){1.0f, -1.0f, -5.0f}); + glm_mat4_mulv(m1, v2, v2); + + ASSERT(test_eq(v2[0], 15.0f)) + ASSERT(test_eq(v2[1], 12.0f)) + ASSERT(test_eq(v2[2], 5.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, translate_to) { + mat4 m1, m2; + vec4 v1 = {1.0f, 2.0f, 3.0f, 1.0f}, v2; + + glm_mat4_identity(m1); + GLM(translate_to)(m1, (vec3){13.0f, 11.0f, 7.0f}, m2); + glm_mat4_mulv(m2, v1, v2); + + ASSERT(test_eq(v2[0], 14.0f)) + ASSERT(test_eq(v2[1], 13.0f)) + ASSERT(test_eq(v2[2], 10.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + glm_mat4_identity(m1); + GLM(translate_to)(m1, (vec3){1.0f, -1.0f, -5.0f}, m2); + glm_mat4_mulv(m2, v2, v2); + + ASSERT(test_eq(v2[0], 15.0f)) + ASSERT(test_eq(v2[1], 12.0f)) + ASSERT(test_eq(v2[2], 5.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, translate_x) { + mat4 m1; + vec4 v1 = {1.0f, 2.0f, 3.0f, 1.0f}, v2; + + glm_mat4_identity(m1); + GLM(translate_x)(m1, 13.0f); + glm_mat4_mulv(m1, v1, v2); + + ASSERT(test_eq(v2[0], 14.0f)) + ASSERT(test_eq(v2[1], 2.0f)) + ASSERT(test_eq(v2[2], 3.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + glm_mat4_identity(m1); + GLM(translate_x)(m1, -1.0f); + glm_mat4_mulv(m1, v2, v2); + + ASSERT(test_eq(v2[0], 13.0f)) + ASSERT(test_eq(v2[1], 2.0f)) + ASSERT(test_eq(v2[2], 3.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, translate_y) { + mat4 m1; + vec4 v1 = {1.0f, 2.0f, 3.0f, 1.0f}, v2; + + glm_mat4_identity(m1); + GLM(translate_y)(m1, 11.0f); + glm_mat4_mulv(m1, v1, v2); + + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 13.0f)) + ASSERT(test_eq(v2[2], 3.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + glm_mat4_identity(m1); + GLM(translate_y)(m1, -1.0f); + glm_mat4_mulv(m1, v2, v2); + + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 12.0f)) + ASSERT(test_eq(v2[2], 3.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, translate_z) { + mat4 m1; + vec4 v1 = {1.0f, 2.0f, 3.0f, 1.0f}, v2; + + glm_mat4_identity(m1); + GLM(translate_z)(m1, 7.0f); + glm_mat4_mulv(m1, v1, v2); + + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 2.0f)) + ASSERT(test_eq(v2[2], 10.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + glm_mat4_identity(m1); + GLM(translate_z)(m1, -5.0f); + glm_mat4_mulv(m1, v2, v2); + + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 2.0f)) + ASSERT(test_eq(v2[2], 5.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, translate_make) { + mat4 m1; + vec4 v1 = {1.0f, 2.0f, 3.0f, 1.0f}, v2; + + glm_mat4_identity(m1); + GLM(translate_make)(m1, (vec3){13.0f, 11.0f, 7.0f}); + glm_mat4_mulv(m1, v1, v2); + + ASSERT(test_eq(v2[0], 14.0f)) + ASSERT(test_eq(v2[1], 13.0f)) + ASSERT(test_eq(v2[2], 10.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + glm_mat4_identity(m1); + GLM(translate_make)(m1, (vec3){1.0f, -1.0f, -5.0f}); + glm_mat4_mulv(m1, v2, v2); + + ASSERT(test_eq(v2[0], 15.0f)) + ASSERT(test_eq(v2[1], 12.0f)) + ASSERT(test_eq(v2[2], 5.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, scale_to) { + mat4 m1, m2; + vec4 v1 = {1.0f, 2.0f, 3.0f, 1.0f}, v2; + + glm_mat4_identity(m1); + GLM(scale_to)(m1, (vec3){13.0f, 11.0f, 7.0f}, m2); + glm_mat4_mulv(m2, v1, v2); + + ASSERT(test_eq(v2[0], 13.0f)) + ASSERT(test_eq(v2[1], 22.0f)) + ASSERT(test_eq(v2[2], 21.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + glm_mat4_identity(m1); + GLM(scale_to)(m1, (vec3){1.0f, -1.0f, -5.0f}, m2); + glm_mat4_mulv(m2, v2, v2); + + ASSERT(test_eq(v2[0], 13.0f)) + ASSERT(test_eq(v2[1], -22.0f)) + ASSERT(test_eq(v2[2], -105.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, scale_make) { + mat4 m1; + vec4 v1 = {1.0f, 2.0f, 3.0f, 1.0f}, v2; + + GLM(scale_make)(m1, (vec3){13.0f, 11.0f, 7.0f}); + glm_mat4_mulv(m1, v1, v2); + + ASSERT(test_eq(v2[0], 13.0f)) + ASSERT(test_eq(v2[1], 22.0f)) + ASSERT(test_eq(v2[2], 21.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + GLM(scale_make)(m1, (vec3){1.0f, -1.0f, -5.0f}); + glm_mat4_mulv(m1, v2, v2); + + ASSERT(test_eq(v2[0], 13.0f)) + ASSERT(test_eq(v2[1], -22.0f)) + ASSERT(test_eq(v2[2], -105.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, scale) { + mat4 m1; + vec4 v1 = {1.0f, 2.0f, 3.0f, 1.0f}, v2; + + glm_mat4_identity(m1); + GLM(scale)(m1, (vec3){13.0f, 11.0f, 7.0f}); + glm_mat4_mulv(m1, v1, v2); + + ASSERT(test_eq(v2[0], 13.0f)) + ASSERT(test_eq(v2[1], 22.0f)) + ASSERT(test_eq(v2[2], 21.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + glm_mat4_identity(m1); + GLM(scale)(m1, (vec3){1.0f, -1.0f, -5.0f}); + glm_mat4_mulv(m1, v2, v2); + + ASSERT(test_eq(v2[0], 13.0f)) + ASSERT(test_eq(v2[1], -22.0f)) + ASSERT(test_eq(v2[2], -105.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, scale_uni) { + mat4 m1; + vec4 v1 = {1.0f, 2.0f, 3.0f, 1.0f}, v2; + + glm_mat4_identity(m1); + GLM(scale_uni)(m1, 13.0f); + glm_mat4_mulv(m1, v1, v2); + + ASSERT(test_eq(v2[0], 13.0f)) + ASSERT(test_eq(v2[1], 26.0f)) + ASSERT(test_eq(v2[2], 39.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + glm_mat4_identity(m1); + GLM(scale_uni)(m1, -5.0f); + glm_mat4_mulv(m1, v2, v2); + + ASSERT(test_eq(v2[0], -65.0f)) + ASSERT(test_eq(v2[1], -130.0f)) + ASSERT(test_eq(v2[2], -195.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, rotate_x) { + mat4 m1 = GLM_MAT4_IDENTITY_INIT; + vec4 v1 = {0.0f, 1.0f, 0.0f, 1.0f}, v2 = {0.0f, 1.0f, 0.0f, 1.0f}; + + GLM(rotate_x)(m1, GLM_PI_2f, m1); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + glm_vec3_copy(v2, v1); + + GLM(rotate_x)(m1, GLM_PI_2f, m1); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], -1.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + glm_vec3_copy(v2, v1); + + GLM(rotate_x)(m1, GLM_PI_2f, m1); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], -1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, rotate_y) { + mat4 m1 = GLM_MAT4_IDENTITY_INIT; + vec4 v1 = {1.0f, 0.0f, 0.0f, 1.0f}, v2 = {1.0f, 0.0f, 0.0f, 1.0f}; + + GLM(rotate_y)(m1, GLM_PI_2f, m1); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], -1.0f)) + + glm_vec3_copy(v2, v1); + + GLM(rotate_y)(m1, GLM_PI_2f, m1); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], -1.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + glm_vec3_copy(v2, v1); + + GLM(rotate_y)(m1, GLM_PI_2f, m1); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, rotate_z) { + mat4 m1 = GLM_MAT4_IDENTITY_INIT; + vec4 v1 = {0.0f, 1.0f, 0.0f, 1.0f}, v2 = {0.0f, 1.0f, 0.0f, 1.0f}; + + GLM(rotate_z)(m1, GLM_PI_2f, m1); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], -1.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + glm_vec3_copy(v2, v1); + + GLM(rotate_z)(m1, GLM_PI_2f, m1); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], -1.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + glm_vec3_copy(v2, v1); + + GLM(rotate_z)(m1, GLM_PI_2f, m1); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 1.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, rotate_make) { + mat4 m1 = GLM_MAT4_IDENTITY_INIT; + vec4 v1 = {1.0f, 0.0f, 0.0f, 1.0f}; + + /* rotate X around Y = -Z */ + GLM(rotate_make)(m1, GLM_PI_2f, GLM_YUP); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], -1.0f)) + + /* rotate -Z around X = Y */ + GLM(rotate_make)(m1, GLM_PI_2f, GLM_XUP); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 1.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + /* rotate Y around X = +Z */ + GLM(rotate_make)(m1, GLM_PI_2f, GLM_XUP); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, rotate) { + mat4 m1 = GLM_MAT4_IDENTITY_INIT, m2 = GLM_MAT4_IDENTITY_INIT; + vec4 v1 = {1.0f, 0.0f, 0.0f, 1.0f}; + + /* 360 deg */ + glm_rotate(m1, GLM_PI_2f, GLM_YUP); + glm_rotate(m1, GLM_PI_2f, GLM_YUP); + glm_rotate(m1, GLM_PI_2f, GLM_YUP); + glm_rotate(m1, GLM_PI_2f, GLM_YUP); + ASSERTIFY(test_assert_mat4_eq(m1, m2)) + + /* rotate X around Y = -Z */ + GLM(rotate)(m1, GLM_PI_2f, GLM_YUP); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], -1.0f)) + + glm_mat4_identity(m1); + + /* rotate -Z around X = Y */ + GLM(rotate)(m1, GLM_PI_2f, GLM_XUP); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 1.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + glm_mat4_identity(m1); + + /* rotate Y around X = +Z */ + GLM(rotate)(m1, GLM_PI_2f, GLM_XUP); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, rotate_at) { + mat4 m1 = GLM_MAT4_IDENTITY_INIT; + vec4 v1 = {1.0f, 0.0f, 0.0f, 1.0f}; + + GLM(rotate_at)(m1, (vec3){0.5f, 0.0f, 0.0f}, GLM_PI_2f, GLM_YUP); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.5f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], -0.5f)) + + glm_mat4_identity(m1); + + GLM(rotate_at)(m1, GLM_VEC3_ZERO, GLM_PI_2f, GLM_ZUP); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.5f)) + ASSERT(test_eq(v1[2], -0.5f)) + + glm_mat4_identity(m1); + + v1[0] = 1.0f; + v1[1] = 1.0f; + v1[2] = 1.0f; + + GLM(rotate_at)(m1, GLM_VEC3_ZERO, GLM_PI_2f, GLM_XUP); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 1.0f)) + ASSERT(test_eq(v1[1], -1.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, rotate_atm) { + mat4 m1 = GLM_MAT4_IDENTITY_INIT; + vec4 v1 = {1.0f, 0.0f, 0.0f, 1.0f}; + + GLM(rotate_atm)(m1, (vec3){0.5f, 0.0f, 0.0f}, GLM_PI_2f, GLM_YUP); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.5f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], -0.5f)) + + GLM(rotate_atm)(m1, GLM_VEC3_ZERO, GLM_PI_2f, GLM_ZUP); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.5f)) + ASSERT(test_eq(v1[2], -0.5f)) + + v1[0] = 1.0f; + v1[1] = 1.0f; + v1[2] = 1.0f; + + GLM(rotate_atm)(m1, GLM_VEC3_ZERO, GLM_PI_2f, GLM_XUP); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 1.0f)) + ASSERT(test_eq(v1[1], -1.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, decompose_scalev) { + mat4 m1; + vec3 s1; + + GLM(scale_make)(m1, (vec3){7.0f, 8.0f, 9.0f}); + GLM(decompose_scalev)(m1, s1); + + ASSERT(test_eq(s1[0], 7.0f)) + ASSERT(test_eq(s1[1], 8.0f)) + ASSERT(test_eq(s1[2], 9.0f)) + + GLM(scale)(m1, (vec3){7.0f, 8.0f, 9.0f}); + GLM(decompose_scalev)(m1, s1); + + ASSERT(test_eq(s1[0], 49.0f)) + ASSERT(test_eq(s1[1], 64.0f)) + ASSERT(test_eq(s1[2], 81.0f)) + + glm_rotate(m1, GLM_PI_4f, (vec3){23.0f, 45.0f, 66.0f}); + ASSERT(test_eq(s1[0], 49.0f)) + ASSERT(test_eq(s1[1], 64.0f)) + ASSERT(test_eq(s1[2], 81.0f)) + + glm_translate(m1, (vec3){4.0f, 5.0f, 6.0f}); + ASSERT(test_eq(s1[0], 49.0f)) + ASSERT(test_eq(s1[1], 64.0f)) + ASSERT(test_eq(s1[2], 81.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, uniscaled) { + mat4 m1; + + GLM(scale_make)(m1, (vec3){7.0f, 8.0f, 9.0f}); + ASSERT(!GLM(uniscaled)(m1)) + + GLM(scale_make)(m1, (vec3){7.0f, 7.0f, 7.0f}); + ASSERT(GLM(uniscaled)(m1)) + + glm_rotate(m1, GLM_PI_4f, (vec3){23.0f, 45.0f, 66.0f}); + ASSERT(GLM(uniscaled)(m1)) + + glm_translate(m1, (vec3){4.0f, 5.0f, 6.0f}); + ASSERT(GLM(uniscaled)(m1)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, decompose_rs) { + mat4 m1, m2, r; + vec3 s1; + + GLM(scale_make)(m1, (vec3){7.0f, 8.0f, 9.0f}); + GLM(decompose_rs)(m1, r, s1); + + ASSERT(test_eq(s1[0], 7.0f)) + ASSERT(test_eq(s1[1], 8.0f)) + ASSERT(test_eq(s1[2], 9.0f)) + ASSERTIFY(test_assert_mat4_eq_identity(r)); + + GLM(scale)(m1, (vec3){7.0f, 8.0f, 9.0f}); + GLM(decompose_rs)(m1, r, s1); + + ASSERT(test_eq(s1[0], 49.0f)) + ASSERT(test_eq(s1[1], 64.0f)) + ASSERT(test_eq(s1[2], 81.0f)) + ASSERTIFY(test_assert_mat4_eq_identity(r)); + + glm_rotate(m1, GLM_PI_4f, (vec3){23.0f, 45.0f, 66.0f}); + ASSERT(test_eq(s1[0], 49.0f)) + ASSERT(test_eq(s1[1], 64.0f)) + ASSERT(test_eq(s1[2], 81.0f)) + GLM(decompose_rs)(m1, r, s1); + + glm_mat4_identity(m2); + glm_mat4_mul(m2, r, m2); + glm_scale(m2, s1); + + ASSERTIFY(test_assert_mat4_eq2(m1, m2, 0.00001f)); + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, decompose) { + mat4 m1, m2, r; + vec4 t; + vec3 s; + + GLM(scale_make)(m1, (vec3){7.0f, 8.0f, 9.0f}); + GLM(decompose)(m1, t, r, s); + + ASSERT(test_eq(s[0], 7.0f)) + ASSERT(test_eq(s[1], 8.0f)) + ASSERT(test_eq(s[2], 9.0f)) + ASSERTIFY(test_assert_mat4_eq_identity(r)); + + GLM(scale)(m1, (vec3){7.0f, 8.0f, 9.0f}); + GLM(decompose)(m1, t, r, s); + + ASSERT(test_eq(s[0], 49.0f)) + ASSERT(test_eq(s[1], 64.0f)) + ASSERT(test_eq(s[2], 81.0f)) + ASSERTIFY(test_assert_mat4_eq_identity(r)); + + glm_rotate(m1, GLM_PI_4f, (vec3){23.0f, 45.0f, 66.0f}); + ASSERT(test_eq(s[0], 49.0f)) + ASSERT(test_eq(s[1], 64.0f)) + ASSERT(test_eq(s[2], 81.0f)) + GLM(decompose)(m1, t, r, s); + + glm_mat4_identity(m2); + glm_mat4_mul(m2, r, m2); + glm_scale(m2, s); + + ASSERTIFY(test_assert_mat4_eq2(m1, m2, 0.00001f)); + + glm_mat4_identity(m1); + glm_translate(m1, (vec3){56.0f, 13.0f, 90.0f}); + glm_rotate(m1, GLM_PI_4f, (vec3){23.0f, 45.0f, 66.0f}); + glm_scale(m1, (vec3){12.0f, 34.0f, 23.0f}); + + GLM(decompose)(m1, t, r, s); + + ASSERT(test_eq(t[0], 56.0f)) + ASSERT(test_eq(t[1], 13.0f)) + ASSERT(test_eq(t[2], 90.0f)) + + ASSERT(test_eq(s[0], 12.0f)) + ASSERT(test_eq(s[1], 34.0f)) + ASSERT(test_eq(s[2], 23.0f)) + + glm_mat4_identity(m2); + glm_translate(m2, t); + glm_mat4_mul(m2, r, m2); + glm_scale(m2, s); + ASSERTIFY(test_assert_mat4_eq2(m1, m2, 0.00001f)); + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_affine2d.h b/cglm/test/src/test_affine2d.h new file mode 100644 index 0000000..8828661 --- /dev/null +++ b/cglm/test/src/test_affine2d.h @@ -0,0 +1,310 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +TEST_IMPL(GLM_PREFIX, translate2d) { + mat3 m1; + vec3 v1 = {2.0f, 3.0f, 1.0f}, v2; + + glm_mat3_identity(m1); + GLM(translate2d)(m1, (vec2){13.0f, 11.0f}); + glm_mat3_mulv(m1, v1, v2); + + ASSERT(test_eq(v2[0], 15.0f)) + ASSERT(test_eq(v2[1], 14.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + glm_mat3_identity(m1); + GLM(translate2d)(m1, (vec2){1.0f, -1.0f}); + glm_mat3_mulv(m1, v2, v2); + + ASSERT(test_eq(v2[0], 16.0f)) + ASSERT(test_eq(v2[1], 13.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, translate2d_to) { + mat3 m1, m2; + vec3 v1 = {2.0f, 3.0f, 1.0f}, v2; + + glm_mat3_identity(m1); + GLM(translate2d_to)(m1, (vec2){13.0f, 11.0f}, m2); + glm_mat3_mulv(m2, v1, v2); + + ASSERT(test_eq(v2[0], 15.0f)) + ASSERT(test_eq(v2[1], 14.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + glm_mat3_identity(m1); + GLM(translate2d_to)(m1, (vec2){1.0f, -1.0f}, m2); + glm_mat3_mulv(m2, v2, v2); + + ASSERT(test_eq(v2[0], 16.0f)) + ASSERT(test_eq(v2[1], 13.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, translate2d_x) { + mat3 m1; + vec3 v1 = {2.0f, 3.0f, 1.0f}, v2; + + glm_mat3_identity(m1); + GLM(translate2d_x)(m1, 13.0f); + glm_mat3_mulv(m1, v1, v2); + + ASSERT(test_eq(v2[0], 15.0f)) + ASSERT(test_eq(v2[1], 3.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + glm_mat3_identity(m1); + GLM(translate2d_x)(m1, -1.0f); + glm_mat3_mulv(m1, v2, v2); + + ASSERT(test_eq(v2[0], 14.0f)) + ASSERT(test_eq(v2[1], 3.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, translate2d_y) { + mat3 m1; + vec3 v1 = {2.0f, 3.0f, 1.0f}, v2; + + glm_mat3_identity(m1); + GLM(translate2d_y)(m1, 11.0f); + glm_mat3_mulv(m1, v1, v2); + + + ASSERT(test_eq(v2[0], 2.0f)) + ASSERT(test_eq(v2[1], 14.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + glm_mat3_identity(m1); + GLM(translate2d_y)(m1, -1.0f); + glm_mat3_mulv(m1, v2, v2); + + ASSERT(test_eq(v2[0], 2.0f)) + ASSERT(test_eq(v2[1], 13.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, translate2d_make) { + mat3 m1; + vec3 v1 = {2.0f, 3.0f, 1.0f}, v2; + + glm_mat3_identity(m1); + GLM(translate2d_make)(m1, (vec2){13.0f, 11.0f}); + glm_mat3_mulv(m1, v1, v2); + + ASSERT(test_eq(v2[0], 15.0f)) + ASSERT(test_eq(v2[1], 14.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + glm_mat3_identity(m1); + GLM(translate2d_make)(m1, (vec2){-1.0f, -5.0f}); + glm_mat3_mulv(m1, v2, v2); + + ASSERT(test_eq(v2[0], 14.0f)) + ASSERT(test_eq(v2[1], 9.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, scale2d_to) { + mat3 m1, m2; + vec3 v1 = {2.0f, 3.0f, 1.0f}, v2; + + glm_mat3_identity(m1); + GLM(scale2d_to)(m1, (vec2){13.0f, 11.0f}, m2); + glm_mat3_mulv(m2, v1, v2); + + ASSERT(test_eq(v2[0], 26.0f)) + ASSERT(test_eq(v2[1], 33.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + glm_mat3_identity(m1); + GLM(scale2d_to)(m1, (vec2){-1.0f, -5.0f}, m2); + glm_mat3_mulv(m2, v2, v2); + + ASSERT(test_eq(v2[0], -26.0f)) + ASSERT(test_eq(v2[1], -165.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, scale2d_make) { + mat3 m1; + vec3 v1 = {2.0f, 3.0f, 1.0f}, v2; + + GLM(scale2d_make)(m1, (vec2){13.0f, 11.0f}); + glm_mat3_mulv(m1, v1, v2); + + ASSERT(test_eq(v2[0], 26.0f)) + ASSERT(test_eq(v2[1], 33.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + GLM(scale2d_make)(m1, (vec3){-1.0f, -5.0f}); + glm_mat3_mulv(m1, v2, v2); + + ASSERT(test_eq(v2[0], -26.0f)) + ASSERT(test_eq(v2[1], -165.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, scale2d) { + mat3 m1; + vec3 v1 = {2.0f, 3.0f, 1.0f}, v2; + + glm_mat3_identity(m1); + GLM(scale2d)(m1, (vec2){13.0f, 11.0f}); + glm_mat3_mulv(m1, v1, v2); + + ASSERT(test_eq(v2[0], 26.0f)) + ASSERT(test_eq(v2[1], 33.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + glm_mat3_identity(m1); + GLM(scale2d)(m1, (vec2){-1.0f, -5.0f}); + glm_mat3_mulv(m1, v2, v2); + + ASSERT(test_eq(v2[0], -26.0f)) + ASSERT(test_eq(v2[1], -165.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, scale2d_uni) { + mat3 m1; + vec3 v1 = {2.0f, 3.0f, 1.0f}, v2; + + glm_mat3_identity(m1); + GLM(scale2d_uni)(m1, 13.0f); + glm_mat3_mulv(m1, v1, v2); + + ASSERT(test_eq(v2[0], 26.0f)) + ASSERT(test_eq(v2[1], 39.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + glm_mat3_identity(m1); + GLM(scale2d_uni)(m1, -5.0f); + glm_mat3_mulv(m1, v2, v2); + + ASSERT(test_eq(v2[0], -130.0f)) + ASSERT(test_eq(v2[1], -195.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, rotate2d_make) { + mat3 m1 = GLM_MAT3_IDENTITY_INIT; + vec3 v1 = {0.0f, 1.0f, 1.0f}, v2 = {0.0f, 1.0f, 1.0f}; + + GLM(rotate2d_make)(m1, GLM_PI_2f); + glm_mat3_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], -1.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + glm_vec3_copy(v2, v1); + + GLM(rotate2d_make)(m1, -GLM_PI_2f); + glm_mat3_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 1.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + glm_vec3_copy(v2, v1); + + GLM(rotate2d_make)(m1, GLM_PIf); + glm_mat3_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], -1.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, rotate2d) { + mat3 m1 = GLM_MAT3_IDENTITY_INIT; + vec3 v1 = {0.0f, 1.0f, 1.0f}, v2 = {0.0f, 1.0f, 1.0f}; + + GLM(rotate2d)(m1, GLM_PI_2f); + glm_mat3_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], -1.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + glm_vec3_copy(v2, v1); + + GLM(rotate2d)(m1, GLM_PI_2f); + glm_mat3_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], -1.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + glm_vec3_copy(v2, v1); + + GLM(rotate2d)(m1, GLM_PI_2f); + glm_mat3_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 1.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, rotate2d_to) { + mat3 m1 = GLM_MAT3_IDENTITY_INIT, m2; + vec3 v1 = {0.0f, 1.0f, 1.0f}, v2 = {0.0f, 1.0f, 1.0f}; + + GLM(rotate2d_to)(m1, GLM_PI_2f, m1); + glm_mat3_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], -1.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + glm_vec3_copy(v2, v1); + + GLM(rotate2d_to)(m1, GLM_PI_2f, m2); + glm_mat3_mulv(m2, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], -1.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + glm_vec3_copy(v2, v1); + + GLM(rotate2d_to)(m2, GLM_PI_2f, m1); + glm_mat3_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 1.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_affine_mat.h b/cglm/test/src/test_affine_mat.h new file mode 100644 index 0000000..e928f8e --- /dev/null +++ b/cglm/test/src/test_affine_mat.h @@ -0,0 +1,112 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +#ifndef glm_affine_mat_test_guard +#define glm_affine_mat_test_guard +CGLM_INLINE +void +glm_inv_tr_raw(mat4 mat) { + CGLM_ALIGN_MAT mat3 r; + CGLM_ALIGN(8) vec3 t; + + /* rotate */ + glm_mat4_pick3t(mat, r); + glm_mat4_ins3(r, mat); + + /* translate */ + glm_mat3_mulv(r, mat[3], t); + glm_vec3_negate(t); + glm_vec3_copy(t, mat[3]); +} +#endif + +TEST_IMPL(GLM_PREFIX, mul) { + mat4 m1 = GLM_MAT4_IDENTITY_INIT; + mat4 m2 = GLM_MAT4_IDENTITY_INIT; + mat4 m3; + mat4 m4 = GLM_MAT4_ZERO_INIT; + int i, j, k; + + test_rand_mat4(m1); + test_rand_mat4(m2); + + GLM(mul)(m1, m2, m3); + for (i = 0; i < 4; i++) { + for (j = 0; j < 4; j++) { + for (k = 0; k < 4; k++) + /* column-major */ + m4[i][j] += m1[k][j] * m2[i][k]; + } + } + + ASSERTIFY(test_assert_mat4_eq(m3, m4)) + + /* test pre compiled */ + GLM(mul)(m1, m2, m3); + ASSERTIFY(test_assert_mat4_eq(m3, m4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mul_rot) { + mat4 m1 = GLM_MAT4_IDENTITY_INIT; + mat4 m2 = GLM_MAT4_IDENTITY_INIT; + mat4 m3; + mat4 m4 = GLM_MAT4_ZERO_INIT; + int i, j, k; + + glm_rotate(m1, drand48(), (vec3){drand48(), drand48(), drand48()}); + glm_rotate(m2, drand48(), (vec3){drand48(), drand48(), drand48()}); + + GLM(mul_rot)(m1, m2, m3); + for (i = 0; i < 4; i++) { + for (j = 0; j < 4; j++) { + for (k = 0; k < 4; k++) + /* column-major */ + m4[i][j] += m1[k][j] * m2[i][k]; + } + } + + ASSERTIFY(test_assert_mat4_eq(m3, m4)) + + /* test pre compiled */ + GLM(mul_rot)(m1, m2, m3); + ASSERTIFY(test_assert_mat4_eq(m3, m4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, inv_tr) { + mat4 m1, m2; + int i; + + for (i = 0; i < 10000; i++) { + test_rand_mat4(m1); + + glm_mat4_copy(m1, m2); + + /* test inverse precise */ + GLM(inv_tr)(m1); + GLM(inv_tr)(m1); + ASSERTIFY(test_assert_mat4_eq(m1, m2)) + + /* test inverse precise */ + GLM(mat4_inv)(m1, m2); + GLM(inv_tr)(m2); + ASSERTIFY(test_assert_mat4_eq(m1, m2)) + + /* test with raw */ + glm_mat4_copy(m1, m2); + glm_inv_tr_raw(m2); + GLM(inv_tr)(m1); + ASSERTIFY(test_assert_mat4_eq(m1, m2)) + } + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_bezier.c b/cglm/test/src/test_bezier.c new file mode 100644 index 0000000..3089dd5 --- /dev/null +++ b/cglm/test/src/test_bezier.c @@ -0,0 +1,66 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +CGLM_INLINE +float +test_bezier_plain(float s, float p0, float c0, float c1, float p1) { + float x, xx, xxx, ss, sss; + + x = 1.0f - s; + xx = x * x; + xxx = xx * x; + ss = s * s; + sss = ss * s; + + return p0 * xxx + 3.0f * (c0 * s * xx + c1 * ss * x) + p1 * sss; +} + +CGLM_INLINE +float +test_hermite_plain(float s, float p0, float t0, float t1, float p1) { + float ss, sss; + + ss = s * s; + sss = ss * s; + + return p0 * (2.0f * sss - 3.0f * ss + 1.0f) + + t0 * (sss - 2.0f * ss + s) + + p1 * (-2.0f * sss + 3.0f * ss) + + t1 * (sss - ss); +} + +TEST_IMPL(bezier) { + float s, p0, p1, c0, c1, smc, Bs, Bs_plain; + + s = test_rand(); + p0 = test_rand(); + p1 = test_rand(); + c0 = test_rand(); + c1 = test_rand(); + + /* test cubic bezier */ + smc = glm_smc(s, GLM_BEZIER_MAT, (vec4){p0, c0, c1, p1}); + Bs = glm_bezier(s, p0, c0, c1, p1); + Bs_plain = test_bezier_plain(s, p0, c0, c1, p1); + + ASSERT(test_eq(Bs, Bs_plain)); + ASSERTIFY(test_assert_eqf(smc, Bs_plain)) + ASSERTIFY(test_assert_eqf(Bs, smc)) + + /* test cubic hermite */ + smc = glm_smc(s, GLM_HERMITE_MAT, (vec4){p0, p1, c0, c1}); + Bs = glm_hermite(s, p0, c0, c1, p1); + Bs_plain = test_hermite_plain(s, p0, c0, c1, p1); + + ASSERT(test_eq(Bs, Bs_plain)); + ASSERT(test_eq(smc, Bs_plain)); + ASSERT(test_eq(Bs, smc)); + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_cam.h b/cglm/test/src/test_cam.h new file mode 100644 index 0000000..9558b6f --- /dev/null +++ b/cglm/test/src/test_cam.h @@ -0,0 +1,115 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +TEST_IMPL(GLM_PREFIX, frustum) { + mat4 proj; + vec4 vp = {0.0f, 0.0f, 800.0f, 600.0f}; + float left, right, top, bottom, znear, zfar; + + znear = 0.1f; + zfar = 100.0f; + left = -100.0f; + right = 100.0f; + bottom = -100.0f; + top = 100.0f; + + GLM(frustum)(left, right, bottom, top, znear, zfar, proj); + + ASSERT(test_eq(proj[0][1], 0.0f)) + ASSERT(test_eq(proj[0][2], 0.0f)) + ASSERT(test_eq(proj[0][3], 0.0f)) + + ASSERT(test_eq(proj[1][0], 0.0f)) + ASSERT(test_eq(proj[1][2], 0.0f)) + ASSERT(test_eq(proj[1][3], 0.0f)) + + ASSERT(test_eq(proj[2][3], -1.0f)) + + ASSERT(test_eq(proj[3][0], 0.0f)) + ASSERT(test_eq(proj[3][1], 0.0f)) + ASSERT(test_eq(proj[3][3], 0.0f)) + + vec4 v1 = {1.0f, 20.0f, znear}; + vec4 v2 = {1.0f, 20.0f, zfar}; + vec4 v3, v4; + + /* perspective test */ + GLM(mat4_mulv)(proj, v1, v3); + GLM(project)(v3, proj, vp, v3); + + ASSERT(v3[0] > v1[0]) + ASSERT(v3[1] > v1[1]) + + GLM(mat4_mulv)(proj, v2, v4); + GLM(project)(v4, proj, vp, v4); + + ASSERT(v4[0] < v3[0]) + ASSERT(v4[1] < v3[1]) + + /* not infinity */ + ASSERT(!GLM(vec4_isinf)(proj[0])) + ASSERT(!GLM(vec4_isinf)(proj[1])) + ASSERT(!GLM(vec4_isinf)(proj[2])) + ASSERT(!GLM(vec4_isinf)(proj[3])) + + /* not NaN */ + ASSERT(!GLM(vec4_isnan)(proj[0])) + ASSERT(!GLM(vec4_isnan)(proj[1])) + ASSERT(!GLM(vec4_isnan)(proj[2])) + ASSERT(!GLM(vec4_isnan)(proj[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, camera_lookat) { + mat4 view1, view2; + vec3 center, + eye = {0.024f, 14.6f, 67.04f}, + dir = {0.0f, 0.0f, -1.0f}, + up = {0.0f, 1.0f, 0.0f}; + + glm_vec3_add(eye, dir, center); + glm_lookat(eye, center, up, view1); + + glm_look(eye, dir, up, view2); + + ASSERTIFY(test_assert_mat4_eq(view1, view2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, camera_decomp) { + mat4 proj, proj2; + vec4 sizes; + float aspect, fovy, nearZ, farZ; + + aspect = 0.782f; + fovy = glm_rad(49.984f); + nearZ = 0.1f; + farZ = 100.0f; + + glm_perspective(fovy, aspect, nearZ, farZ, proj); + ASSERT(fabsf(aspect - glm_persp_aspect(proj)) < GLM_FLT_EPSILON) + ASSERT(fabsf(fovy - glm_persp_fovy(proj)) < GLM_FLT_EPSILON) + ASSERT(fabsf(49.984f - glm_deg(glm_persp_fovy(proj))) < GLM_FLT_EPSILON) + + glm_persp_sizes(proj, fovy, sizes); + + glm_frustum(-sizes[0] * 0.5f, + sizes[0] * 0.5f, + -sizes[1] * 0.5f, + sizes[1] * 0.5f, + nearZ, + farZ, + proj2); + + ASSERTIFY(test_assert_mat4_eq(proj, proj2)) + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_cam_lh_no.h b/cglm/test/src/test_cam_lh_no.h new file mode 100644 index 0000000..d282a1a --- /dev/null +++ b/cglm/test/src/test_cam_lh_no.h @@ -0,0 +1,38 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" +#include "../../include/cglm/clipspace/persp_lh_no.h" +#include "../../include/cglm/call/clipspace/persp_lh_no.h" + +TEST_IMPL(GLM_PREFIX, perspective_lh_no) { + mat4 dst; + const float fovy = glm_rad(45.0f); + const float aspect = 640/480.0f; + const float zNearVal = 0.1f; + const float zFarVal = 100.0f; + + GLM(perspective_lh_no)(fovy, aspect, zNearVal, zFarVal, dst); + + /* Sanity mk. I: longhand version */ + ASSERT(test_eq(dst[0][0], 1.0f / (tanf(fovy / 2) * aspect))) + ASSERT(test_eq(dst[1][1], 1.0f / tanf(fovy / 2))) + ASSERT(test_eq(dst[2][2], (zFarVal + zNearVal) / (zFarVal - zNearVal))) + ASSERT(test_eq(dst[2][3], 1.0f)) + ASSERT(test_eq(dst[3][2], -2 * zFarVal * zNearVal / (zFarVal - zNearVal))) + + /* Sanity mk. II */ + /*reference test data for glm_perspective_lh_no*/ + mat4 cmp = {0}; + cmp[0][0] = 1.8106601f; + cmp[1][1] = 2.4142134f; + cmp[2][2] = 1.0020020f; + cmp[2][3] = 1.0000000f; + cmp[3][2] = -0.2002002f; + + return test_assert_mat4_eq(dst, cmp); +} diff --git a/cglm/test/src/test_cam_lh_zo.h b/cglm/test/src/test_cam_lh_zo.h new file mode 100644 index 0000000..98be0fb --- /dev/null +++ b/cglm/test/src/test_cam_lh_zo.h @@ -0,0 +1,38 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" +#include "../../include/cglm/clipspace/persp_lh_zo.h" +#include "../../include/cglm/call/clipspace/persp_lh_zo.h" + +TEST_IMPL(GLM_PREFIX, perspective_lh_zo) { + mat4 dst; + const float fovy = glm_rad(45.0f); + const float aspect = 640/480.0f; + const float zNearVal = 0.1f; + const float zFarVal = 100.0f; + + GLM(perspective_lh_zo)(fovy, aspect, zNearVal, zFarVal, dst); + + /* Sanity mk. I: longhand version */ + ASSERT(test_eq(dst[0][0], 1.0f / (tanf(fovy / 2) * aspect))) + ASSERT(test_eq(dst[1][1], 1.0f / tanf(fovy / 2))) + ASSERT(test_eq(dst[2][2], zFarVal / (zFarVal - zNearVal))) + ASSERT(test_eq(dst[2][3], 1.0f)) + ASSERT(test_eq(dst[3][2], -1 * zFarVal * zNearVal / (zFarVal - zNearVal))) + + /* Sanity mk. II */ + /* "Reference values" generated by GLM's glm::perspectiveLH_ZO */ + mat4 cmp = {0}; + cmp[0][0] = 1.8106601f; + cmp[1][1] = 2.4142134f; + cmp[2][2] = 1.0010010f; + cmp[2][3] = 1.0000000f; + cmp[3][2] = -0.1001001f; + + return test_assert_mat4_eq(dst, cmp); +} diff --git a/cglm/test/src/test_cam_rh_no.h b/cglm/test/src/test_cam_rh_no.h new file mode 100644 index 0000000..1fcd597 --- /dev/null +++ b/cglm/test/src/test_cam_rh_no.h @@ -0,0 +1,38 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" +#include "../../include/cglm/clipspace/persp_rh_no.h" +#include "../../include/cglm/call/clipspace/persp_rh_no.h" + +TEST_IMPL(GLM_PREFIX, perspective_rh_no) { + mat4 dst; + const float fovy = glm_rad(45.0f); + const float aspect = 640/480.0f; + const float zNearVal = 0.1f; + const float zFarVal = 100.0f; + + GLM(perspective_rh_no)(fovy, aspect, zNearVal, zFarVal, dst); + + /* Sanity mk. I: longhand version */ + ASSERT(test_eq(dst[0][0], 1.0f / (tanf(fovy / 2) * aspect))) + ASSERT(test_eq(dst[1][1], 1.0f / tanf(fovy / 2))) + ASSERT(test_eq(dst[2][2], -1.0f * (zFarVal + zNearVal) / (zFarVal - zNearVal))) + ASSERT(test_eq(dst[2][3], -1.0f)) + ASSERT(test_eq(dst[3][2], -2 * zFarVal * zNearVal / (zFarVal - zNearVal))) + + /* Sanity mk. II */ + /*reference test data for glm_perspective_rh_no*/ + mat4 cmp = {0}; + cmp[0][0] = 1.8106601f; + cmp[1][1] = 2.4142134f; + cmp[2][2] = -1.0020020f; + cmp[2][3] = -1.0000000f; + cmp[3][2] = -0.2002002f; + + return test_assert_mat4_eq(dst, cmp); +} diff --git a/cglm/test/src/test_cam_rh_zo.h b/cglm/test/src/test_cam_rh_zo.h new file mode 100644 index 0000000..76f72ea --- /dev/null +++ b/cglm/test/src/test_cam_rh_zo.h @@ -0,0 +1,38 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" +#include "../../include/cglm/clipspace/persp_rh_zo.h" +#include "../../include/cglm/call/clipspace/persp_rh_zo.h" + +TEST_IMPL(GLM_PREFIX, perspective_rh_zo) { + mat4 dst; + const float fovy = glm_rad(45.0f); + const float aspect = 640/480.0f; + const float zNearVal = 0.1f; + const float zFarVal = 100.0f; + + GLM(perspective_rh_zo)(fovy, aspect, zNearVal, zFarVal, dst); + + /* Sanity mk. I: longhand version */ + ASSERT(test_eq(dst[0][0], 1 / (tanf(fovy / 2) * aspect))) + ASSERT(test_eq(dst[1][1], 1 / tanf(fovy / 2))) + ASSERT(test_eq(dst[2][2], zFarVal / (zNearVal - zFarVal))) + ASSERT(test_eq(dst[2][3], -1.0f)) + ASSERT(test_eq(dst[3][2], -1 * zFarVal * zNearVal / (zFarVal - zNearVal))) + + /* Sanity mk. II */ + /*reference test data for glm_perspective_rh_zo*/ + mat4 cmp = {0}; + cmp[0][0] = 1.8106601f; + cmp[1][1] = 2.4142134f; + cmp[2][2] = -1.0010010f; + cmp[2][3] = -1.0000000f; + cmp[3][2] = -0.1001001f; + + return test_assert_mat4_eq(dst, cmp); +} diff --git a/cglm/test/src/test_clamp.c b/cglm/test/src/test_clamp.c new file mode 100644 index 0000000..fdd4652 --- /dev/null +++ b/cglm/test/src/test_clamp.c @@ -0,0 +1,31 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +TEST_IMPL(clamp) { + vec3 v3 = {15.07f, 0.4f, 17.3f}; + vec4 v4 = {5.07f, 2.3f, 1.3f, 1.4f}; + + ASSERT(glm_eq(glm_clamp(1.6f, 0.0f, 1.0f), 1.0f)) + ASSERT(glm_eq(glm_clamp(-1.6f, 0.0f, 1.0f), 0.0f)) + ASSERT(glm_eq(glm_clamp(0.6f, 0.0f, 1.0f), 0.6f)) + + glm_vec3_clamp(v3, 0.0, 1.0); + glm_vec4_clamp(v4, 1.5, 3.0); + + ASSERT(glm_eq(v3[0], 1.0f)) + ASSERT(glm_eq(v3[1], 0.4f)) + ASSERT(glm_eq(v3[2], 1.0f)) + + ASSERT(glm_eq(v4[0], 3.0f)) + ASSERT(glm_eq(v4[1], 2.3f)) + ASSERT(glm_eq(v4[2], 1.5f)) + ASSERT(glm_eq(v4[3], 1.5f)) + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_common.c b/cglm/test/src/test_common.c new file mode 100644 index 0000000..a6c31c6 --- /dev/null +++ b/cglm/test/src/test_common.c @@ -0,0 +1,581 @@ +/* + * Copyright (c), Recep Aslantas. + * MIT License (MIT), http://opensource.org/licenses/MIT + */ + +#include "test_common.h" +#include + +void +test_rand_mat4(mat4 dest) { + glm_mat4_copy(GLM_MAT4_IDENTITY, dest); + + /* random position */ + dest[3][0] = drand48(); + dest[3][1] = drand48(); + dest[3][2] = drand48(); + + /* random rotatation around random axis with random angle */ + glm_rotate(dest, drand48(), (vec3){drand48(), drand48(), drand48()}); + + /* random scale */ + /* glm_scale(dest, (vec3){drand48(), drand48(), drand48()}); */ +} + +void +test_rand_mat4x2(mat4x2 dest) { + dest[0][0] = drand48(); + dest[0][1] = drand48(); + + dest[1][0] = drand48(); + dest[1][1] = drand48(); + + dest[2][0] = drand48(); + dest[2][1] = drand48(); + + dest[3][0] = drand48(); + dest[3][1] = drand48(); +} + +void +test_rand_mat4x3(mat4x3 dest) { + dest[0][0] = drand48(); + dest[0][1] = drand48(); + dest[0][2] = drand48(); + + dest[1][0] = drand48(); + dest[1][1] = drand48(); + dest[1][2] = drand48(); + + dest[2][0] = drand48(); + dest[2][1] = drand48(); + dest[2][2] = drand48(); + + dest[3][0] = drand48(); + dest[3][1] = drand48(); + dest[3][2] = drand48(); +} + +void +test_rand_mat3(mat3 dest) { + mat4 m4; + + /* random rotatation around random axis with random angle */ + glm_rotate_make(m4, drand48(), (vec3){drand48(), drand48(), drand48()}); + glm_mat4_pick3(m4, dest); +} + +void +test_rand_mat3x2(mat3x2 dest) { + dest[0][0] = drand48(); + dest[0][1] = drand48(); + dest[1][0] = drand48(); + dest[1][1] = drand48(); + dest[2][0] = drand48(); + dest[2][1] = drand48(); +} + +void +test_rand_mat3x4(mat3x4 dest) { + dest[0][0] = drand48(); + dest[0][1] = drand48(); + dest[0][2] = drand48(); + dest[0][3] = drand48(); + + dest[1][0] = drand48(); + dest[1][1] = drand48(); + dest[1][2] = drand48(); + dest[1][3] = drand48(); + + dest[2][0] = drand48(); + dest[2][1] = drand48(); + dest[2][2] = drand48(); + dest[2][3] = drand48(); +} + +void +test_rand_mat2(mat2 dest) { + dest[0][0] = drand48(); + dest[0][1] = drand48(); + dest[1][0] = drand48(); + dest[1][1] = drand48(); +} + +void +test_rand_mat2x3(mat2x3 dest) { + dest[0][0] = drand48(); + dest[0][1] = drand48(); + dest[0][2] = drand48(); + dest[1][0] = drand48(); + dest[1][1] = drand48(); + dest[1][2] = drand48(); +} + +void +test_rand_mat2x4(mat2x4 dest) { + dest[0][0] = drand48(); + dest[0][1] = drand48(); + dest[0][2] = drand48(); + dest[0][3] = drand48(); + dest[1][0] = drand48(); + dest[1][1] = drand48(); + dest[1][2] = drand48(); + dest[1][3] = drand48(); +} + +void +test_rand_vec3(vec3 dest) { + dest[0] = drand48(); + dest[1] = drand48(); + dest[2] = drand48(); +} + +vec3s +test_rand_vec3s(void) { + vec3s r; + test_rand_vec3(r.raw); + return r; +} + +void +test_rand_vec4(vec4 dest) { + dest[0] = drand48(); + dest[1] = drand48(); + dest[2] = drand48(); + dest[3] = drand48(); +} + +vec4s +test_rand_vec4s(void) { + vec4s r; + test_rand_vec4(r.raw); + return r; +} + +float +test_rand(void) { + return drand48(); +} + +void +test_rand_quat(versor q) { + glm_quat(q, drand48(), drand48(), drand48(), drand48()); + glm_quat_normalize(q); +} + +test_status_t +test_assert_mat4_eq(mat4 m1, mat4 m2) { + int i, j; + + for (i = 0; i < 4; i++) { + for (j = 0; j < 4; j++) { + ASSERT(fabsf(m1[i][j] - m2[i][j]) <= 0.0000009) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat4_eqt(mat4 m1, mat4 m2) { + int i, j; + + for (i = 0; i < 4; i++) { + for (j = 0; j < 4; j++) { + ASSERT(fabsf(m1[j][i] - m2[i][j]) <= 0.0000009) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat4_eq2(mat4 m1, mat4 m2, float eps) { + int i, j; + + for (i = 0; i < 4; i++) { + for (j = 0; j < 4; j++) { + ASSERT(fabsf(m1[i][j] - m2[i][j]) <= eps); + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat2_eqt(mat2 m1, mat2 m2) { + int i, j; + + for (i = 0; i < 2; i++) { + for (j = 0; j < 2; j++) { + ASSERT(fabsf(m1[j][i] - m2[i][j]) <= 0.0000009); + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat2_eq(mat2 m1, mat2 m2) { + int i, j; + + for (i = 0; i < 2; i++) { + for (j = 0; j < 2; j++) { + ASSERT(fabsf(m1[i][j] - m2[i][j]) <= 0.0000009); + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat2_eq_identity(mat2 m2) { + int i, j; + + for (i = 0; i < 2; i++) { + for (j = 0; j < 2; j++) { + if (i == j) { + ASSERT(test_eq(m2[i][j], 1.0f)) + } else { + ASSERT(test_eq(m2[i][j], 0.0f)) + } + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat2_eq_zero(mat2 m2) { + int i, j; + + for (i = 0; i < 2; i++) { + for (j = 0; j < 2; j++) { + ASSERT(test_eq(m2[i][j], 0.0f)) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat2x3_eq_zero(mat2x3 m2x3) { + int i, j; + + for (i = 0; i < 2; i++) { + for (j = 0; j < 3; j++) { + ASSERT(test_eq(m2x3[i][j], 0.0f)) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat2x3_eq(mat2x3 m1, mat2x3 m2) { + int i, j; + + for (i = 0; i < 2; i++) { + for (j = 0; j < 3; j++) { + ASSERT(fabsf(m1[i][j] - m2[i][j]) <= 0.0000009) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat2x4_eq_zero(mat2x4 m2x4) { + int i, j; + + for (i = 0; i < 2; i++) { + for (j = 0; j < 4; j++) { + ASSERT(test_eq(m2x4[i][j], 0.0f)) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat2x4_eq(mat2x4 m1, mat2x4 m2) { + int i, j; + + for (i = 0; i < 2; i++) { + for (j = 0; j < 4; j++) { + ASSERT(fabsf(m1[i][j] - m2[i][j]) <= 0.0000009) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat3_eq(mat3 m1, mat3 m2) { + int i, j; + + for (i = 0; i < 3; i++) { + for (j = 0; j < 3; j++) { + ASSERT(fabsf(m1[i][j] - m2[i][j]) <= 0.0000009); + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat3_eqt(mat3 m1, mat3 m2) { + int i, j; + + for (i = 0; i < 3; i++) { + for (j = 0; j < 3; j++) { + ASSERT(fabsf(m1[j][i] - m2[i][j]) <= 0.0000009); + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat3_eq_identity(mat3 m3) { + int i, j; + + for (i = 0; i < 3; i++) { + for (j = 0; j < 3; j++) { + if (i == j) { + ASSERT(test_eq(m3[i][j], 1.0f)) + } else { + ASSERT(test_eq(m3[i][j], 0.0f)) + } + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat3_eq_zero(mat3 m3) { + int i, j; + + for (i = 0; i < 3; i++) { + for (j = 0; j < 3; j++) { + ASSERT(test_eq(m3[i][j], 0.0f)) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat3x2_eq_zero(mat3x2 m3x2) { + int i, j; + + for (i = 0; i < 3; i++) { + for (j = 0; j < 2; j++) { + ASSERT(test_eq(m3x2[i][j], 0.0f)) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat3x2_eq(mat3x2 m1, mat3x2 m2) { + int i, j; + + for (i = 0; i < 3; i++) { + for (j = 0; j < 2; j++) { + ASSERT(fabsf(m1[i][j] - m2[i][j]) <= 0.0000009) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat3x4_eq_zero(mat3x4 m3x4) { + int i, j; + + for (i = 0; i < 3; i++) { + for (j = 0; j < 4; j++) { + ASSERT(test_eq(m3x4[i][j], 0.0f)) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat3x4_eq(mat3x4 m1, mat3x4 m2) { + int i, j; + + for (i = 0; i < 3; i++) { + for (j = 0; j < 4; j++) { + ASSERT(fabsf(m1[i][j] - m2[i][j]) <= 0.0000009) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat4_eq_identity(mat4 m4) { + int i, j; + + for (i = 0; i < 4; i++) { + for (j = 0; j < 4; j++) { + if (i == j) { + ASSERT(test_eq(m4[i][j], 1.0f)) + } else { + ASSERT(test_eq(m4[i][j], 0.0f)) + } + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat4_eq_zero(mat4 m4) { + int i, j; + + for (i = 0; i < 4; i++) { + for (j = 0; j < 4; j++) { + ASSERT(test_eq(m4[i][j], 0.0f)) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat4x2_eq_zero(mat4x2 m4x2) { + int i, j; + + for (i = 0; i < 4; i++) { + for (j = 0; j < 2; j++) { + ASSERT(test_eq(m4x2[i][j], 0.0f)) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat4x2_eq(mat4x2 m1, mat4x2 m2) { + int i, j; + + for (i = 0; i < 4; i++) { + for (j = 0; j < 2; j++) { + ASSERT(fabsf(m1[i][j] - m2[i][j]) <= 0.0000009) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat4x3_eq_zero(mat4x3 m4x3) { + int i, j; + + for (i = 0; i < 4; i++) { + for (j = 0; j < 3; j++) { + ASSERT(test_eq(m4x3[i][j], 0.0f)) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_mat4x3_eq(mat4x3 m1, mat4x3 m2) { + int i, j; + + for (i = 0; i < 4; i++) { + for (j = 0; j < 3; j++) { + ASSERT(fabsf(m1[i][j] - m2[i][j]) <= 0.0000009) + } + } + + TEST_SUCCESS +} + +test_status_t +test_assert_eqf(float a, float b) { + ASSERT(fabsf(a - b) <= 0.000009); /* rounding errors */ + + TEST_SUCCESS +} + +test_status_t +test_assert_vec2_eq(vec2 v1, vec2 v2) { + ASSERT(fabsf(v1[0] - v2[0]) <= 0.000009); /* rounding errors */ + ASSERT(fabsf(v1[1] - v2[1]) <= 0.000009); + + TEST_SUCCESS +} + +test_status_t +test_assert_vec3_eq(vec3 v1, vec3 v2) { + ASSERT(fabsf(v1[0] - v2[0]) <= 0.000009); /* rounding errors */ + ASSERT(fabsf(v1[1] - v2[1]) <= 0.000009); + ASSERT(fabsf(v1[2] - v2[2]) <= 0.000009); + + TEST_SUCCESS +} + +test_status_t +test_assert_vec3s_eq(vec3s v1, vec3s v2) { + test_assert_vec3_eq(v1.raw, v2.raw); + + TEST_SUCCESS +} + +test_status_t +test_assert_vec4_eq(vec4 v1, vec4 v2) { + ASSERT(fabsf(v1[0] - v2[0]) <= 0.000009); /* rounding errors */ + ASSERT(fabsf(v1[1] - v2[1]) <= 0.000009); + ASSERT(fabsf(v1[2] - v2[2]) <= 0.000009); + ASSERT(fabsf(v1[3] - v2[3]) <= 0.000009); + + TEST_SUCCESS +} + +test_status_t +test_assert_vec4s_eq(vec4s v1, vec4s v2) { + test_assert_vec4_eq(v1.raw, v2.raw); + + TEST_SUCCESS +} + +test_status_t +test_assert_quat_eq_abs(versor v1, versor v2) { + ASSERT(fabsf(fabsf(v1[0]) - fabsf(v2[0])) <= 0.0009); /* rounding errors */ + ASSERT(fabsf(fabsf(v1[1]) - fabsf(v2[1])) <= 0.0009); + ASSERT(fabsf(fabsf(v1[2]) - fabsf(v2[2])) <= 0.0009); + ASSERT(fabsf(fabsf(v1[3]) - fabsf(v2[3])) <= 0.0009); + + TEST_SUCCESS +} + +test_status_t +test_assert_quat_eq(versor v1, versor v2) { + ASSERT(fabsf(v1[0] - v2[0]) <= 0.000009); /* rounding errors */ + ASSERT(fabsf(v1[1] - v2[1]) <= 0.000009); + ASSERT(fabsf(v1[2] - v2[2]) <= 0.000009); + ASSERT(fabsf(v1[3] - v2[3]) <= 0.000009); + + TEST_SUCCESS +} + +test_status_t +test_assert_quat_eq_identity(versor q) { + versor p = GLM_QUAT_IDENTITY_INIT; + + ASSERT(fabsf(q[0] - p[0]) <= 0.000009); /* rounding errors */ + ASSERT(fabsf(q[1] - p[1]) <= 0.000009); + ASSERT(fabsf(q[2] - p[2]) <= 0.000009); + ASSERT(fabsf(q[3] - p[3]) <= 0.000009); + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_common.h b/cglm/test/src/test_common.h new file mode 100644 index 0000000..4fff040 --- /dev/null +++ b/cglm/test/src/test_common.h @@ -0,0 +1,176 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef test_common_h +#define test_common_h + +#include "../include/common.h" + +#if !defined(_WIN32) && !defined(_MSC_VER) +# pragma GCC diagnostic ignored "-Wstrict-prototypes" +#endif + +void +test_rand_mat4(mat4 dest); + +void +test_rand_mat4x2(mat4x2 dest); + +void +test_rand_mat4x3(mat4x3 dest); + +void +test_rand_mat3(mat3 dest); + +void +test_rand_mat3x2(mat3x2 dest); + +void +test_rand_mat3x4(mat3x4 dest); + +void +test_rand_mat2(mat2 dest); + +void +test_rand_mat2x3(mat2x3 dest); + +void +test_rand_mat2x4(mat2x4 dest); + +test_status_t +test_assert_eqf(float a, float b); + +test_status_t +test_assert_mat4_eq(mat4 m1, mat4 m2); + +test_status_t +test_assert_mat4_eqt(mat4 m1, mat4 m2); + +test_status_t +test_assert_mat4_eq2(mat4 m1, mat4 m2, float eps); + +test_status_t +test_assert_mat4_eq_identity(mat4 m4); + +test_status_t +test_assert_mat4_eq_zero(mat4 m4); + +test_status_t +test_assert_mat4x2_eq_zero(mat4x2 m4x2); + +test_status_t +test_assert_mat4x2_eq(mat4x2 m1, mat4x2 m2); + +test_status_t +test_assert_mat4x3_eq_zero(mat4x3 m4x3); + +test_status_t +test_assert_mat4x3_eq(mat4x3 m1, mat4x3 m2); + +test_status_t +test_assert_mat2_eqt(mat2 m1, mat2 m2); + +test_status_t +test_assert_mat2_eq(mat2 m1, mat2 m2); + +test_status_t +test_assert_mat2_eq_identity(mat2 m2); + +test_status_t +test_assert_mat2_eq_zero(mat2 m2); + +test_status_t +test_assert_mat2x3_eq_zero(mat2x3 m2x3); + +test_status_t +test_assert_mat2x3_eq(mat2x3 m1, mat2x3 m2); + +test_status_t +test_assert_mat2x4_eq_zero(mat2x4 m2x4); + +test_status_t +test_assert_mat2x4_eq(mat2x4 m1, mat2x4 m2); + +test_status_t +test_assert_mat3_eq(mat3 m1, mat3 m2); + +test_status_t +test_assert_vec2_eq(vec2 v1, vec2 v2); + +test_status_t +test_assert_mat3_eqt(mat3 m1, mat3 m2); + +test_status_t +test_assert_mat3_eq_identity(mat3 m3); + +test_status_t +test_assert_mat3_eq_zero(mat3 m3); + +test_status_t +test_assert_mat3x2_eq_zero(mat3x2 m3x2); + +test_status_t +test_assert_mat3x2_eq(mat3x2 m1, mat3x2 m2); + +test_status_t +test_assert_mat3x4_eq_zero(mat3x4 m3x4); + +test_status_t +test_assert_mat3x4_eq(mat3x4 m1, mat3x4 m2); + +test_status_t +test_assert_vec3_eq(vec3 v1, vec3 v2); + +test_status_t +test_assert_vec3s_eq(vec3s v1, vec3s v2); + +test_status_t +test_assert_vec4_eq(vec4 v1, vec4 v2); + +test_status_t +test_assert_vec4s_eq(vec4s v1, vec4s v2); + +test_status_t +test_assert_quat_eq(versor v1, versor v2); + +test_status_t +test_assert_quat_eq_identity(versor q) ; + +test_status_t +test_assert_quat_eq_abs(versor v1, versor v2); + +void +test_rand_vec3(vec3 dest); + +vec3s +test_rand_vec3s(void); + +void +test_rand_vec4(vec4 dest); + +vec4s +test_rand_vec4s(void); + +float +test_rand(void); + +void +test_rand_quat(versor q); + +CGLM_INLINE +bool +test_eq(float a, float b) { + return fabsf(a - b) <= GLM_FLT_EPSILON * 10; +} + +CGLM_INLINE +bool +test_eq_th(float a, float b, float th) { + return fabsf(a - b) <= th; +} + +#endif /* test_common_h */ diff --git a/cglm/test/src/test_euler.c b/cglm/test/src/test_euler.c new file mode 100644 index 0000000..b71e0f0 --- /dev/null +++ b/cglm/test/src/test_euler.c @@ -0,0 +1,45 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +TEST_IMPL(euler) { + mat4 rot1, rot2; + vec3 inAngles, outAngles; + + inAngles[0] = glm_rad(-45.0f); /* X angle */ + inAngles[1] = glm_rad(88.0f); /* Y angle */ + inAngles[2] = glm_rad(18.0f); /* Z angle */ + + glm_euler_xyz(inAngles, rot1); + + /* extract angles */ + glmc_euler_angles(rot1, outAngles); + + /* angles must be equal in that range */ + ASSERTIFY(test_assert_vec3_eq(inAngles, outAngles)) + + /* matrices must be equal */ + glmc_euler_xyz(outAngles, rot2); + ASSERTIFY(test_assert_mat4_eq(rot1, rot2)) + + /* change range */ + inAngles[0] = glm_rad(-145.0f); /* X angle */ + inAngles[1] = glm_rad(818.0f); /* Y angle */ + inAngles[2] = glm_rad(181.0f); /* Z angle */ + + glm_euler_xyz(inAngles, rot1); + glmc_euler_angles(rot1, outAngles); + + /* angles may not be equal but matrices MUST! */ + + /* matrices must be equal */ + glmc_euler_xyz(outAngles, rot2); + ASSERTIFY(test_assert_mat4_eq(rot1, rot2)) + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_euler_to_quat_lh.h b/cglm/test/src/test_euler_to_quat_lh.h new file mode 100644 index 0000000..d4351b0 --- /dev/null +++ b/cglm/test/src/test_euler_to_quat_lh.h @@ -0,0 +1,493 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" +#include "../../include/cglm/handed/euler_to_quat_lh.h" + +TEST_IMPL(GLM_PREFIX, euler_xyz_quat_lh) { + vec3 axis_x = {1.0f, 0.0f, 0.0f}; + vec3 axis_y = {0.0f, 1.0f, 0.0f}; + vec3 axis_z = {0.0f, 0.0f,-1.0f}; + + /* random angles for testing */ + vec3 angles; + + /* quaternion representations for rotations */ + versor rot_x, rot_y, rot_z; + + versor expected; + versor result; + versor tmp; + + /* 100 randomized tests */ + for (int i = 0; i < 100; i++) { + test_rand_vec3(angles); + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in xyz order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + + glm_euler_xyz_quat_lh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + } + + + /* Start gimbal lock tests */ + for (float x = -90.0f; x <= 90.0f; x += 90.0f) { + for (float y = -90.0f; y <= 90.0f; y += 90.0f) { + for (float z = -90.0f; z <= 90.0f; z += 90.0f) { + angles[0] = x; + angles[1] = y; + angles[2] = z; + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in xyz order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + + /* use my function to get the quaternion */ + glm_euler_xyz_quat_lh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + } + } + } + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, euler_xzy_quat_lh) { + vec3 axis_x = {1.0f, 0.0f, 0.0f}; + vec3 axis_y = {0.0f, 1.0f, 0.0f}; + vec3 axis_z = {0.0f, 0.0f,-1.0f}; + + /* random angles for testing */ + vec3 angles; + + /* quaternion representations for rotations */ + versor rot_x, rot_y, rot_z; + + versor expected; + versor result; + versor tmp; + + /* 100 randomized tests */ + for (int i = 0; i < 100; i++) { + test_rand_vec3(angles); + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in xzy order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + + glm_euler_xzy_quat_lh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + } + + + /* Start gimbal lock tests */ + for (float x = -90.0f; x <= 90.0f; x += 90.0f) { + for (float y = -90.0f; y <= 90.0f; y += 90.0f) { + for (float z = -90.0f; z <= 90.0f; z += 90.0f) { + angles[0] = x; + angles[1] = y; + angles[2] = z; + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in xzy order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + + /* use my function to get the quaternion */ + glm_euler_xzy_quat_lh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + } + } + } + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, euler_yxz_quat_lh) { + vec3 axis_x = {1.0f, 0.0f, 0.0f}; + vec3 axis_y = {0.0f, 1.0f, 0.0f}; + vec3 axis_z = {0.0f, 0.0f,-1.0f}; + + /* random angles for testing */ + vec3 angles; + + /* quaternion representations for rotations */ + versor rot_x, rot_y, rot_z; + + versor expected; + versor result; + versor tmp; + + /* 100 randomized tests */ + for (int i = 0; i < 100; i++) { + test_rand_vec3(angles); + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in yxz order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + + glm_euler_yxz_quat_lh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + } + + + /* Start gimbal lock tests */ + for (float x = -90.0f; x <= 90.0f; x += 90.0f) { + for (float y = -90.0f; y <= 90.0f; y += 90.0f) { + for (float z = -90.0f; z <= 90.0f; z += 90.0f) { + angles[0] = x; + angles[1] = y; + angles[2] = z; + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in yxz order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + + /* use my function to get the quaternion */ + glm_euler_yxz_quat_lh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + ASSERTIFY(test_assert_quat_eq(result, expected)) + } + } + } + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, euler_yzx_quat_lh) { + vec3 axis_x = {1.0f, 0.0f, 0.0f}; + vec3 axis_y = {0.0f, 1.0f, 0.0f}; + vec3 axis_z = {0.0f, 0.0f,-1.0f}; + + /* random angles for testing */ + vec3 angles; + + /* quaternion representations for rotations */ + versor rot_x, rot_y, rot_z; + + versor expected; + versor result; + versor tmp; + + /* 100 randomized tests */ + for (int i = 0; i < 100; i++) { + test_rand_vec3(angles); + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in yzx order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + + glm_euler_yzx_quat_lh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + } + + + /* Start gimbal lock tests */ + for (float x = -90.0f; x <= 90.0f; x += 90.0f) { + for (float y = -90.0f; y <= 90.0f; y += 90.0f) { + for (float z = -90.0f; z <= 90.0f; z += 90.0f) { + angles[0] = x; + angles[1] = y; + angles[2] = z; + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in yzx order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + + /* use my function to get the quaternion */ + glm_euler_yzx_quat_lh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + ASSERTIFY(test_assert_quat_eq(result, expected)) + } + } + } + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, euler_zxy_quat_lh) { + vec3 axis_x = {1.0f, 0.0f, 0.0f}; + vec3 axis_y = {0.0f, 1.0f, 0.0f}; + vec3 axis_z = {0.0f, 0.0f,-1.0f}; + + /* random angles for testing */ + vec3 angles; + + /* quaternion representations for rotations */ + versor rot_x, rot_y, rot_z; + + versor expected; + versor result; + versor tmp; + + /* 100 randomized tests */ + for (int i = 0; i < 100; i++) { + test_rand_vec3(angles); + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in zxy order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + + glm_euler_zxy_quat_lh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + } + + + /* Start gimbal lock tests */ + for (float x = -90.0f; x <= 90.0f; x += 90.0f) { + for (float y = -90.0f; y <= 90.0f; y += 90.0f) { + for (float z = -90.0f; z <= 90.0f; z += 90.0f) { + angles[0] = x; + angles[1] = y; + angles[2] = z; + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in zxy order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + + /* use my function to get the quaternion */ + glm_euler_zxy_quat_lh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + } + } + } + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, euler_zyx_quat_lh) { + vec3 axis_x = {1.0f, 0.0f, 0.0f}; + vec3 axis_y = {0.0f, 1.0f, 0.0f}; + vec3 axis_z = {0.0f, 0.0f,-1.0f}; + + /* random angles for testing */ + vec3 angles; + + /* quaternion representations for rotations */ + versor rot_x, rot_y, rot_z; + + versor expected; + versor result; + versor tmp; + + /* 100 randomized tests */ + for (int i = 0; i < 100; i++) { + test_rand_vec3(angles); + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in zyx order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + + glm_euler_zyx_quat_lh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + } + + + /* Start gimbal lock tests */ + for (float x = -90.0f; x <= 90.0f; x += 90.0f) { + for (float y = -90.0f; y <= 90.0f; y += 90.0f) { + for (float z = -90.0f; z <= 90.0f; z += 90.0f) { + angles[0] = x; + angles[1] = y; + angles[2] = z; + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in xyz order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + + /* use my function to get the quaternion */ + glm_euler_zyx_quat_lh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + } + } + } + TEST_SUCCESS +} diff --git a/cglm/test/src/test_euler_to_quat_rh.h b/cglm/test/src/test_euler_to_quat_rh.h new file mode 100644 index 0000000..6140a6b --- /dev/null +++ b/cglm/test/src/test_euler_to_quat_rh.h @@ -0,0 +1,573 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" +#include "../../include/cglm/handed/euler_to_quat_rh.h" + +TEST_IMPL(GLM_PREFIX, euler_xyz_quat_rh) { + vec3 axis_x = {1.0f, 0.0f, 0.0f}; + vec3 axis_y = {0.0f, 1.0f, 0.0f}; + vec3 axis_z = {0.0f, 0.0f, 1.0f}; + + /* random angles for testing */ + vec3 angles; + + /* quaternion representations for rotations */ + versor rot_x, rot_y, rot_z; + + versor expected; + versor result; + versor tmp; + + mat4 expected_mat4; + + /* 100 randomized tests */ + for (int i = 0; i < 100; i++) { + test_rand_vec3(angles); + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in xyz order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + + glm_euler_xyz_quat_rh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + + /* verify that it acts the same as glm_euler_by_order */ + glm_euler_by_order(angles, GLM_EULER_XYZ, expected_mat4); + glm_mat4_quat(expected_mat4, expected); + + ASSERTIFY(test_assert_quat_eq_abs(result, expected)); + } + + + /* Start gimbal lock tests */ + for (float x = -90.0f; x <= 90.0f; x += 90.0f) { + for (float y = -90.0f; y <= 90.0f; y += 90.0f) { + for (float z = -90.0f; z <= 90.0f; z += 90.0f) { + angles[0] = x; + angles[1] = y; + angles[2] = z; + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in xyz order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + + /* use my function to get the quaternion */ + glm_euler_xyz_quat_rh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + ASSERTIFY(test_assert_quat_eq(result, expected)) + + /* verify that it acts the same as glm_euler_by_order */ + glm_euler_by_order(angles, GLM_EULER_XYZ, expected_mat4); + glm_mat4_quat(expected_mat4, expected); + + ASSERTIFY(test_assert_quat_eq_abs(result, expected)); + } + } + } + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, euler_xzy_quat_rh) { + vec3 axis_x = {1.0f, 0.0f, 0.0f}; + vec3 axis_y = {0.0f, 1.0f, 0.0f}; + vec3 axis_z = {0.0f, 0.0f, 1.0f}; + + /* random angles for testing */ + vec3 angles; + + /* quaternion representations for rotations */ + versor rot_x, rot_y, rot_z; + + versor expected; + versor result; + versor tmp; + + mat4 expected_mat4; + + /* 100 randomized tests */ + for (int i = 0; i < 100; i++) { + test_rand_vec3(angles); + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in xzy order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + + glm_euler_xzy_quat_rh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + + /* verify that it acts the same as glm_euler_by_order */ + glm_euler_by_order(angles, GLM_EULER_XZY, expected_mat4); + glm_mat4_quat(expected_mat4, expected); + + ASSERTIFY(test_assert_quat_eq_abs(result, expected)); + } + + + /* Start gimbal lock tests */ + for (float x = -90.0f; x <= 90.0f; x += 90.0f) { + for (float y = -90.0f; y <= 90.0f; y += 90.0f) { + for (float z = -90.0f; z <= 90.0f; z += 90.0f) { + angles[0] = x; + angles[1] = y; + angles[2] = z; + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in xzy order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + + /* use my function to get the quaternion */ + glm_euler_xzy_quat_rh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + ASSERTIFY(test_assert_quat_eq(result, expected)) + + /* verify that it acts the same as glm_euler_by_order */ + glm_euler_by_order(angles, GLM_EULER_XZY, expected_mat4); + glm_mat4_quat(expected_mat4, expected); + + ASSERTIFY(test_assert_quat_eq_abs(result, expected)); + } + } + } + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, euler_yxz_quat_rh) { + vec3 axis_x = {1.0f, 0.0f, 0.0f}; + vec3 axis_y = {0.0f, 1.0f, 0.0f}; + vec3 axis_z = {0.0f, 0.0f, 1.0f}; + + /* random angles for testing */ + vec3 angles; + + /* quaternion representations for rotations */ + versor rot_x, rot_y, rot_z; + + versor expected; + versor result; + versor tmp; + + mat4 expected_mat4; + + /* 100 randomized tests */ + for (int i = 0; i < 100; i++) { + test_rand_vec3(angles); + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in yxz order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + + glm_euler_yxz_quat_rh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + + /* verify that it acts the same as glm_euler_by_order */ + glm_euler_by_order(angles, GLM_EULER_YXZ, expected_mat4); + glm_mat4_quat(expected_mat4, expected); + + ASSERTIFY(test_assert_quat_eq_abs(result, expected)); + } + + + /* Start gimbal lock tests */ + for (float x = -90.0f; x <= 90.0f; x += 90.0f) { + for (float y = -90.0f; y <= 90.0f; y += 90.0f) { + for (float z = -90.0f; z <= 90.0f; z += 90.0f) { + angles[0] = x; + angles[1] = y; + angles[2] = z; + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in yxz order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + + /* use my function to get the quaternion */ + glm_euler_yxz_quat_rh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + ASSERTIFY(test_assert_quat_eq(result, expected)) + + /* verify that it acts the same as glm_euler_by_order */ + glm_euler_by_order(angles, GLM_EULER_YXZ, expected_mat4); + glm_mat4_quat(expected_mat4, expected); + + ASSERTIFY(test_assert_quat_eq_abs(result, expected)); + } + } + } + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, euler_yzx_quat_rh) { + vec3 axis_x = {1.0f, 0.0f, 0.0f}; + vec3 axis_y = {0.0f, 1.0f, 0.0f}; + vec3 axis_z = {0.0f, 0.0f, 1.0f}; + + /* random angles for testing */ + vec3 angles; + + /* quaternion representations for rotations */ + versor rot_x, rot_y, rot_z; + + versor expected; + versor result; + versor tmp; + + mat4 expected_mat4; + + /* 100 randomized tests */ + for (int i = 0; i < 100; i++) { + test_rand_vec3(angles); + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in yzx order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + + glm_euler_yzx_quat_rh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + + /* verify that it acts the same as glm_euler_by_order */ + glm_euler_by_order(angles, GLM_EULER_YZX, expected_mat4); + glm_mat4_quat(expected_mat4, expected); + + ASSERTIFY(test_assert_quat_eq_abs(result, expected)); + } + + + /* Start gimbal lock tests */ + for (float x = -90.0f; x <= 90.0f; x += 90.0f) { + for (float y = -90.0f; y <= 90.0f; y += 90.0f) { + for (float z = -90.0f; z <= 90.0f; z += 90.0f) { + angles[0] = x; + angles[1] = y; + angles[2] = z; + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in yzx order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + + /* use my function to get the quaternion */ + glm_euler_yzx_quat_rh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + ASSERTIFY(test_assert_quat_eq(result, expected)) + + /* verify that it acts the same as glm_euler_by_order */ + glm_euler_by_order(angles, GLM_EULER_YZX, expected_mat4); + glm_mat4_quat(expected_mat4, expected); + + ASSERTIFY(test_assert_quat_eq_abs(result, expected)); + } + } + } + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, euler_zxy_quat_rh) { + vec3 axis_x = {1.0f, 0.0f, 0.0f}; + vec3 axis_y = {0.0f, 1.0f, 0.0f}; + vec3 axis_z = {0.0f, 0.0f, 1.0f}; + + /* random angles for testing */ + vec3 angles; + + /* quaternion representations for rotations */ + versor rot_x, rot_y, rot_z; + + versor expected; + versor result; + versor tmp; + + mat4 expected_mat4; + + /* 100 randomized tests */ + for (int i = 0; i < 100; i++) { + test_rand_vec3(angles); + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in zxy order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + + glm_euler_zxy_quat_rh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + + /* verify that it acts the same as glm_euler_by_order */ + glm_euler_by_order(angles, GLM_EULER_ZXY, expected_mat4); + glm_mat4_quat(expected_mat4, expected); + + ASSERTIFY(test_assert_quat_eq_abs(result, expected)); + } + + + /* Start gimbal lock tests */ + for (float x = -90.0f; x <= 90.0f; x += 90.0f) { + for (float y = -90.0f; y <= 90.0f; y += 90.0f) { + for (float z = -90.0f; z <= 90.0f; z += 90.0f) { + angles[0] = x; + angles[1] = y; + angles[2] = z; + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in zxy order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + + /* use my function to get the quaternion */ + glm_euler_zxy_quat_rh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + ASSERTIFY(test_assert_quat_eq(result, expected)) + + /* verify that it acts the same as glm_euler_by_order */ + glm_euler_by_order(angles, GLM_EULER_ZXY, expected_mat4); + glm_mat4_quat(expected_mat4, expected); + + ASSERTIFY(test_assert_quat_eq_abs(result, expected)); + } + } + } + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, euler_zyx_quat_rh) { + vec3 axis_x = {1.0f, 0.0f, 0.0f}; + vec3 axis_y = {0.0f, 1.0f, 0.0f}; + vec3 axis_z = {0.0f, 0.0f, 1.0f}; + + /* random angles for testing */ + vec3 angles; + + /* quaternion representations for rotations */ + versor rot_x, rot_y, rot_z; + + versor expected; + versor result; + versor tmp; + + mat4 expected_mat4; + + /* 100 randomized tests */ + for (int i = 0; i < 100; i++) { + test_rand_vec3(angles); + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in zyx order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + + glm_euler_zyx_quat_rh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + /* verify that it acts the same as rotating by 3 axis quaternions */ + ASSERTIFY(test_assert_quat_eq(result, expected)) + + /* verify that it acts the same as glm_euler_by_order */ + glm_euler_by_order(angles, GLM_EULER_ZYX, expected_mat4); + glm_mat4_quat(expected_mat4, expected); + + ASSERTIFY(test_assert_quat_eq_abs(result, expected)); + } + + + /* Start gimbal lock tests */ + for (float x = -90.0f; x <= 90.0f; x += 90.0f) { + for (float y = -90.0f; y <= 90.0f; y += 90.0f) { + for (float z = -90.0f; z <= 90.0f; z += 90.0f) { + angles[0] = x; + angles[1] = y; + angles[2] = z; + + /* create the rotation quaternions using the angles and axises */ + glm_quatv(rot_x, angles[0], axis_x); + glm_quatv(rot_y, angles[1], axis_y); + glm_quatv(rot_z, angles[2], axis_z); + + /* apply the rotations to a unit quaternion in xyz order */ + glm_quat_identity(expected); + + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_z, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_y, expected); + glm_quat_copy(expected, tmp); + glm_quat_mul(tmp, rot_x, expected); + + /* use my function to get the quaternion */ + glm_euler_zyx_quat_rh(angles, result); + + /* verify if the magnitude of the quaternion stays 1 */ + ASSERT(test_eq(glm_quat_norm(result), 1.0f)) + + ASSERTIFY(test_assert_quat_eq(result, expected)) + + /* verify that it acts the same as glm_euler_by_order */ + glm_euler_by_order(angles, GLM_EULER_ZYX, expected_mat4); + glm_mat4_quat(expected_mat4, expected); + + ASSERTIFY(test_assert_quat_eq_abs(result, expected)); + } + } + } + TEST_SUCCESS +} diff --git a/cglm/test/src/test_ivec2.h b/cglm/test/src/test_ivec2.h new file mode 100644 index 0000000..9e32d71 --- /dev/null +++ b/cglm/test/src/test_ivec2.h @@ -0,0 +1,536 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +TEST_IMPL(GLM_PREFIX, ivec2) { + ivec4 v4 = {2, 3, 5, 7}; + ivec3 v3 = {11, 13, 17}; + ivec2 v2; + + GLM(ivec2)(v4, v2); + ASSERT(v2[0] == 2) + ASSERT(v2[1] == 3) + + GLM(ivec2)(v3, v2); + ASSERT(v2[0] == 11) + ASSERT(v2[1] == 13) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_copy) { + ivec2 src = {7, 5}; + ivec2 dst = {99, 99}; + + GLM(ivec2_copy)(src, dst); + ASSERT(dst[0] == 7) + ASSERT(dst[1] == 5) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_zero) { + ivec2 v = {2, 3}; + + GLM(ivec2_zero)(v); + ASSERT(v[0] == 0) + ASSERT(v[1] == 0) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_one) { + ivec2 v = {-2, 9}; + + GLM(ivec2_one)(v); + ASSERT(v[0] == 1) + ASSERT(v[1] == 1) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_dot) { + ivec2 a = {2, 3}; + ivec2 b = {4, 4}; + int dot1, dot2; + + dot1 = GLM(ivec2_dot)(a, b); + dot2 = a[0] * b[0] + a[1] * b[1]; + + ASSERT(dot1 == dot2) + ASSERT(dot1 == 20) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_cross) { + ivec2 a = {10, 9}; + ivec2 b = {1, 2}; + int cross1, cross2; + + cross1 = GLM(ivec2_cross)(a, b); + cross2 = a[0] * b[1] - a[1] * b[0]; + + ASSERT(cross1 == cross2) + ASSERT(cross1 == 11) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_add) { + ivec2 a = {14, 3}; + ivec2 b = {-3, 2}; + ivec2 v = {99, 99}; + + GLM(ivec2_add)(a, b, v); + ASSERT(v[0] == 11) + ASSERT(v[1] == 5) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_adds) { + ivec2 a = {-3, 1}; + ivec2 v = {99, 99}; + int s = 2; + + GLM(ivec2_adds)(a, s, v); + ASSERT(v[0] == -1) + ASSERT(v[1] == 3) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_sub) { + ivec2 a = {-2, 9}; + ivec2 b = {3, 2}; + ivec2 v = {99, 99}; + + GLM(ivec2_sub)(a, b, v); + ASSERT(v[0] == -5) + ASSERT(v[1] == 7) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_subs) { + ivec2 a = {5, -2}; + ivec2 v = {99, 99}; + int s = -3; + + GLM(ivec2_subs)(a, s, v); + ASSERT(v[0] == 8) + ASSERT(v[1] == 1) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_mul) { + ivec2 a = {3, 4}; + ivec2 b = {-2, 3}; + ivec2 v = {99, 99}; + + GLM(ivec2_mul)(a, b, v); + ASSERT(v[0] == -6) + ASSERT(v[1] == 12) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_scale) { + ivec2 a = {-9, 2}; + ivec2 v = {99, 99}; + int s = -2; + + GLM(ivec2_scale)(a, s, v); + ASSERT(v[0] == 18) + ASSERT(v[1] == -4) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_div) { + ivec2 v1 = {6, 5}, + v2 = {-2, 4}, + v3; + + GLM(ivec2_div)(v1, v2, v3); + + ASSERT(v1[0] / v2[0] == v3[0]) + ASSERT(v1[1] / v2[1] == v3[1]) + ASSERT(v3[0] == -3) + ASSERT(v3[1] == 1) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_divs) { + ivec2 v1 = {16, -8}, v2; + int s = 4; + + GLM(ivec2_divs)(v1, s, v2); + + ASSERT(v1[0] / s == v2[0]) + ASSERT(v1[1] / s == v2[1]) + ASSERT(v2[0] == 4) + ASSERT(v2[1] == -2) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_mod) { + ivec2 a = {16, -8}; + ivec2 b = {3, 5}; + ivec2 dest; + + GLM(ivec2_mod)(a, b, dest); + + ASSERT(a[0] % b[0] == dest[0]) + ASSERT(a[1] % b[1] == dest[1]) + ASSERT(dest[0] == 1) + ASSERT(dest[1] == -3) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_addadd) { + ivec2 a = {2, -3}, + b = {-3, 4}, + c = {1, 2}, + d = {1, 2}; + + GLM(ivec2_addadd)(a, b, c); + + ASSERT(d[0] + a[0] + b[0] == c[0]) + ASSERT(d[1] + a[1] + b[1] == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_addadds) { + ivec2 a = {2, -3}, + c = {1, 2}, + d = {1, 2}; + int s = 9; + + GLM(ivec2_addadds)(a, s, c); + + ASSERT(d[0] + a[0] + s == c[0]) + ASSERT(d[1] + a[1] + s == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_subadd) { + ivec2 a = {2, -3}, + b = {-3, 4}, + c = {1, 2}, + d = {1, 2}; + + GLM(ivec2_subadd)(a, b, c); + + ASSERT(d[0] + (a[0] - b[0]) == c[0]) + ASSERT(d[1] + (a[1] - b[1]) == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_subadds) { + ivec2 a = {2, -3}, + c = {1, 2}, + d = {1, 2}; + int s = 9; + + GLM(ivec2_subadds)(a, s, c); + + ASSERT(d[0] + a[0] - s == c[0]) + ASSERT(d[1] + a[1] - s == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_muladd) { + ivec2 a = {2, -3}, + b = {-3, 4}, + c = {1, 2}, + d = {1, 2}; + + GLM(ivec2_muladd)(a, b, c); + + ASSERT(d[0] + a[0] * b[0] == c[0]) + ASSERT(d[1] + a[1] * b[1] == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_muladds) { + ivec2 a = {2, -3}, + c = {1, 2}, + d = {1, 2}; + int s = 9; + + GLM(ivec2_muladds)(a, s, c); + + ASSERT(d[0] + a[0] * s == c[0]) + ASSERT(d[1] + a[1] * s == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_maxadd) { + ivec2 a = {2, -3}, + b = {-3, 4}, + c = {1, 2}, + d = {1, 2}; + + GLM(ivec2_maxadd)(a, b, c); + + ASSERT(d[0] += glm_imax(a[0], b[0]) == c[0]) + ASSERT(d[1] += glm_imax(a[1], b[1]) == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_minadd) { + ivec2 a = {2, -3}, + b = {-3, 4}, + c = {1, 2}, + d = {1, 2}; + + GLM(ivec2_minadd)(a, b, c); + + ASSERT(d[0] += glm_imin(a[0], b[0]) == c[0]) + ASSERT(d[1] += glm_imin(a[1], b[1]) == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_subsub) { + ivec2 a = {2, -3}, + b = {-3, 4}, + c = {1, 2}, + d = {1, 2}; + + GLM(ivec2_subsub)(a, b, c); + + ASSERT(d[0] - (a[0] - b[0]) == c[0]) + ASSERT(d[1] - (a[1] - b[1]) == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_subsubs) { + ivec2 a = {2, -3}, + c = {1, 2}, + d = {1, 2}; + int s = 9; + + GLM(ivec2_subsubs)(a, s, c); + + ASSERT(d[0] - (a[0] - s) == c[0]) + ASSERT(d[1] - (a[1] - s) == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_addsub) { + ivec2 a = {2, -3}, + b = {-3, 4}, + c = {1, 2}, + d = {1, 2}; + + GLM(ivec2_addsub)(a, b, c); + + ASSERT(d[0] - (a[0] + b[0]) == c[0]) + ASSERT(d[1] - (a[1] + b[1]) == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_addsubs) { + ivec2 a = {2, -3}, + c = {1, 2}, + d = {1, 2}; + int s = 9; + + GLM(ivec2_addsubs)(a, s, c); + + ASSERT(d[0] - (a[0] + s) == c[0]) + ASSERT(d[1] - (a[1] + s) == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_mulsub) { + ivec2 a = {2, -3}, + b = {-3, 4}, + c = {1, 2}, + d = {1, 2}; + + GLM(ivec2_mulsub)(a, b, c); + + ASSERT(d[0] - a[0] * b[0] == c[0]) + ASSERT(d[1] - a[1] * b[1] == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_mulsubs) { + ivec2 a = {2, -3}, + c = {1, 2}, + d = {1, 2}; + int s = 9; + + GLM(ivec2_mulsubs)(a, s, c); + + ASSERT(d[0] - a[0] * s == c[0]) + ASSERT(d[1] - a[1] * s == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_maxsub) { + ivec2 a = {2, -3}, + b = {-3, 4}, + c = {1, 2}, + d = {1, 2}; + + GLM(ivec2_maxsub)(a, b, c); + + ASSERT(d[0] -= glm_imax(a[0], b[0]) == c[0]) + ASSERT(d[1] -= glm_imax(a[1], b[1]) == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_minsub) { + ivec2 a = {2, -3}, + b = {-3, 4}, + c = {1, 2}, + d = {1, 2}; + + GLM(ivec2_minsub)(a, b, c); + + ASSERT(d[0] -= glm_imin(a[0], b[0]) == c[0]) + ASSERT(d[1] -= glm_imin(a[1], b[1]) == c[1]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_distance2) { + ivec2 a = {-1, 3}; + ivec2 b = {5, 4}; + int v; + + v = GLM(ivec2_distance2)(a, b); + ASSERT(v == 37) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_distance) { + ivec2 a = {3, 2}; + ivec2 b = {-2, 5}; + float v; + + v = GLM(ivec2_distance)(a, b); + ASSERT(test_eq(v, 5.8309518948f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_fill) { + ivec2 v1; + ivec2 v2 = {-1, 3}; + + GLM(ivec2_fill)(v1, 1); + GLM(ivec2_fill)(v2, 2); + + ASSERT(GLM(ivec2_eq)(v1, 1)) + ASSERT(GLM(ivec2_eq)(v2, 2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_eq) { + ivec2 v1 = {-1, 2}; + + GLM(ivec2_fill)(v1, 2); + + ASSERT(GLM(ivec2_eq)(v1, 2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_eqv) { + ivec2 v1, v2, v3; + + GLM(ivec2_fill)(v1, 1); + GLM(ivec2_fill)(v2, 2); + GLM(ivec2_fill)(v3, 1); + + ASSERT(GLM(ivec2_eqv)(v1, v3)) + ASSERT(!GLM(ivec2_eqv)(v1, v2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_maxv) { + ivec2 a = {9, -20}; + ivec2 b = {8, -1}; + ivec2 v = {99, 99}; + + GLM(ivec2_maxv)(a, b, v); + ASSERT(v[0] == 9) + ASSERT(v[1] == -1) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_minv) { + ivec2 a = {16, 0}; + ivec2 b = {-15, 10}; + ivec2 v = {99, 99}; + + GLM(ivec2_minv)(a, b, v); + ASSERT(v[0] == -15) + ASSERT(v[1] == 0) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_clamp) { + ivec2 v = {3, -1}; + + GLM(ivec2_clamp)(v, -2, 4); + ASSERT(v[0] == 3) + ASSERT(v[1] == -1) + + v[0] = -15; + v[1] = 4; + GLM(ivec2_clamp)(v, -9, 3); + ASSERT(v[0] == -9) + ASSERT(v[1] == 3) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec2_abs) { + ivec2 v1 = {2, -3}, v2 = {-12, -31}; + ivec2 v3, v4; + + GLM(ivec2_abs)(v1, v3); + GLM(ivec2_abs)(v2, v4); + + ASSERT(v3[0] == 2) + ASSERT(v3[1] == 3) + ASSERT(v4[0] == 12) + ASSERT(v4[1] == 31) + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_ivec3.h b/cglm/test/src/test_ivec3.h new file mode 100644 index 0000000..aa62771 --- /dev/null +++ b/cglm/test/src/test_ivec3.h @@ -0,0 +1,582 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +TEST_IMPL(GLM_PREFIX, ivec3) { + ivec4 v4 = {2, 3, 5, 7}; + ivec3 v3 = {99, 99, 99}; + + GLM(ivec3)(v4, v3); + ASSERT(v3[0] == 2) + ASSERT(v3[1] == 3) + ASSERT(v3[2] == 5) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_copy) { + ivec3 src = {7, 5, 2}; + ivec3 dst = {99, 99, 99}; + + GLM(ivec3_copy)(src, dst); + ASSERT(dst[0] == 7) + ASSERT(dst[1] == 5) + ASSERT(dst[2] == 2) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_zero) { + ivec3 v = {2, 3, 5}; + + GLM(ivec3_zero)(v); + ASSERT(v[0] == 0) + ASSERT(v[1] == 0) + ASSERT(v[2] == 0) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_one) { + ivec3 v = {-2, 9, 12}; + + GLM(ivec3_one)(v); + ASSERT(v[0] == 1) + ASSERT(v[1] == 1) + ASSERT(v[2] == 1) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_dot) { + ivec3 a = {2, 3, 1}; + ivec3 b = {4, 4, 2}; + int dot1, dot2; + + dot1 = GLM(ivec3_dot)(a, b); + dot2 = a[0] * b[0] + a[1] * b[1] + a[2] * b[2]; + + ASSERT(dot1 == dot2) + ASSERT(dot1 == 22) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_norm2) { + ivec3 v = {2, 3, 4}; + int norm2_1, norm2_2; + + norm2_1 = GLM(ivec3_norm2)(v); + norm2_2 = v[0] * v[0] + v[1] * v[1] + v[2] * v[2]; + + ASSERT(norm2_1 == norm2_2) + ASSERT(norm2_1 == 29) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_norm) { + ivec3 v = {2, 3, 4}; + int norm1, norm2; + + norm1 = GLM(ivec3_norm)(v); + norm2 = (int)sqrtf((float)(v[0] * v[0] + v[1] * v[1] + v[2] * v[2])); + + ASSERT(norm1 == norm2) + ASSERT(norm1 == 5) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_add) { + ivec3 a = {14, 3, 2}; + ivec3 b = {-3, 2, 1}; + ivec3 v = {99, 99, 99}; + + GLM(ivec3_add)(a, b, v); + ASSERT(v[0] == 11) + ASSERT(v[1] == 5) + ASSERT(v[2] == 3) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_adds) { + ivec3 a = {-3, 1, 4}; + ivec3 v = {99, 99, 99}; + int s = 2; + + GLM(ivec3_adds)(a, s, v); + ASSERT(v[0] == -1) + ASSERT(v[1] == 3) + ASSERT(v[2] == 6) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_sub) { + ivec3 a = {-2, 9, 1}; + ivec3 b = {3, 2, -1}; + ivec3 v = {99, 99, 99}; + + GLM(ivec3_sub)(a, b, v); + ASSERT(v[0] == -5) + ASSERT(v[1] == 7) + ASSERT(v[2] == 2) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_subs) { + ivec3 a = {5, -2, 6}; + ivec3 v = {99, 99, 99}; + int s = -3; + + GLM(ivec3_subs)(a, s, v); + ASSERT(v[0] == 8) + ASSERT(v[1] == 1) + ASSERT(v[2] == 9) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_mul) { + ivec3 a = {3, 4, 5}; + ivec3 b = {-2, 3, 1}; + ivec3 v = {99, 99, 99}; + + GLM(ivec3_mul)(a, b, v); + ASSERT(v[0] == -6) + ASSERT(v[1] == 12) + ASSERT(v[2] == 5) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_scale) { + ivec3 a = {-9, 2, 3}; + ivec3 v = {99, 99, 99}; + int s = -2; + + GLM(ivec3_scale)(a, s, v); + ASSERT(v[0] == 18) + ASSERT(v[1] == -4) + ASSERT(v[2] == -6) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_div) { + ivec3 v1 = {6, 5, 8}, + v2 = {-2, 4, 2}, + v3; + + GLM(ivec3_div)(v1, v2, v3); + + ASSERT(v1[0] / v2[0] == v3[0]) + ASSERT(v1[1] / v2[1] == v3[1]) + ASSERT(v1[2] / v2[2] == v3[2]) + ASSERT(v3[0] == -3) + ASSERT(v3[1] == 1) + ASSERT(v3[2] == 4) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_divs) { + ivec3 v1 = {16, -8, 12}, v2; + int s = 4; + + GLM(ivec3_divs)(v1, s, v2); + + ASSERT(v1[0] / s == v2[0]) + ASSERT(v1[1] / s == v2[1]) + ASSERT(v1[2] / s == v2[2]) + ASSERT(v2[0] == 4) + ASSERT(v2[1] == -2) + ASSERT(v2[2] == 3) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_mod) { + ivec3 a = {16, -8, 10}; + ivec3 b = {3, 5, 4}; + ivec3 dest; + + GLM(ivec3_mod)(a, b, dest); + + ASSERT(a[0] % b[0] == dest[0]) + ASSERT(a[1] % b[1] == dest[1]) + ASSERT(a[2] % b[2] == dest[2]) + ASSERT(dest[0] == 1) + ASSERT(dest[1] == -3) + ASSERT(dest[2] == 2) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_addadd) { + ivec3 a = {2, -3, 8}, + b = {-3, 4, 5}, + c = {1, 2, 6}, + d = {1, 2, 6}; + + GLM(ivec3_addadd)(a, b, c); + + ASSERT(d[0] + a[0] + b[0] == c[0]) + ASSERT(d[1] + a[1] + b[1] == c[1]) + ASSERT(d[2] + a[2] + b[2] == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_addadds) { + ivec3 a = {2, -3, 5}, + c = {1, 2, -9}, + d = {1, 2, -9}; + int s = 9; + + GLM(ivec3_addadds)(a, s, c); + + ASSERT(d[0] + a[0] + s == c[0]) + ASSERT(d[1] + a[1] + s == c[1]) + ASSERT(d[2] + a[2] + s == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_subadd) { + ivec3 a = {2, -3, 8}, + b = {-3, 4, 5}, + c = {1, 2, 6}, + d = {1, 2, 6}; + + GLM(ivec3_subadd)(a, b, c); + + ASSERT(d[0] + a[0] - b[0] == c[0]) + ASSERT(d[1] + a[1] - b[1] == c[1]) + ASSERT(d[2] + a[2] - b[2] == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_subadds) { + ivec3 a = {2, -3, 5}, + c = {1, 2, -9}, + d = {1, 2, -9}; + int s = 9; + + GLM(ivec3_subadds)(a, s, c); + + ASSERT(d[0] + a[0] - s == c[0]) + ASSERT(d[1] + a[1] - s == c[1]) + ASSERT(d[2] + a[2] - s == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_muladd) { + ivec3 a = {2, -3, 8}, + b = {-3, 4, 5}, + c = {1, 2, 6}, + d = {1, 2, 6}; + + GLM(ivec3_muladd)(a, b, c); + + ASSERT(d[0] + a[0] * b[0] == c[0]) + ASSERT(d[1] + a[1] * b[1] == c[1]) + ASSERT(d[2] + a[2] * b[2] == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_muladds) { + ivec3 a = {2, -3, 5}, + c = {1, 2, -9}, + d = {1, 2, -9}; + int s = 9; + + GLM(ivec3_muladds)(a, s, c); + + ASSERT(d[0] + a[0] * s == c[0]) + ASSERT(d[1] + a[1] * s == c[1]) + ASSERT(d[2] + a[2] * s == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_maxadd) { + ivec3 a = {2, -3, 8}, + b = {-3, 4, 5}, + c = {1, 2, 6}, + d = {1, 2, 6}; + + GLM(ivec3_maxadd)(a, b, c); + + ASSERT(d[0] += glm_imax(a[0], b[0]) == c[0]) + ASSERT(d[1] += glm_imax(a[1], b[1]) == c[1]) + ASSERT(d[2] += glm_imax(a[2], b[2]) == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_minadd) { + ivec3 a = {2, -3, 8}, + b = {-3, 4, 5}, + c = {1, 2, 6}, + d = {1, 2, 6}; + + GLM(ivec3_minadd)(a, b, c); + + ASSERT(d[0] += glm_imin(a[0], b[0]) == c[0]) + ASSERT(d[1] += glm_imin(a[1], b[1]) == c[1]) + ASSERT(d[2] += glm_imin(a[2], b[2]) == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_subsub) { + ivec3 a = {2, -3, 8}, + b = {-3, 4, 5}, + c = {1, 2, 6}, + d = {1, 2, 6}; + + GLM(ivec3_subsub)(a, b, c); + + ASSERT(d[0] - (a[0] - b[0]) == c[0]) + ASSERT(d[1] - (a[1] - b[1]) == c[1]) + ASSERT(d[2] - (a[2] - b[2]) == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_subsubs) { + ivec3 a = {2, -3, 5}, + c = {1, 2, -9}, + d = {1, 2, -9}; + int s = 9; + + GLM(ivec3_subsubs)(a, s, c); + + ASSERT(d[0] - (a[0] - s) == c[0]) + ASSERT(d[1] - (a[1] - s) == c[1]) + ASSERT(d[2] - (a[2] - s) == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_addsub) { + ivec3 a = {2, -3, 8}, + b = {-3, 4, 5}, + c = {1, 2, 6}, + d = {1, 2, 6}; + + GLM(ivec3_addsub)(a, b, c); + + ASSERT(d[0] - (a[0] + b[0]) == c[0]) + ASSERT(d[1] - (a[1] + b[1]) == c[1]) + ASSERT(d[2] - (a[2] + b[2]) == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_addsubs) { + ivec3 a = {2, -3, 5}, + c = {1, 2, -9}, + d = {1, 2, -9}; + int s = 9; + + GLM(ivec3_addsubs)(a, s, c); + + ASSERT(d[0] - (a[0] + s) == c[0]) + ASSERT(d[1] - (a[1] + s) == c[1]) + ASSERT(d[2] - (a[2] + s) == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_mulsub) { + ivec3 a = {2, -3, 8}, + b = {-3, 4, 5}, + c = {1, 2, 6}, + d = {1, 2, 6}; + + GLM(ivec3_mulsub)(a, b, c); + + ASSERT(d[0] - a[0] * b[0] == c[0]) + ASSERT(d[1] - a[1] * b[1] == c[1]) + ASSERT(d[2] - a[2] * b[2] == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_mulsubs) { + ivec3 a = {2, -3, 5}, + c = {1, 2, -9}, + d = {1, 2, -9}; + int s = 9; + + GLM(ivec3_mulsubs)(a, s, c); + + ASSERT(d[0] - a[0] * s == c[0]) + ASSERT(d[1] - a[1] * s == c[1]) + ASSERT(d[2] - a[2] * s == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_maxsub) { + ivec3 a = {2, -3, 8}, + b = {-3, 4, 5}, + c = {1, 2, 6}, + d = {1, 2, 6}; + + GLM(ivec3_maxsub)(a, b, c); + + ASSERT(d[0] -= glm_imax(a[0], b[0]) == c[0]) + ASSERT(d[1] -= glm_imax(a[1], b[1]) == c[1]) + ASSERT(d[2] -= glm_imax(a[2], b[2]) == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_minsub) { + ivec3 a = {2, -3, 8}, + b = {-3, 4, 5}, + c = {1, 2, 6}, + d = {1, 2, 6}; + + GLM(ivec3_minsub)(a, b, c); + + ASSERT(d[0] -= glm_imin(a[0], b[0]) == c[0]) + ASSERT(d[1] -= glm_imin(a[1], b[1]) == c[1]) + ASSERT(d[2] -= glm_imin(a[2], b[2]) == c[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_distance2) { + ivec3 a = {-1, 3, 0}; + ivec3 b = {5, 4, 2}; + int v; + + v = GLM(ivec3_distance2)(a, b); + ASSERT(v == 41) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_distance) { + ivec3 a = {3, 2, 4}; + ivec3 b = {-2, 5, 2}; + float v; + + v = GLM(ivec3_distance)(a, b); + ASSERT(test_eq(v, 6.1644140029f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_fill) { + ivec3 v1; + ivec3 v2 = {-1, 3, 4}; + + GLM(ivec3_fill)(v1, 1); + GLM(ivec3_fill)(v2, 2); + + ASSERT(GLM(ivec3_eq)(v1, 1)) + ASSERT(GLM(ivec3_eq)(v2, 2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_eq) { + ivec3 v1 = { -1, 2, 4 }; + + GLM(ivec3_fill)(v1, 2); + + ASSERT(GLM(ivec3_eq)(v1, 2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_eqv) { + ivec3 v1, v2, v3; + + GLM(ivec3_fill)(v1, 1); + GLM(ivec3_fill)(v2, 2); + GLM(ivec3_fill)(v3, 1); + + ASSERT(GLM(ivec3_eqv)(v1, v3)) + ASSERT(!GLM(ivec3_eqv)(v1, v2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_maxv) { + ivec3 a = {9, -20, 5}; + ivec3 b = {8, -1, 2}; + ivec3 v = {99, 99, 99}; + + GLM(ivec3_maxv)(a, b, v); + ASSERT(v[0] == 9) + ASSERT(v[1] == -1) + ASSERT(v[2] == 5) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_minv) { + ivec3 a = {16, 0, 4}; + ivec3 b = {-15, 10, 8}; + ivec3 v = {99, 99, 99}; + + GLM(ivec3_minv)(a, b, v); + ASSERT(v[0] == -15) + ASSERT(v[1] == 0) + ASSERT(v[2] == 4) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_clamp) { + ivec3 v = {3, -1, 10}; + + GLM(ivec3_clamp)(v, -2, 4); + ASSERT(v[0] == 3) + ASSERT(v[1] == -1) + ASSERT(v[2] == 4) + + v[0] = -15; + v[1] = 4; + v[2] = 1; + GLM(ivec3_clamp)(v, -9, 3); + ASSERT(v[0] == -9) + ASSERT(v[1] == 3) + ASSERT(v[2] == 1) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec3_abs) { + ivec3 v1 = {2, -3, 4}, v2 = {-12, -31, -42}; + ivec3 v3, v4; + + GLM(ivec3_abs)(v1, v3); + GLM(ivec3_abs)(v2, v4); + + ASSERT(v3[0] == 2) + ASSERT(v3[1] == 3) + ASSERT(v3[2] == 4) + ASSERT(v4[0] == 12) + ASSERT(v4[1] == 31) + ASSERT(v4[2] == 42) + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_ivec4.h b/cglm/test/src/test_ivec4.h new file mode 100644 index 0000000..77a4036 --- /dev/null +++ b/cglm/test/src/test_ivec4.h @@ -0,0 +1,488 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +TEST_IMPL(GLM_PREFIX, ivec4) { + ivec4 v3 = {2, 3, 5}; + ivec4 v4; + + GLM(ivec4)(v3, 7, v4); + ASSERT(v4[0] == 2) + ASSERT(v4[1] == 3) + ASSERT(v4[2] == 5) + ASSERT(v4[3] == 7) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_copy) { + ivec4 src = {7, 5, 2, 11}; + ivec4 dst = {99, 99, 99, 99}; + + GLM(ivec4_copy)(src, dst); + ASSERT(dst[0] == 7) + ASSERT(dst[1] == 5) + ASSERT(dst[2] == 2) + ASSERT(dst[3] == 11) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_zero) { + ivec4 v = {2, 3, 5, 7}; + + GLM(ivec4_zero)(v); + ASSERT(v[0] == 0) + ASSERT(v[1] == 0) + ASSERT(v[2] == 0) + ASSERT(v[3] == 0) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_one) { + ivec4 v = {-2, 9, 12, 7}; + + GLM(ivec4_one)(v); + ASSERT(v[0] == 1) + ASSERT(v[1] == 1) + ASSERT(v[2] == 1) + ASSERT(v[3] == 1) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_add) { + ivec4 a = {14, 3, 2, -4}; + ivec4 b = {-3, 2, 1, -1}; + ivec4 v = {99, 99, 99, 99}; + + GLM(ivec4_add)(a, b, v); + ASSERT(v[0] == 11) + ASSERT(v[1] == 5) + ASSERT(v[2] == 3) + ASSERT(v[3] == -5) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_adds) { + ivec4 a = {-3, 1, 4, 2}; + ivec4 v = {99, 99, 99, 99}; + int s = -2; + + GLM(ivec4_adds)(a, s, v); + ASSERT(v[0] == -5) + ASSERT(v[1] == -1) + ASSERT(v[2] == 2) + ASSERT(v[3] == 0) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_sub) { + ivec4 a = {-2, 9, 1, 5}; + ivec4 b = {3, 2, -1, 2}; + ivec4 v = {99, 99, 99, 99}; + + GLM(ivec4_sub)(a, b, v); + ASSERT(v[0] == -5) + ASSERT(v[1] == 7) + ASSERT(v[2] == 2) + ASSERT(v[3] == 3) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_subs) { + ivec4 a = {5, -2, 6, 1}; + ivec4 v = {99, 99, 99, 99}; + int s = 2; + + GLM(ivec4_subs)(a, s, v); + ASSERT(v[0] == 3) + ASSERT(v[1] == -4) + ASSERT(v[2] == 4) + ASSERT(v[3] == -1) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_mul) { + ivec4 a = {3, 4, 5, -3}; + ivec4 b = {-2, 3, 1, 2}; + ivec4 v = {99, 99, 99, 99}; + + GLM(ivec4_mul)(a, b, v); + ASSERT(v[0] == -6) + ASSERT(v[1] == 12) + ASSERT(v[2] == 5) + ASSERT(v[3] == -6) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_scale) { + ivec4 a = {-9, 2, 3, 0}; + ivec4 v = {99, 99, 99, 99}; + int s = -2; + + GLM(ivec4_scale)(a, s, v); + ASSERT(v[0] == 18) + ASSERT(v[1] == -4) + ASSERT(v[2] == -6) + ASSERT(v[3] == 0) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_addadd) { + ivec4 a = {2, -3, 8, -7}, + b = {-3, 4, 5, -2}, + c = {1, 2, 6, 15}, + d = {1, 2, 6, 15}; + + GLM(ivec4_addadd)(a, b, c); + + ASSERT(d[0] + a[0] + b[0] == c[0]) + ASSERT(d[1] + a[1] + b[1] == c[1]) + ASSERT(d[2] + a[2] + b[2] == c[2]) + ASSERT(d[3] + a[3] + b[3] == c[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_addadds) { + ivec4 a = {2, -3, 4, 7}, + c = {1, 2, -8, 5}, + d = {1, 2, -8, 5}; + int s = 9; + + GLM(ivec4_addadds)(a, s, c); + + ASSERT(d[0] + a[0] + s == c[0]) + ASSERT(d[1] + a[1] + s == c[1]) + ASSERT(d[2] + a[2] + s == c[2]) + ASSERT(d[3] + a[3] + s == c[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_subadd) { + ivec4 a = {2, -3, 8, -7}, + b = {-3, 4, 5, -2}, + c = {1, 2, 6, 15}, + d = {1, 2, 6, 15}; + + GLM(ivec4_subadd)(a, b, c); + + ASSERT(d[0] + a[0] - b[0] == c[0]) + ASSERT(d[1] + a[1] - b[1] == c[1]) + ASSERT(d[2] + a[2] - b[2] == c[2]) + ASSERT(d[3] + a[3] - b[3] == c[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_subadds) { + ivec4 a = {2, -3, 4, 7}, + c = {1, 2, -8, 5}, + d = {1, 2, -8, 5}; + int s = 9; + + GLM(ivec4_subadds)(a, s, c); + + ASSERT(d[0] + a[0] - s == c[0]) + ASSERT(d[1] + a[1] - s == c[1]) + ASSERT(d[2] + a[2] - s == c[2]) + ASSERT(d[3] + a[3] - s == c[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_muladd) { + ivec4 a = {2, -3, 8, -7}, + b = {-3, 4, 5, -2}, + c = {1, 2, 6, 15}, + d = {1, 2, 6, 15}; + + GLM(ivec4_muladd)(a, b, c); + + ASSERT(d[0] + a[0] * b[0] == c[0]) + ASSERT(d[1] + a[1] * b[1] == c[1]) + ASSERT(d[2] + a[2] * b[2] == c[2]) + ASSERT(d[3] + a[3] * b[3] == c[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_muladds) { + ivec4 a = {2, -3, 4, 7}, + c = {1, 2, -8, 5}, + d = {1, 2, -8, 5}; + int s = 9; + + GLM(ivec4_muladds)(a, s, c); + + ASSERT(d[0] + a[0] * s == c[0]) + ASSERT(d[1] + a[1] * s == c[1]) + ASSERT(d[2] + a[2] * s == c[2]) + ASSERT(d[3] + a[3] * s == c[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_maxadd) { + ivec4 a = {2, -3, 8, -7}, + b = {-3, 4, 5, -2}, + c = {1, 2, 6, 15}, + d = {1, 2, 6, 15}; + + GLM(ivec4_maxadd)(a, b, c); + + ASSERT(d[0] += glm_imax(a[0], b[0]) == c[0]) + ASSERT(d[1] += glm_imax(a[1], b[1]) == c[1]) + ASSERT(d[2] += glm_imax(a[2], b[2]) == c[2]) + ASSERT(d[3] += glm_imax(a[3], b[3]) == c[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_minadd) { + ivec4 a = {2, -3, 8, -7}, + b = {-3, 4, 5, -2}, + c = {1, 2, 6, 15}, + d = {1, 2, 6, 15}; + + GLM(ivec4_minadd)(a, b, c); + + ASSERT(d[0] += glm_imin(a[0], b[0]) == c[0]) + ASSERT(d[1] += glm_imin(a[1], b[1]) == c[1]) + ASSERT(d[2] += glm_imin(a[2], b[2]) == c[2]) + ASSERT(d[3] += glm_imin(a[3], b[3]) == c[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_subsub) { + ivec4 a = {2, -3, 8, -7}, + b = {-3, 4, 5, -2}, + c = {1, 2, 6, 15}, + d = {1, 2, 6, 15}; + + GLM(ivec4_subsub)(a, b, c); + + ASSERT(d[0] - (a[0] - b[0]) == c[0]) + ASSERT(d[1] - (a[1] - b[1]) == c[1]) + ASSERT(d[2] - (a[2] - b[2]) == c[2]) + ASSERT(d[3] - (a[3] - b[3]) == c[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_subsubs) { + ivec4 a = {2, -3, 4, 7}, + c = {1, 2, -8, 5}, + d = {1, 2, -8, 5}; + int s = 9; + + GLM(ivec4_subsubs)(a, s, c); + + ASSERT(d[0] - (a[0] - s) == c[0]) + ASSERT(d[1] - (a[1] - s) == c[1]) + ASSERT(d[2] - (a[2] - s) == c[2]) + ASSERT(d[3] - (a[3] - s) == c[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_addsub) { + ivec4 a = {2, -3, 8, -7}, + b = {-3, 4, 5, -2}, + c = {1, 2, 6, 15}, + d = {1, 2, 6, 15}; + + GLM(ivec4_addsub)(a, b, c); + + ASSERT(d[0] - (a[0] + b[0]) == c[0]) + ASSERT(d[1] - (a[1] + b[1]) == c[1]) + ASSERT(d[2] - (a[2] + b[2]) == c[2]) + ASSERT(d[3] - (a[3] + b[3]) == c[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_addsubs) { + ivec4 a = {2, -3, 4, 7}, + c = {1, 2, -8, 5}, + d = {1, 2, -8, 5}; + int s = 9; + + GLM(ivec4_addsubs)(a, s, c); + + ASSERT(d[0] - (a[0] + s) == c[0]) + ASSERT(d[1] - (a[1] + s) == c[1]) + ASSERT(d[2] - (a[2] + s) == c[2]) + ASSERT(d[3] - (a[3] + s) == c[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_mulsub) { + ivec4 a = {2, -3, 8, -7}, + b = {-3, 4, 5, -2}, + c = {1, 2, 6, 15}, + d = {1, 2, 6, 15}; + + GLM(ivec4_mulsub)(a, b, c); + + ASSERT(d[0] - a[0] * b[0] == c[0]) + ASSERT(d[1] - a[1] * b[1] == c[1]) + ASSERT(d[2] - a[2] * b[2] == c[2]) + ASSERT(d[3] - a[3] * b[3] == c[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_mulsubs) { + ivec4 a = {2, -3, 4, 7}, + c = {1, 2, -8, 5}, + d = {1, 2, -8, 5}; + int s = 9; + + GLM(ivec4_mulsubs)(a, s, c); + + ASSERT(d[0] - a[0] * s == c[0]) + ASSERT(d[1] - a[1] * s == c[1]) + ASSERT(d[2] - a[2] * s == c[2]) + ASSERT(d[3] - a[3] * s == c[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_maxsub) { + ivec4 a = {2, -3, 8, -7}, + b = {-3, 4, 5, -2}, + c = {1, 2, 6, 15}, + d = {1, 2, 6, 15}; + + GLM(ivec4_maxsub)(a, b, c); + + ASSERT(d[0] -= glm_imax(a[0], b[0]) == c[0]) + ASSERT(d[1] -= glm_imax(a[1], b[1]) == c[1]) + ASSERT(d[2] -= glm_imax(a[2], b[2]) == c[2]) + ASSERT(d[3] -= glm_imax(a[3], b[3]) == c[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_minsub) { + ivec4 a = {2, -3, 8, -7}, + b = {-3, 4, 5, -2}, + c = {1, 2, 6, 15}, + d = {1, 2, 6, 15}; + + GLM(ivec4_minsub)(a, b, c); + + ASSERT(d[0] -= glm_imin(a[0], b[0]) == c[0]) + ASSERT(d[1] -= glm_imin(a[1], b[1]) == c[1]) + ASSERT(d[2] -= glm_imin(a[2], b[2]) == c[2]) + ASSERT(d[3] -= glm_imin(a[3], b[3]) == c[3]) + + TEST_SUCCESS +} +TEST_IMPL(GLM_PREFIX, ivec4_distance2) { + ivec4 a = {-1, 3, 0, 4}; + ivec4 b = {5, 4, 2, 6}; + int v; + + v = GLM(ivec4_distance2)(a, b); + ASSERT(v == 45) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_distance) { + ivec4 a = {3, 2, 4, -1}; + ivec4 b = {-2, 5, 2, 4}; + float v; + + v = GLM(ivec4_distance)(a, b); + ASSERT(test_eq(v, 7.9372539331f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_maxv) { + ivec4 a = {9, -20, 5, -3}; + ivec4 b = {8, -1, 2, 2}; + ivec4 v = {99, 99, 99, 99}; + + GLM(ivec4_maxv)(a, b, v); + ASSERT(v[0] == 9) + ASSERT(v[1] == -1) + ASSERT(v[2] == 5) + ASSERT(v[3] == 2) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_minv) { + ivec4 a = {16, 0, 4, 5}; + ivec4 b = {-15, 10, 8, 2}; + ivec4 v = {99, 99, 99, 99}; + + GLM(ivec4_minv)(a, b, v); + ASSERT(v[0] == -15) + ASSERT(v[1] == 0) + ASSERT(v[2] == 4) + ASSERT(v[3] == 2) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_clamp) { + ivec4 v = {3, -1, 10, -100}; + + GLM(ivec4_clamp)(v, -2, 4); + ASSERT(v[0] == 3) + ASSERT(v[1] == -1) + ASSERT(v[2] == 4) + ASSERT(v[3] == -2) + + v[0] = -15; + v[1] = 4; + v[2] = 1; + v[3] = 0; + GLM(ivec4_clamp)(v, -9, 3); + ASSERT(v[0] == -9) + ASSERT(v[1] == 3) + ASSERT(v[2] == 1) + ASSERT(v[3] == 0) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ivec4_abs) { + ivec4 v1 = {2, -3, 4, -5}, v2 = {-12, -31, -42, -50}; + ivec4 v3, v4; + + GLM(ivec4_abs)(v1, v3); + GLM(ivec4_abs)(v2, v4); + + ASSERT(v3[0] == 2) + ASSERT(v3[1] == 3) + ASSERT(v3[2] == 4) + ASSERT(v3[3] == 5) + ASSERT(v4[0] == 12) + ASSERT(v4[1] == 31) + ASSERT(v4[2] == 42) + ASSERT(v4[3] == 50) + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_mat2.h b/cglm/test/src/test_mat2.h new file mode 100644 index 0000000..6fa622a --- /dev/null +++ b/cglm/test/src/test_mat2.h @@ -0,0 +1,302 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +#define A_MATRIX2x2 {{1,2},{5,6}} +#define MAT2_ARRAY {1, 5, 2, 7} + +#ifndef CGLM_TEST_MAT2_ONCE +#define CGLM_TEST_MAT2_ONCE + +TEST_IMPL(MACRO_GLM_MAT2_IDENTITY_INIT) { + mat2 m = GLM_MAT2_IDENTITY_INIT; + + ASSERT(test_eq(m[0][0], 1.0f)) + ASSERT(test_eq(m[0][1], 0.0f)) + ASSERT(test_eq(m[1][0], 0.0f)) + ASSERT(test_eq(m[1][1], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_MAT2_ZERO_INIT) { + mat2 m = GLM_MAT2_ZERO_INIT; + + ASSERT(test_eq(m[0][0], 0.0f)) + ASSERT(test_eq(m[0][1], 0.0f)) + ASSERT(test_eq(m[1][0], 0.0f)) + ASSERT(test_eq(m[1][1], 0.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_MAT2_IDENTITY) { + ASSERT(test_eq(GLM_MAT2_IDENTITY[0][0], 1.0f)) + ASSERT(test_eq(GLM_MAT2_IDENTITY[0][1], 0.0f)) + ASSERT(test_eq(GLM_MAT2_IDENTITY[1][0], 0.0f)) + ASSERT(test_eq(GLM_MAT2_IDENTITY[1][1], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_MAT2_ZERO) { + ASSERT(test_eq(GLM_MAT2_ZERO[0][0], 0.0f)) + ASSERT(test_eq(GLM_MAT2_ZERO[0][1], 0.0f)) + ASSERT(test_eq(GLM_MAT2_ZERO[1][0], 0.0f)) + ASSERT(test_eq(GLM_MAT2_ZERO[1][1], 0.0f)) + + TEST_SUCCESS +} + +#endif /* CGLM_TEST_MAT2_ONCE */ + +TEST_IMPL(GLM_PREFIX, mat2_copy) { + mat2 m1 = A_MATRIX2x2; + mat2 m2 = GLM_MAT2_IDENTITY_INIT; + + GLM(mat2_copy)(m1, m2); + + test_assert_mat2_eq(m1, m2); + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2_identity) { + mat2 m1 = GLM_MAT2_IDENTITY_INIT; + mat2 m2 = GLM_MAT2_IDENTITY_INIT; + mat2 m3; + + GLM(mat2_identity)(m3); + + ASSERTIFY(test_assert_mat2_eq_identity(m1)) + ASSERTIFY(test_assert_mat2_eq_identity(m2)) + ASSERTIFY(test_assert_mat2_eq_identity(m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2_identity_array) { + size_t i, count; + mat2 matrices[4] = { + A_MATRIX2x2, + A_MATRIX2x2, + A_MATRIX2x2, + A_MATRIX2x2 + }; + + count = 4; + + GLM(mat2_identity_array)(matrices, count); + + for (i = 0; i < count; i++) { + ASSERTIFY(test_assert_mat2_eq_identity(matrices[i])) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2_zero) { + mat2 m1 = GLM_MAT2_ZERO_INIT; + mat2 m2 = GLM_MAT2_ZERO_INIT; + mat2 m3; + + GLM(mat2_zero)(m3); + + ASSERTIFY(test_assert_mat2_eq_zero(m1)) + ASSERTIFY(test_assert_mat2_eq_zero(m2)) + ASSERTIFY(test_assert_mat2_eq_zero(m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2_mul) { + mat2 m1 = GLM_MAT2_IDENTITY_INIT; + mat2 m2 = GLM_MAT2_IDENTITY_INIT; + mat2 m3; + mat2 m4 = GLM_MAT2_ZERO_INIT; + int i, j, k; + + /* test random matrices */ + /* random matrices */ + test_rand_mat2(m1); + test_rand_mat2(m2); + + GLM(mat2_mul)(m1, m2, m3); + for (i = 0; i < 2; i++) { + for (j = 0; j < 2; j++) { + for (k = 0; k < 2; k++) + /* column-major */ + m4[i][j] += m1[k][j] * m2[i][k]; + } + } + + ASSERTIFY(test_assert_mat2_eq(m3, m4)) + + /* test pre compiled */ + GLM(mat2_mul)(m1, m2, m3); + ASSERTIFY(test_assert_mat2_eq(m3, m4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2_transpose_to) { + mat2 mat = A_MATRIX2x2; + mat2 m1; + + GLM(mat2_transpose_to)(mat, m1); + + ASSERTIFY(test_assert_mat2_eqt(mat, m1)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2_transpose) { + mat2 mat = A_MATRIX2x2; + mat2 m1; + + GLM(mat2_copy)(mat, m1); + GLM(mat2_transpose)(m1); + + ASSERTIFY(test_assert_mat2_eqt(mat, m1)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2_mulv) { + vec2 res; + mat2 mat = A_MATRIX2x2; + vec2 v = {11.0f, 21.0f}; + int i; + + GLM(mat2_mulv)(mat, v, res); + + for (i = 0; i < 2; i++) { + ASSERT(test_eq(res[i], v[0] * mat[0][i] + v[1] * mat[1][i])) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2_trace) { + mat2 mat = A_MATRIX2x2; + float trace; + + trace = GLM(mat2_trace)(mat); + + ASSERT(test_eq(trace, mat[0][0] + mat[1][1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2_scale) { + mat2 m1 = A_MATRIX2x2; + mat2 m2 = A_MATRIX2x2; + int i, j, scale; + + scale = rand() % 100; + + GLM(mat2_scale)(m1, (float)scale); + + for (i = 0; i < 2; i++) { + for (j = 0; j < 2; j++) { + ASSERT(test_eq(m1[i][j], m2[i][j] * scale)) + } + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2_det) { + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2_inv) { + mat2 m1 = GLM_MAT2_IDENTITY_INIT; + mat2 m2 = GLM_MAT2_IDENTITY_INIT; + mat2 m3; + int i; + + m1[0][0] = 41.0f; + m1[0][1] = 0.0f; + m1[1][0] = 0.0f; + m1[1][1] = 70.0f; + + for (i = 0; i < 10000; i++) { + /* test inverse precise */ + GLM(mat2_inv)(m1, m2); + GLM(mat2_inv)(m2, m3); + + ASSERTIFY(test_assert_mat2_eq(m1, m3)) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2_swap_col) { + mat2 m1 = A_MATRIX2x2; + mat2 m2 = A_MATRIX2x2; + + GLM(mat2_swap_col)(m1, 0, 1); + + ASSERTIFY(test_assert_vec2_eq(m1[0], m2[1])) + ASSERTIFY(test_assert_vec2_eq(m1[1], m2[0])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2_swap_row) { + mat2 m1 = A_MATRIX2x2; + mat2 m2 = A_MATRIX2x2; + + GLM(mat2_swap_row)(m1, 0, 1); + + ASSERT(test_eq(m1[0][0], m2[0][1])) + ASSERT(test_eq(m1[0][1], m2[0][0])) + + ASSERT(test_eq(m1[1][0], m2[1][1])) + ASSERT(test_eq(m1[1][1], m2[1][0])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2_rmc) { + mat2 mat = A_MATRIX2x2; + vec2 v = {11.0f, 12.0f}; + vec2 v1; + float r1, r2; + int i; + + r1 = GLM(mat2_rmc)(v, mat, v); + + for (i = 0; i < 2; i++) { + v1[i] = v[0] * mat[i][0] + v[1] * mat[i][1]; + } + + r2 = v[0] * v1[0] + v[1] * v1[1]; + + ASSERT(test_eq(r1, r2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2_make) { + mat2 dest; + unsigned int i, j; + float src[4] = MAT2_ARRAY; + + GLM(mat2_make)(src, dest); + + for (i = 0, j = 0; i < sizeof(src) / sizeof(float); i+=2, j++) { + ASSERT(test_eq(dest[j][0], src[i])) + ASSERT(test_eq(dest[j][1], src[i+1])) + } + + TEST_SUCCESS +} + +#undef A_MATRIX2x2 diff --git a/cglm/test/src/test_mat2x3.h b/cglm/test/src/test_mat2x3.h new file mode 100644 index 0000000..8eb8144 --- /dev/null +++ b/cglm/test/src/test_mat2x3.h @@ -0,0 +1,156 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +#define A_MATRIX2X3 {{1,2,3},{5,6,7}} +#define A_MATRIX2X3_TRANSPOSE {{1,5}, {2,6}, {3,7}} + +#ifndef CGLM_TEST_MAT2X3_ONCE +#define CGLM_TEST_MAT2X3_ONCE + +TEST_IMPL(MACRO_GLM_MAT2X3_ZERO_INIT) { + mat2x3 mat2x3_zero = GLM_MAT2X3_ZERO_INIT; + test_assert_mat2x3_eq_zero(mat2x3_zero); + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_MAT2X3_ZERO) { + mat2x3 mat2x3_zero = GLM_MAT2X3_ZERO; + test_assert_mat2x3_eq_zero(mat2x3_zero); + TEST_SUCCESS +} + +#endif /* CGLM_TEST_MAT2X3_ONCE */ + +TEST_IMPL(GLM_PREFIX, mat2x3_copy) { + mat2x3 m1 = A_MATRIX2X3; + mat2x3 m2 = GLM_MAT2X3_ZERO_INIT; + + GLM(mat2x3_copy)(m1, m2); + + test_assert_mat2x3_eq(m1, m2); + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2x3_zero) { + mat2x3 m1 = GLM_MAT2X3_ZERO_INIT; + mat2x3 m2 = GLM_MAT2X3_ZERO_INIT; + mat2x3 m3; + + GLM(mat2x3_zero)(m3); + + ASSERTIFY(test_assert_mat2x3_eq_zero(m1)) + ASSERTIFY(test_assert_mat2x3_eq_zero(m2)) + ASSERTIFY(test_assert_mat2x3_eq_zero(m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2x3_make) { + float src[18] = { + 0.5f, 1.7f, 10.3f, 4.2f, 8.9f, 1.1f, + 2.3f, 4.2f, 66.5f, 23.7f, 6.6f, 8.9f, + 5.3f, 4.8f, 96.3f, 13.7f, 4.7f, 5.5f + }; + + mat2x3 dest[3]; + + float *srcp = src; + unsigned int i, j, k; + + for (i = 0, j = 0, k = 0; i < sizeof(src) / sizeof(float); i+=6,j++) { + GLM(mat2x3_make)(srcp + i, dest[j]); + + ASSERT(test_eq(src[ i ], dest[j][k][0])); + ASSERT(test_eq(src[i+1], dest[j][k][1])); + ASSERT(test_eq(src[i+2], dest[j][k][2])); + + ASSERT(test_eq(src[i+3], dest[j][k+1][0])); + ASSERT(test_eq(src[i+4], dest[j][k+1][1])); + ASSERT(test_eq(src[i+5], dest[j][k+1][2])); + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2x3_mul) { + mat2x3 m1 = GLM_MAT2X3_ZERO_INIT; + mat3x2 m2 = GLM_MAT3X2_ZERO_INIT; + + mat3 m3 = GLM_MAT3_ZERO_INIT; + mat3 m4 = GLM_MAT3_ZERO_INIT; + + int c, r, k; + + /* test random matrices */ + /* random matrices */ + test_rand_mat2x3(m1); + test_rand_mat3x2(m2); + + for (r = 0; r < 3; r++) { + for (c = 0; c < 3; c++) { + for (k = 0; k < 2; k++) { + m4[c][r] += m1[k][r] * m2[c][k]; + } + } + } + + GLM(mat2x3_mul)(m1, m2, m3); + ASSERTIFY(test_assert_mat3_eq(m3, m4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2x3_mulv) { + mat2x3 mat = A_MATRIX2X3; + vec2 v = {11.0f, 21.0f}; + + int i; + vec3 dest; + float res = 0.0; + + GLM(mat2x3_mulv)(mat, v, dest); + + for (i = 0; i < 3; i++) { + res = mat[0][i] * v[0] + mat[1][i] * v[1]; + ASSERT(test_eq(dest[i], res)) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2x3_transpose) { + mat2x3 m1 = A_MATRIX2X3; + + mat3x2 m2; + mat3x2 m3 = A_MATRIX2X3_TRANSPOSE; + GLM(mat2x3_transpose)(m1, m2); + + ASSERTIFY(test_assert_mat3x2_eq(m2, m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2x3_scale) { + mat2x3 m1 = A_MATRIX2X3; + mat2x3 m2 = A_MATRIX2X3; + int i, j, scale; + + scale = rand() % 100; + + GLM(mat2x3_scale)(m1, (float) scale); + + for (i = 0; i < 2; i++) { + for (j = 0; j < 3; j++) { + ASSERT(test_eq(m1[i][j], m2[i][j] * scale)) + } + } + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_mat2x4.h b/cglm/test/src/test_mat2x4.h new file mode 100644 index 0000000..17ee35f --- /dev/null +++ b/cglm/test/src/test_mat2x4.h @@ -0,0 +1,159 @@ + +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +#define A_MATRIX2X4 {{1,2,3,4},{5,6,7,8}} +#define A_MATRIX2X4_TRANSPOSE {{1,5}, {2,6}, {3,7}, {4,8}} + +#ifndef CGLM_TEST_MAT2X4_ONCE +#define CGLM_TEST_MAT2X4_ONCE + +TEST_IMPL(MACRO_GLM_MAT2X4_ZERO_INIT) { + mat2x4 mat2x4_zero = GLM_MAT2X4_ZERO_INIT; + test_assert_mat2x4_eq_zero(mat2x4_zero); + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_MAT2X4_ZERO) { + mat2x4 mat2x4_zero = GLM_MAT2X4_ZERO; + test_assert_mat2x4_eq_zero(mat2x4_zero); + TEST_SUCCESS +} + +#endif /* CGLM_TEST_MAT2X4_ONCE */ + +TEST_IMPL(GLM_PREFIX, mat2x4_copy) { + mat2x4 m1 = A_MATRIX2X4; + mat2x4 m2 = GLM_MAT2X4_ZERO_INIT; + + GLM(mat2x4_copy)(m1, m2); + + test_assert_mat2x4_eq(m1, m2); + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2x4_zero) { + mat2x4 m1 = GLM_MAT2X4_ZERO_INIT; + mat2x4 m2 = GLM_MAT2X4_ZERO_INIT; + mat2x4 m3; + + GLM(mat2x4_zero)(m3); + + ASSERTIFY(test_assert_mat2x4_eq_zero(m1)) + ASSERTIFY(test_assert_mat2x4_eq_zero(m2)) + ASSERTIFY(test_assert_mat2x4_eq_zero(m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2x4_make) { + float src[24] = { + 0.5f, 1.7f, 10.3f, 4.2f, 8.9f, 1.1f, 77.3f, 88.4f, + 2.3f, 4.2f, 66.5f, 23.7f, 6.6f, 8.9f, 90.3f, 34.2f, + 5.3f, 4.8f, 96.3f, 13.7f, 4.7f, 5.5f, 22.9f, 5.5f + }; + + mat2x4 dest[3]; + + float *srcp = src; + unsigned int i, j, k; + + for (i = 0, j = 0, k = 0; i < sizeof(src) / sizeof(float); i+=8,j++) { + GLM(mat2x4_make)(srcp + i, dest[j]); + + ASSERT(test_eq(src[ i ], dest[j][k][0])); + ASSERT(test_eq(src[i+1], dest[j][k][1])); + ASSERT(test_eq(src[i+2], dest[j][k][2])); + ASSERT(test_eq(src[i+3], dest[j][k][3])); + + ASSERT(test_eq(src[i+4], dest[j][k+1][0])); + ASSERT(test_eq(src[i+5], dest[j][k+1][1])); + ASSERT(test_eq(src[i+6], dest[j][k+1][2])); + ASSERT(test_eq(src[i+7], dest[j][k+1][3])); + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2x4_mul) { + mat2x4 m1 = GLM_MAT2X4_ZERO_INIT; + mat4x2 m2 = GLM_MAT4X2_ZERO_INIT; + + mat4 m3 = GLM_MAT4_ZERO_INIT; + mat4 m4 = GLM_MAT4_ZERO_INIT; + + int c, r, k; + + /* test random matrices */ + /* random matrices */ + test_rand_mat2x4(m1); + test_rand_mat4x2(m2); + + for (r = 0; r < 4; r++) { + for (c = 0; c < 4; c++) { + for (k = 0; k < 2; k++) { + m4[c][r] += m1[k][r] * m2[c][k]; + } + } + } + + GLM(mat2x4_mul)(m1, m2, m3); + ASSERTIFY(test_assert_mat4_eq(m3, m4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2x4_mulv) { + mat2x4 mat = A_MATRIX2X4; + vec2 v = {11.0f, 21.0f}; + + int i; + vec4 dest; + float res = 0.0; + + GLM(mat2x4_mulv)(mat, v, dest); + + for (i = 0; i < 4; i++) { + res = mat[0][i] * v[0] + mat[1][i] * v[1]; + ASSERT(test_eq(dest[i], res)) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2x4_transpose) { + mat2x4 m1 = A_MATRIX2X4; + + mat4x2 m2; + mat4x2 m3 = A_MATRIX2X4_TRANSPOSE; + GLM(mat2x4_transpose)(m1, m2); + + ASSERTIFY(test_assert_mat4x2_eq(m2, m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat2x4_scale) { + mat2x4 m1 = A_MATRIX2X4; + mat2x4 m2 = A_MATRIX2X4; + int i, j, scale; + + scale = rand() % 100; + + GLM(mat2x4_scale)(m1, (float) scale); + + for (i = 0; i < 2; i++) { + for (j = 0; j < 4; j++) { + ASSERT(test_eq(m1[i][j], m2[i][j] * scale)) + } + } + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_mat3.h b/cglm/test/src/test_mat3.h new file mode 100644 index 0000000..2f151cd --- /dev/null +++ b/cglm/test/src/test_mat3.h @@ -0,0 +1,329 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +#define A_MATRIX {{1,2,3},{5,6,7},{9,10,11}} +#define MAT3_ARRAY {1, 5, 2, 7, 12, 1, 4, 6, 0} + +TEST_IMPL(GLM_PREFIX, mat3_copy) { + mat3 m1 = A_MATRIX; + mat3 m2 = GLM_MAT3_IDENTITY_INIT; + + GLM(mat3_copy)(m1, m2); + + test_assert_mat3_eq(m1, m2); + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3_identity) { + mat3 m1 = GLM_MAT3_IDENTITY_INIT; + mat3 m2 = GLM_MAT3_IDENTITY_INIT; + mat3 m3; + + GLM(mat3_identity)(m3); + + ASSERTIFY(test_assert_mat3_eq_identity(m1)) + ASSERTIFY(test_assert_mat3_eq_identity(m2)) + ASSERTIFY(test_assert_mat3_eq_identity(m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3_identity_array) { + size_t i, count; + mat3 matrices[4] = { + A_MATRIX, + A_MATRIX, + A_MATRIX, + A_MATRIX + }; + + count = 4; + + GLM(mat3_identity_array)(matrices, count); + + for (i = 0; i < count; i++) { + ASSERTIFY(test_assert_mat3_eq_identity(matrices[i])) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3_zero) { + mat3 m1 = GLM_MAT3_ZERO_INIT; + mat3 m2 = GLM_MAT3_ZERO_INIT; + mat3 m3; + + GLM(mat3_zero)(m3); + + ASSERTIFY(test_assert_mat3_eq_zero(m1)) + ASSERTIFY(test_assert_mat3_eq_zero(m2)) + ASSERTIFY(test_assert_mat3_eq_zero(m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3_mul) { + mat3 m1 = GLM_MAT3_IDENTITY_INIT; + mat3 m2 = GLM_MAT3_IDENTITY_INIT; + mat3 m3; + mat3 m4 = GLM_MAT3_ZERO_INIT; + int i, j, k; + + /* test random matrices */ + /* random matrices */ + test_rand_mat3(m1); + test_rand_mat3(m2); + + GLM(mat3_mul)(m1, m2, m3); + for (i = 0; i < 3; i++) { + for (j = 0; j < 3; j++) { + for (k = 0; k < 3; k++) + /* column-major */ + m4[i][j] += m1[k][j] * m2[i][k]; + } + } + + ASSERTIFY(test_assert_mat3_eq(m3, m4)) + + /* test pre compiled */ + GLM(mat3_mul)(m1, m2, m3); + ASSERTIFY(test_assert_mat3_eq(m3, m4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3_mulv) { + vec4 res; + mat3 mat = A_MATRIX; + vec4 v = {1.0f, 2.0f, 3.0f, 4.0f}; + int i; + + GLM(mat3_mulv)(mat, v, res); + + for (i = 0; i < 3; i++) { + ASSERT(test_eq(res[i], + v[0] * mat[0][i] + + v[1] * mat[1][i] + + v[2] * mat[2][i])) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3_trace) { + mat3 mat = A_MATRIX; + float trace; + + trace = GLM(mat3_trace)(mat); + + ASSERT(test_eq(trace, mat[0][0] + mat[1][1] + mat[2][2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3_quat) { + mat3 m1, m3; + mat4 m2; + versor q1, q2, q3; + vec3 axis1; + vec3 axis2 = {1.9f, 2.3f, 4.5f}; + + GLM(quat)(q1, GLM_PI_4f, 1.9f, 2.3f, 4.5f); + GLM(quat_mat3)(q1, m1); + GLM(mat3_quat)(m1, q2); + + GLM(rotate_make)(m2, GLM_PI_4f, axis2); + GLM(mat3_quat)(m1, q3); + + GLM(quat_axis)(q3, axis1); + + GLM(vec3_normalize)(axis1); + GLM(vec3_normalize)(axis2); + + GLM(mat4_pick3)(m2, m3); + + ASSERT(test_eq(glm_quat_angle(q3), GLM_PI_4f)) + ASSERTIFY(test_assert_vec3_eq(axis1, axis2)) + ASSERTIFY(test_assert_vec4_eq(q1, q2)) + ASSERTIFY(test_assert_mat3_eq(m1, m3)) + ASSERTIFY(test_assert_vec4_eq(q1, q3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3_transpose_to) { + mat3 mat = A_MATRIX; + mat3 m1; + + GLM(mat3_transpose_to)(mat, m1); + + ASSERTIFY(test_assert_mat3_eqt(mat, m1)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3_transpose) { + mat3 mat = A_MATRIX; + mat3 m1; + + GLM(mat3_copy)(mat, m1); + GLM(mat3_transpose)(m1); + + ASSERTIFY(test_assert_mat3_eqt(mat, m1)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3_scale) { + mat3 m1 = A_MATRIX; + mat3 m2 = A_MATRIX; + int i, j, k, scale; + + scale = rand() % 100; + + GLM(mat3_scale)(m1, (float)scale); + + for (i = 0; i < 3; i++) { + for (j = 0; j < 3; j++) { + for (k = 0; k < 3; k++) + ASSERT(test_eq(m1[i][j], m2[i][j] * scale)) + } + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3_det) { + mat3 mat; + float a, b, c, + d, e, f, + g, h, i; + float det1, det2; + + test_rand_mat3(mat); + + a = mat[0][0]; b = mat[0][1]; c = mat[0][2]; + d = mat[1][0]; e = mat[1][1]; f = mat[1][2]; + g = mat[2][0]; h = mat[2][1]; i = mat[2][2]; + + det1 = a * (e * i - h * f) - d * (b * i - c * h) + g * (b * f - c * e); + det2 = GLM(mat3_det)(mat); + + ASSERT(test_eq(det1, det2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3_inv) { + mat3 m1 = GLM_MAT3_IDENTITY_INIT; + mat3 m2 = GLM_MAT3_IDENTITY_INIT; + mat3 m3; + int i; + + for (i = 0; i < 100000; i++) { + test_rand_mat3(m1); + test_rand_mat3(m2); + + /* test inverse precise */ + GLM(mat3_inv)(m1, m2); + GLM(mat3_inv)(m2, m3); + ASSERTIFY(test_assert_mat3_eq(m1, m3)) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3_swap_col) { + mat3 m1 = A_MATRIX; + mat3 m2 = A_MATRIX; + + GLM(mat3_swap_col)(m1, 0, 1); + + ASSERTIFY(test_assert_vec3_eq(m1[0], m2[1])) + ASSERTIFY(test_assert_vec3_eq(m1[1], m2[0])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3_swap_row) { + mat3 m1 = A_MATRIX; + mat3 m2 = A_MATRIX; + + GLM(mat3_swap_row)(m1, 0, 1); + + ASSERT(test_eq(m1[0][0], m2[0][1])) + ASSERT(test_eq(m1[0][1], m2[0][0])) + ASSERT(test_eq(m1[0][2], m2[0][2])) + + ASSERT(test_eq(m1[1][0], m2[1][1])) + ASSERT(test_eq(m1[1][1], m2[1][0])) + ASSERT(test_eq(m1[1][2], m2[1][2])) + + ASSERT(test_eq(m1[2][0], m2[2][1])) + ASSERT(test_eq(m1[2][1], m2[2][0])) + ASSERT(test_eq(m1[2][2], m2[2][2])) + + GLM(mat3_swap_row)(m1, 1, 2); + + ASSERT(test_eq(m1[0][0], m2[0][1])) + ASSERT(test_eq(m1[0][1], m2[0][2])) + ASSERT(test_eq(m1[0][2], m2[0][0])) + + ASSERT(test_eq(m1[1][0], m2[1][1])) + ASSERT(test_eq(m1[1][1], m2[1][2])) + ASSERT(test_eq(m1[1][2], m2[1][0])) + + ASSERT(test_eq(m1[2][0], m2[2][1])) + ASSERT(test_eq(m1[2][1], m2[2][2])) + ASSERT(test_eq(m1[2][2], m2[2][0])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3_rmc) { + mat3 mat = A_MATRIX; + vec3 v = {11.0f, 12.0f, 13.0f}; + vec3 v1; + float r1, r2; + int i; + + r1 = GLM(mat3_rmc)(v, mat, v); + + for (i = 0; i < 3; i++) { + v1[i] = v[0] * mat[i][0] + + v[1] * mat[i][1] + + v[2] * mat[i][2]; + } + + r2 = v[0] * v1[0] + v[1] * v1[1] + v[2] * v1[2]; + + ASSERT(test_eq(r1, r2)) + + TEST_SUCCESS +} + + +TEST_IMPL(GLM_PREFIX, mat3_make) { + mat3 dest; + unsigned int i, j; + float src[9] = MAT3_ARRAY; + + GLM(mat3_make)(src, dest); + + for (i = 0, j = 0; i < sizeof(src) / sizeof(float); i+=3, j++) { + ASSERT(test_eq(dest[j][0], src[i])) + ASSERT(test_eq(dest[j][1], src[i+1])) + ASSERT(test_eq(dest[j][2], src[i+2])) + } + + TEST_SUCCESS +} + +#undef A_MATRIX diff --git a/cglm/test/src/test_mat3x2.h b/cglm/test/src/test_mat3x2.h new file mode 100644 index 0000000..397879b --- /dev/null +++ b/cglm/test/src/test_mat3x2.h @@ -0,0 +1,155 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +#define A_MATRIX3X2 {{1,2},{5,6},{3,7}} +#define A_MATRIX3X2_TRANSPOSE {{1,5,3}, {2,6,7}} + +#ifndef CGLM_TEST_MAT3X2_ONCE +#define CGLM_TEST_MAT3X2_ONCE + +TEST_IMPL(MACRO_GLM_MAT3X2_ZERO_INIT) { + mat3x2 mat3x2_zero = GLM_MAT3X2_ZERO_INIT; + test_assert_mat3x2_eq_zero(mat3x2_zero); + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_MAT3X2_ZERO) { + mat3x2 mat3x2_zero = GLM_MAT3X2_ZERO; + test_assert_mat3x2_eq_zero(mat3x2_zero); + TEST_SUCCESS +} + +#endif /* CGLM_TEST_MAT3X2_ONCE */ + +TEST_IMPL(GLM_PREFIX, mat3x2_copy) { + mat3x2 m1 = A_MATRIX3X2; + mat3x2 m2 = GLM_MAT3X2_ZERO_INIT; + + GLM(mat3x2_copy)(m1, m2); + + test_assert_mat3x2_eq(m1, m2); + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3x2_zero) { + mat3x2 m1 = GLM_MAT3X2_ZERO_INIT; + mat3x2 m2 = GLM_MAT3X2_ZERO_INIT; + mat3x2 m3; + + GLM(mat3x2_zero)(m3); + + ASSERTIFY(test_assert_mat3x2_eq_zero(m1)) + ASSERTIFY(test_assert_mat3x2_eq_zero(m2)) + ASSERTIFY(test_assert_mat3x2_eq_zero(m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3x2_make) { + float src[18] = { + 0.5f, 1.7f, 10.3f, 4.2f, 8.9f, 1.1f, + 2.3f, 4.2f, 66.5f, 23.7f, 6.6f, 8.9f, + 5.3f, 4.8f, 96.3f, 13.7f, 4.7f, 5.5f + }; + + mat3x2 dest[3]; + + float *srcp = src; + unsigned int i, j, k; + + for (i = 0, j = 0, k = 0; i < sizeof(src) / sizeof(float); i+=6,j++) { + GLM(mat3x2_make)(srcp + i, dest[j]); + + ASSERT(test_eq(src[ i ], dest[j][k][0])); + ASSERT(test_eq(src[i+1], dest[j][k][1])); + + ASSERT(test_eq(src[i+2], dest[j][k+1][0])); + ASSERT(test_eq(src[i+3], dest[j][k+1][1])); + + ASSERT(test_eq(src[i+4], dest[j][k+2][0])); + ASSERT(test_eq(src[i+5], dest[j][k+2][1])); + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3x2_mul) { + mat3x2 m1 = GLM_MAT3X2_ZERO_INIT; + mat2x3 m2 = GLM_MAT2X3_ZERO_INIT; + + mat2 m3 = GLM_MAT2_ZERO_INIT; + mat2 m4 = GLM_MAT2_ZERO_INIT; + + int c, r, k; + + test_rand_mat3x2(m1); + test_rand_mat2x3(m2); + + for (r = 0; r < 2; r++) { + for (c = 0; c < 2; c++) { + for (k = 0; k < 3; k++) { + m4[c][r] += m1[k][r] * m2[c][k]; + } + } + } + + GLM(mat3x2_mul)(m1, m2, m3); + ASSERTIFY(test_assert_mat2_eq(m3, m4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3x2_mulv) { + mat3x2 mat = A_MATRIX3X2; + vec3 v = {11.0f, 21.0f, 31.0f}; + + int i; + vec2 dest; + float res = 0.0; + + GLM(mat3x2_mulv)(mat, v, dest); + + for (i = 0; i < 2; i++) { + res = mat[0][i] * v[0] + mat[1][i] * v[1] + mat[2][i] * v[2]; + ASSERT(test_eq(dest[i], res)) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3x2_transpose) { + mat3x2 m1 = A_MATRIX3X2; + + mat2x3 m2; + mat2x3 m3 = A_MATRIX3X2_TRANSPOSE; + GLM(mat3x2_transpose)(m1, m2); + + ASSERTIFY(test_assert_mat2x3_eq(m2, m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3x2_scale) { + mat3x2 m1 = A_MATRIX3X2; + mat3x2 m2 = A_MATRIX3X2; + int i, j, scale; + + scale = rand() % 100; + + GLM(mat3x2_scale)(m1, (float) scale); + + for (i = 0; i < 3; i++) { + for (j = 0; j < 2; j++) { + ASSERT(test_eq(m1[i][j], m2[i][j] * scale)) + } + } + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_mat3x4.h b/cglm/test/src/test_mat3x4.h new file mode 100644 index 0000000..16520ac --- /dev/null +++ b/cglm/test/src/test_mat3x4.h @@ -0,0 +1,161 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +#define A_MATRIX3X4 {{1,2,4,5},{6,7,8,9},{10,11,12,13}} +#define A_MATRIX3X4_TRANSPOSE {{1,6,10}, {2,7,1}, {4,8,12}, {5,9,13}} + +#ifndef CGLM_TEST_MAT3X4_ONCE +#define CGLM_TEST_MAT3X4_ONCE + +TEST_IMPL(MACRO_GLM_MAT3X4_ZERO_INIT) { + mat3x4 mat3x4_zero = GLM_MAT3X4_ZERO_INIT; + test_assert_mat3x4_eq_zero(mat3x4_zero); + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_MAT3X4_ZERO) { + mat3x4 mat3x4_zero = GLM_MAT3X4_ZERO; + test_assert_mat3x4_eq_zero(mat3x4_zero); + TEST_SUCCESS +} + +#endif /* CGLM_TEST_MAT3X4_ONCE */ + +TEST_IMPL(GLM_PREFIX, mat3x4_copy) { + mat3x4 m1 = A_MATRIX3X4; + mat3x4 m2 = GLM_MAT3X4_ZERO_INIT; + + GLM(mat3x4_copy)(m1, m2); + + test_assert_mat3x4_eq(m1, m2); + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3x4_zero) { + mat3x4 m1 = GLM_MAT3X4_ZERO_INIT; + mat3x4 m2 = GLM_MAT3X4_ZERO_INIT; + mat3x4 m3; + + GLM(mat3x4_zero)(m3); + + ASSERTIFY(test_assert_mat3x4_eq_zero(m1)) + ASSERTIFY(test_assert_mat3x4_eq_zero(m2)) + ASSERTIFY(test_assert_mat3x4_eq_zero(m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3x4_make) { + float src[36] = { + 0.5f, 1.7f, 10.3f, 4.2f, 8.9f, 1.1f, 2.3f, 4.2f, 66.5f, 23.7f, 6.6f, 8.9f, + 2.3f, 4.2f, 66.5f, 23.7f, 6.6f, 8.9f, 0.5f, 1.7f, 10.3f, 4.2f, 8.9f, 1.1f, + 5.3f, 4.8f, 96.3f, 13.7f, 4.7f, 5.5f, 2.3f, 4.2f, 66.5f, 23.7f, 6.6f, 8.9f + }; + + mat3x4 dest[3]; + + float *srcp = src; + unsigned int i, j, k; + + for (i = 0, j = 0, k = 0; i < sizeof(src) / sizeof(float); i+=12,j++) { + GLM(mat3x4_make)(srcp + i, dest[j]); + + ASSERT(test_eq(src[ i ], dest[j][k][0])); + ASSERT(test_eq(src[i+1], dest[j][k][1])); + ASSERT(test_eq(src[i+2], dest[j][k][2])); + ASSERT(test_eq(src[i+3], dest[j][k][3])); + + ASSERT(test_eq(src[i+4], dest[j][k+1][0])); + ASSERT(test_eq(src[i+5], dest[j][k+1][1])); + ASSERT(test_eq(src[i+6], dest[j][k+1][2])); + ASSERT(test_eq(src[i+7], dest[j][k+1][3])); + + ASSERT(test_eq(src[i+8], dest[j][k+2][0])); + ASSERT(test_eq(src[i+9], dest[j][k+2][1])); + ASSERT(test_eq(src[i+10], dest[j][k+2][2])); + ASSERT(test_eq(src[i+11], dest[j][k+2][3])); + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3x4_mul) { + mat3x4 m1 = GLM_MAT3X4_ZERO_INIT; + mat4x3 m2 = GLM_MAT4X3_ZERO_INIT; + + mat4 m3 = GLM_MAT4_ZERO_INIT; + mat4 m4 = GLM_MAT4_ZERO_INIT; + + int c, r, k; + + test_rand_mat3x4(m1); + test_rand_mat4x3(m2); + + for (r = 0; r < 4; r++) { + for (c = 0; c < 4; c++) { + for (k = 0; k < 3; k++) { + m4[c][r] += m1[k][r] * m2[c][k]; + } + } + } + + GLM(mat3x4_mul)(m1, m2, m3); + ASSERTIFY(test_assert_mat4_eq(m3, m4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3x4_mulv) { + mat3x4 mat = A_MATRIX3X4; + vec4 v = {11.0f, 21.0f, 31.0f}; + + int i; + vec4 dest; + float res = 0.0; + + GLM(mat3x4_mulv)(mat, v, dest); + + for (i = 0; i < 4; i++) { + res = mat[0][i] * v[0] + mat[1][i] * v[1] + mat[2][i] * v[2]; + ASSERT(test_eq(dest[i], res)) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3x4_transpose) { + mat3x4 m1 = A_MATRIX3X4; + + mat4x3 m2; + mat4x3 m3 = A_MATRIX3X2_TRANSPOSE; + GLM(mat3x4_transpose)(m1, m2); + + ASSERTIFY(test_assert_mat4x3_eq(m2, m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat3x4_scale) { + mat3x4 m1 = A_MATRIX3X4; + mat3x4 m2 = A_MATRIX3X4; + int i, j, scale; + + scale = rand() % 100; + + GLM(mat3x4_scale)(m1, (float) scale); + + for (i = 0; i < 3; i++) { + for (j = 0; j < 4; j++) { + ASSERT(test_eq(m1[i][j], m2[i][j] * scale)) + } + } + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_mat4.h b/cglm/test/src/test_mat4.h new file mode 100644 index 0000000..16d80bb --- /dev/null +++ b/cglm/test/src/test_mat4.h @@ -0,0 +1,506 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +#define A_MATRIX {{1,2,3,4},{5,6,7,8},{9,10,11,12},{13,14,15,16}} +#define A_MATRIX3 {{1,2,3},{5,6,7},{9,10,11}} +#define MAT4_ARRAY {1, 5, 2, 7, 12, 1, 4, 6, 0, 8, 1, 0, 6, 5, 0, 1} + +TEST_IMPL(GLM_PREFIX, mat4_ucopy) { + mat4 m1 = A_MATRIX; + mat4 m2 = GLM_MAT4_IDENTITY_INIT; + + GLM(mat4_ucopy)(m1, m2); + + ASSERTIFY(test_assert_mat4_eq(m1, m2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_copy) { + mat4 m1 = A_MATRIX; + mat4 m2 = GLM_MAT4_IDENTITY_INIT; + + GLM(mat4_copy)(m1, m2); + + test_assert_mat4_eq(m1, m2); + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_identity) { + mat4 m1 = GLM_MAT4_IDENTITY_INIT; + mat4 m2 = GLM_MAT4_IDENTITY_INIT; + mat4 m3; + + GLM(mat4_identity)(m3); + + ASSERTIFY(test_assert_mat4_eq_identity(m1)) + ASSERTIFY(test_assert_mat4_eq_identity(m2)) + ASSERTIFY(test_assert_mat4_eq_identity(m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_identity_array) { + size_t i, count; + mat4 matrices[4] = { + A_MATRIX, + A_MATRIX, + A_MATRIX, + A_MATRIX + }; + + count = 4; + + GLM(mat4_identity_array)(matrices, count); + + for (i = 0; i < count; i++) { + ASSERTIFY(test_assert_mat4_eq_identity(matrices[i])) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_zero) { + mat4 m1 = GLM_MAT4_ZERO_INIT; + mat4 m2 = GLM_MAT4_ZERO_INIT; + mat4 m3; + + GLM(mat4_zero)(m3); + + ASSERTIFY(test_assert_mat4_eq_zero(m1)) + ASSERTIFY(test_assert_mat4_eq_zero(m2)) + ASSERTIFY(test_assert_mat4_eq_zero(m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_pick3) { + mat4 m1 = A_MATRIX; + mat3 m2 = GLM_MAT3_ZERO_INIT; + mat3 m3 = A_MATRIX3; + + GLM(mat4_pick3)(m1, m2); + + ASSERTIFY(test_assert_mat3_eq(m2, m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_pick3t) { + mat4 m1 = A_MATRIX; + mat3 m2 = GLM_MAT3_ZERO_INIT; + mat3 m3 = A_MATRIX3; + + GLM(mat4_pick3t)(m1, m2); + + ASSERTIFY(test_assert_mat3_eqt(m2, m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_ins3) { + mat4 m1 = GLM_MAT4_IDENTITY_INIT; + mat3 m2 = A_MATRIX3; + int i, j; + + GLM(mat4_ins3)(m2, m1); + + for (i = 0; i < 3; i++) { + for (j = 0; j < 3; j++) { + ASSERT(m1[i][j] == m2[i][j]) + } + } + + ASSERT(test_eq(m1[3][0], 0.0f)) + ASSERT(test_eq(m1[3][1], 0.0f)) + ASSERT(test_eq(m1[3][2], 0.0f)) + ASSERT(test_eq(m1[3][3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_mul) { + mat4 m1 = GLM_MAT4_IDENTITY_INIT; + mat4 m2 = GLM_MAT4_IDENTITY_INIT; + mat4 m3; + mat4 m4 = GLM_MAT4_ZERO_INIT; + int i, j, k; + + /* test random matrices */ + /* random matrices */ + test_rand_mat4(m1); + test_rand_mat4(m2); + + GLM(mat4_mul)(m1, m2, m3); + for (i = 0; i < 4; i++) { + for (j = 0; j < 4; j++) { + for (k = 0; k < 4; k++) + /* column-major */ + m4[i][j] += m1[k][j] * m2[i][k]; + } + } + + ASSERTIFY(test_assert_mat4_eq(m3, m4)) + + /* test pre compiled */ + GLM(mat4_mul)(m1, m2, m3); + ASSERTIFY(test_assert_mat4_eq(m3, m4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_mulN) { + mat4 res1, res2; + mat4 m1 = A_MATRIX; + mat4 m2 = A_MATRIX; + mat4 m3 = A_MATRIX; + + mat4 *matrices[] = { + &m1, &m2, &m3 + }; + + GLM(mat4_mulN)(matrices, sizeof(matrices) / sizeof(matrices[0]), res1); + + GLM(mat4_mul)(*matrices[0], *matrices[1], res2); + GLM(mat4_mul)(res2, *matrices[2], res2); + + ASSERTIFY(test_assert_mat4_eq(res1, res1)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_mulv) { + vec4 res; + mat4 mat = A_MATRIX; + vec4 v = {1.0f, 2.0f, 3.0f, 4.0f}; + int i; + + GLM(mat4_mulv)(mat, v, res); + + for (i = 0; i < 4; i++) { + ASSERT(test_eq(res[i], + v[0] * mat[0][i] + + v[1] * mat[1][i] + + v[2] * mat[2][i] + + v[3] * mat[3][i])) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_mulv3) { + vec4 res; + mat4 mat = A_MATRIX; + vec3 v = {1.0f, 2.0f, 3.0f}; + float last; + int i; + + last = 1.0f; + + GLM(mat4_mulv3)(mat, v, last, res); + + for (i = 0; i < 3; i++) { + ASSERT(test_eq(res[i], + v[0] * mat[0][i] + + v[1] * mat[1][i] + + v[2] * mat[2][i] + + last * mat[3][i])) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_trace) { + mat4 mat = A_MATRIX; + float trace; + + trace = GLM(mat4_trace)(mat); + + ASSERT(test_eq(trace, mat[0][0] + mat[1][1] + mat[2][2] + mat[3][3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_trace3) { + mat4 mat = A_MATRIX; + float trace; + + trace = GLM(mat4_trace3)(mat); + + ASSERT(test_eq(trace, mat[0][0] + mat[1][1] + mat[2][2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_quat) { + mat4 m1, m2; + versor q1, q2, q3; + vec3 axis1; + vec3 axis2 = {1.9f, 2.3f, 4.5f}; + + GLM(quat)(q1, GLM_PI_4f, 1.9f, 2.3f, 4.5f); + GLM(quat_mat4)(q1, m1); + GLM(mat4_quat)(m1, q2); + + GLM(rotate_make)(m2, GLM_PI_4f, axis2); + GLM(mat4_quat)(m1, q3); + + GLM(quat_axis)(q3, axis1); + + GLM(vec3_normalize)(axis1); + GLM(vec3_normalize)(axis2); + + ASSERT(test_eq(glm_quat_angle(q3), GLM_PI_4f)) + ASSERTIFY(test_assert_vec3_eq(axis1, axis2)) + ASSERTIFY(test_assert_vec4_eq(q1, q2)) + ASSERTIFY(test_assert_mat4_eq(m1, m2)) + ASSERTIFY(test_assert_vec4_eq(q1, q3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_transpose_to) { + mat4 mat = A_MATRIX; + mat4 m1; + + GLM(mat4_transpose_to)(mat, m1); + + ASSERTIFY(test_assert_mat4_eqt(mat, m1)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_transpose) { + mat4 mat = A_MATRIX; + mat4 m1; + + GLM(mat4_copy)(mat, m1); + GLM(mat4_transpose)(m1); + + ASSERTIFY(test_assert_mat4_eqt(mat, m1)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_scale_p) { + mat4 m1 = A_MATRIX; + mat4 m2 = A_MATRIX; + int i, j, k, scale; + + scale = rand() % 100; + + GLM(mat4_scale_p)(m1, (float)scale); + + for (i = 0; i < 4; i++) { + for (j = 0; j < 4; j++) { + for (k = 0; k < 4; k++) + ASSERT(test_eq(m1[i][j], m2[i][j] * scale)) + } + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_scale) { + mat4 m1 = A_MATRIX; + mat4 m2 = A_MATRIX; + int i, j, k, scale; + + scale = rand() % 100; + + GLM(mat4_scale)(m1, (float)scale); + + for (i = 0; i < 4; i++) { + for (j = 0; j < 4; j++) { + for (k = 0; k < 4; k++) + ASSERT(test_eq(m1[i][j], m2[i][j] * scale)) + } + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_det) { + mat4 mat = GLM_MAT4_IDENTITY_INIT; + float t[6]; + float a, b, c, d, + e, f, g, h, + i, j, k, l, + m, n, o, p; + float det1, det2; + + test_rand_mat4(mat); + + a = mat[0][0]; b = mat[0][1]; c = mat[0][2]; d = mat[0][3]; + e = mat[1][0]; f = mat[1][1]; g = mat[1][2]; h = mat[1][3]; + i = mat[2][0]; j = mat[2][1]; k = mat[2][2]; l = mat[2][3]; + m = mat[3][0]; n = mat[3][1]; o = mat[3][2]; p = mat[3][3]; + + t[0] = k * p - o * l; + t[1] = j * p - n * l; + t[2] = j * o - n * k; + t[3] = i * p - m * l; + t[4] = i * o - m * k; + t[5] = i * n - m * j; + + det1 = a * (f * t[0] - g * t[1] + h * t[2]) + - b * (e * t[0] - g * t[3] + h * t[4]) + + c * (e * t[1] - f * t[3] + h * t[5]) + - d * (e * t[2] - f * t[4] + g * t[5]); + + det2 = GLM(mat4_det(mat)); + + ASSERT(test_eq(det1, det2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_inv) { + mat4 m1, m2, m3; + int i; + + for (i = 0; i < 100000; i++) { + test_rand_mat4(m1); + test_rand_mat4(m2); + + /* test inverse precise */ + GLM(mat4_inv)(m1, m2); + GLM(mat4_inv)(m2, m3); + ASSERTIFY(test_assert_mat4_eq(m1, m3)) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_inv_precise) { + mat4 m1, m2, m3; + mat4 m4, m5, m6; + int i; + + for (i = 0; i < 100000; i++) { + test_rand_mat4(m1); + test_rand_mat4(m2); + + glm_mat4_inv_precise(m1, m2); + glm_mat4_inv_precise(m2, m3); + ASSERTIFY(test_assert_mat4_eq(m1, m3)) + + test_rand_mat4(m4); + test_rand_mat4(m5); + + glmc_mat4_inv_precise(m4, m5); + glmc_mat4_inv_precise(m5, m6); + ASSERTIFY(test_assert_mat4_eq(m4, m6)) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_inv_fast) { + mat4 m1, m2, m3; + int i; + + for (i = 0; i < 100000; i++) { + test_rand_mat4(m1); + test_rand_mat4(m2); + + /* test inverse precise */ + GLM(mat4_inv_fast)(m1, m2); + GLM(mat4_inv_fast)(m2, m3); + ASSERTIFY(test_assert_mat4_eq2(m1, m3, 0.0009f)) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_swap_col) { + mat4 m1 = A_MATRIX; + mat4 m2 = A_MATRIX; + + GLM(mat4_swap_col)(m1, 0, 1); + GLM(mat4_swap_col)(m1, 2, 3); + + ASSERTIFY(test_assert_vec4_eq(m1[0], m2[1])) + ASSERTIFY(test_assert_vec4_eq(m1[1], m2[0])) + ASSERTIFY(test_assert_vec4_eq(m1[2], m2[3])) + ASSERTIFY(test_assert_vec4_eq(m1[3], m2[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_swap_row) { + mat4 m1 = A_MATRIX; + mat4 m2 = A_MATRIX; + + GLM(mat4_swap_row)(m1, 0, 1); + GLM(mat4_swap_row)(m1, 2, 3); + + ASSERT(test_eq(m1[0][0], m2[0][1])) + ASSERT(test_eq(m1[0][1], m2[0][0])) + ASSERT(test_eq(m1[0][2], m2[0][3])) + ASSERT(test_eq(m1[0][3], m2[0][2])) + ASSERT(test_eq(m1[1][0], m2[1][1])) + ASSERT(test_eq(m1[1][1], m2[1][0])) + ASSERT(test_eq(m1[1][2], m2[1][3])) + ASSERT(test_eq(m1[1][3], m2[1][2])) + ASSERT(test_eq(m1[2][0], m2[2][1])) + ASSERT(test_eq(m1[2][1], m2[2][0])) + ASSERT(test_eq(m1[2][2], m2[2][3])) + ASSERT(test_eq(m1[2][3], m2[2][2])) + ASSERT(test_eq(m1[3][0], m2[3][1])) + ASSERT(test_eq(m1[3][1], m2[3][0])) + ASSERT(test_eq(m1[3][2], m2[3][3])) + ASSERT(test_eq(m1[3][3], m2[3][2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_rmc) { + mat4 mat = A_MATRIX; + vec4 v = {1.0f, 2.0f, 3.0f, 4.0f}; + vec4 v1; + float r1, r2; + int i; + + r1 = GLM(mat4_rmc)(v, mat, v); + + for (i = 0; i < 4; i++) { + v1[i] = v[0] * mat[i][0] + + v[1] * mat[i][1] + + v[2] * mat[i][2] + + v[3] * mat[i][3]; + } + + r2 = v[0] * v1[0] + v[1] * v1[1] + v[2] * v1[2] + v[3] * v1[3]; + + ASSERT(test_eq(r1, r2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4_make) { + mat4 dest; + unsigned int i, j; + float src[16] = MAT4_ARRAY; + + GLM(mat4_make)(src, dest); + + for (i = 0, j = 0; i < sizeof(src) / sizeof(float); i+=4, j++) { + ASSERT(test_eq(dest[j][0], src[i])) + ASSERT(test_eq(dest[j][1], src[i+1])) + ASSERT(test_eq(dest[j][2], src[i+2])) + ASSERT(test_eq(dest[j][3], src[i+3])) + } + + TEST_SUCCESS +} + +#undef A_MATRIX +#undef A_MATRIX3 diff --git a/cglm/test/src/test_mat4x2.h b/cglm/test/src/test_mat4x2.h new file mode 100644 index 0000000..b58d974 --- /dev/null +++ b/cglm/test/src/test_mat4x2.h @@ -0,0 +1,158 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +#define A_MATRIX4X2 {{1,2},{3,4},{5,6},{7,8}} +#define A_MATRIX4X2_TRANSPOSE {{1,3,5,7}, {2,4,6,8}} + +#ifndef CGLM_TEST_MAT4X2_ONCE +#define CGLM_TEST_MAT4X2_ONCE + +TEST_IMPL(MACRO_GLM_MAT4X2_ZERO_INIT) { + mat4x2 mat4x2_zero = GLM_MAT4X2_ZERO_INIT; + test_assert_mat4x2_eq_zero(mat4x2_zero); + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_MAT4X2_ZERO) { + mat4x2 mat4x2_zero = GLM_MAT4X2_ZERO; + test_assert_mat4x2_eq_zero(mat4x2_zero); + TEST_SUCCESS +} + +#endif /* CGLM_TEST_MAT4X2_ONCE */ + +TEST_IMPL(GLM_PREFIX, mat4x2_copy) { + mat4x2 m1 = A_MATRIX4X2; + mat4x2 m2 = GLM_MAT4X2_ZERO_INIT; + + GLM(mat4x2_copy)(m1, m2); + + test_assert_mat4x2_eq(m1, m2); + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4x2_zero) { + mat4x2 m1 = GLM_MAT4X2_ZERO_INIT; + mat4x2 m2 = GLM_MAT4X2_ZERO_INIT; + mat4x2 m3; + + GLM(mat4x2_zero)(m3); + + ASSERTIFY(test_assert_mat4x2_eq_zero(m1)) + ASSERTIFY(test_assert_mat4x2_eq_zero(m2)) + ASSERTIFY(test_assert_mat4x2_eq_zero(m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4x2_make) { + float src[24] = { + 0.5f, 1.7f, 10.3f, 4.2f, 8.9f, 1.1f, 2.3f, 4.2f, + 2.3f, 4.2f, 66.5f, 23.7f, 6.6f, 8.9f, 0.5f, 1.7f, + 5.3f, 4.8f, 96.3f, 13.7f, 4.7f, 5.5f, 2.3f, 4.2f + }; + + mat4x2 dest[3]; + + float *srcp = src; + unsigned int i, j, k; + + for (i = 0, j = 0, k = 0; i < sizeof(src) / sizeof(float); i+=8,j++) { + GLM(mat4x2_make)(srcp + i, dest[j]); + + ASSERT(test_eq(src[ i ], dest[j][k][0])); + ASSERT(test_eq(src[i+1], dest[j][k][1])); + + ASSERT(test_eq(src[i+2], dest[j][k+1][0])); + ASSERT(test_eq(src[i+3], dest[j][k+1][1])); + + ASSERT(test_eq(src[i+4], dest[j][k+2][0])); + ASSERT(test_eq(src[i+5], dest[j][k+2][1])); + + ASSERT(test_eq(src[i+6], dest[j][k+3][0])); + ASSERT(test_eq(src[i+7], dest[j][k+3][1])); + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4x2_mul) { + mat4x2 m1 = GLM_MAT4X2_ZERO_INIT; + mat2x4 m2 = GLM_MAT2X4_ZERO_INIT; + + mat2 m3 = GLM_MAT2_ZERO_INIT; + mat2 m4 = GLM_MAT2_ZERO_INIT; + + int c, r, k; + + test_rand_mat4x2(m1); + test_rand_mat2x4(m2); + + for (r = 0; r < 2; r++) { + for (c = 0; c < 2; c++) { + for (k = 0; k < 4; k++) { + m4[c][r] += m1[k][r] * m2[c][k]; + } + } + } + + GLM(mat4x2_mul)(m1, m2, m3); + ASSERTIFY(test_assert_mat2_eq(m3, m4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4x2_mulv) { + mat4x2 mat = A_MATRIX4X2; + vec4 v = {11.0f, 21.0f, 31.0f, 41.0f}; + + int i; + vec2 dest; + float res = 0.0; + + GLM(mat4x2_mulv)(mat, v, dest); + + for (i = 0; i < 2; i++) { + res = mat[0][i] * v[0] + mat[1][i] * v[1] + mat[2][i] * v[2] + mat[3][i] * v[3]; + ASSERT(test_eq(dest[i], res)) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4x2_transpose) { + mat4x2 m1 = A_MATRIX4X2; + + mat2x4 m2; + mat2x4 m3 = A_MATRIX4X2_TRANSPOSE; + GLM(mat4x2_transpose)(m1, m2); + + ASSERTIFY(test_assert_mat2x4_eq(m2, m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4x2_scale) { + mat4x2 m1 = A_MATRIX4X2; + mat4x2 m2 = A_MATRIX4X2; + int i, j, scale; + + scale = rand() % 100; + + GLM(mat4x2_scale)(m1, (float) scale); + + for (i = 0; i < 4; i++) { + for (j = 0; j < 2; j++) { + ASSERT(test_eq(m1[i][j], m2[i][j] * scale)) + } + } + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_mat4x3.h b/cglm/test/src/test_mat4x3.h new file mode 100644 index 0000000..dc7c28e --- /dev/null +++ b/cglm/test/src/test_mat4x3.h @@ -0,0 +1,162 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +#define A_MATRIX4X3 {{1,2,3},{4,5,6},{7,8,9},{10,11,12}} +#define A_MATRIX4X3_TRANSPOSE {{1,4,7,10},{2,5,8,11},{3,6,9,12}} + +#ifndef CGLM_TEST_MAT4X3_ONCE +#define CGLM_TEST_MAT4X3_ONCE + +TEST_IMPL(MACRO_GLM_MAT4X3_ZERO_INIT) { + mat4x3 mat4x3_zero = GLM_MAT4X3_ZERO_INIT; + test_assert_mat4x3_eq_zero(mat4x3_zero); + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_MAT4X3_ZERO) { + mat4x3 mat4x3_zero = GLM_MAT4X3_ZERO; + test_assert_mat4x3_eq_zero(mat4x3_zero); + TEST_SUCCESS +} + +#endif /* CGLM_TEST_MAT4X3_ONCE */ + +TEST_IMPL(GLM_PREFIX, mat4x3_copy) { + mat4x3 m1 = A_MATRIX4X3; + mat4x3 m2 = GLM_MAT4X3_ZERO_INIT; + + GLM(mat4x3_copy)(m1, m2); + + test_assert_mat4x3_eq(m1, m2); + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4x3_zero) { + mat4x3 m1 = GLM_MAT4X3_ZERO_INIT; + mat4x3 m2 = GLM_MAT4X3_ZERO_INIT; + mat4x3 m3; + + GLM(mat4x3_zero)(m3); + + ASSERTIFY(test_assert_mat4x3_eq_zero(m1)) + ASSERTIFY(test_assert_mat4x3_eq_zero(m2)) + ASSERTIFY(test_assert_mat4x3_eq_zero(m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4x3_make) { + float src[36] = { + 0.5f, 1.7f, 10.3f, 4.2f, 8.9f, 1.1f, 2.3f, 4.2f, 5.3f, 4.8f, 96.3f, 13.7f, + 4.7f, 5.5f, 2.3f, 4.2f, 2.3f, 4.2f, 66.5f, 23.7f, 6.6f, 8.9f, 0.5f, 1.7f, + 5.3f, 4.8f, 96.3f, 13.7f, 4.7f, 5.5f, 2.3f, 4.2f, 0.5f, 1.7f, 10.3f, 4.2f + }; + + mat4x3 dest[3]; + + float *srcp = src; + unsigned int i, j, k; + + for (i = 0, j = 0, k = 0; i < sizeof(src) / sizeof(float); i+=12,j++) { + GLM(mat4x3_make)(srcp + i, dest[j]); + + ASSERT(test_eq(src[ i ], dest[j][k][0])); + ASSERT(test_eq(src[i+1], dest[j][k][1])); + ASSERT(test_eq(src[i+2], dest[j][k][2])); + + ASSERT(test_eq(src[i+3], dest[j][k+1][0])); + ASSERT(test_eq(src[i+4], dest[j][k+1][1])); + ASSERT(test_eq(src[i+5], dest[j][k+1][2])); + + ASSERT(test_eq(src[i+6], dest[j][k+2][0])); + ASSERT(test_eq(src[i+7], dest[j][k+2][1])); + ASSERT(test_eq(src[i+8], dest[j][k+2][2])); + + ASSERT(test_eq(src[i+9], dest[j][k+3][0])); + ASSERT(test_eq(src[i+10], dest[j][k+3][1])); + ASSERT(test_eq(src[i+11], dest[j][k+3][2])); + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4x3_mul) { + mat4x3 m1 = GLM_MAT4X3_ZERO_INIT; + mat3x4 m2 = GLM_MAT3X4_ZERO_INIT; + + mat3 m3 = GLM_MAT3_ZERO_INIT; + mat3 m4 = GLM_MAT3_ZERO_INIT; + + int c, r, k; + + test_rand_mat4x3(m1); + test_rand_mat3x4(m2); + + for (r = 0; r < 3; r++) { + for (c = 0; c < 3; c++) { + for (k = 0; k < 4; k++) { + m4[c][r] += m1[k][r] * m2[c][k]; + } + } + } + + GLM(mat4x3_mul)(m1, m2, m3); + ASSERTIFY(test_assert_mat3_eq(m3, m4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4x3_mulv) { + mat4x3 mat = A_MATRIX4X3; + vec4 v = {11.0f, 21.0f, 31.0f, 41.0f}; + + int i; + vec3 dest; + float res = 0.0; + + GLM(mat4x3_mulv)(mat, v, dest); + + for (i = 0; i < 3; i++) { + res = mat[0][i] * v[0] + mat[1][i] * v[1] + mat[2][i] * v[2] + mat[3][i] * v[3]; + ASSERT(test_eq(dest[i], res)) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4x3_transpose) { + mat4x3 m1 = A_MATRIX4X3; + + mat3x4 m2; + mat3x4 m3 = A_MATRIX4X3_TRANSPOSE; + GLM(mat4x3_transpose)(m1, m2); + + ASSERTIFY(test_assert_mat3x4_eq(m2, m3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, mat4x3_scale) { + mat4x3 m1 = A_MATRIX4X3; + mat4x3 m2 = A_MATRIX4X3; + int i, j, scale; + + scale = rand() % 100; + + GLM(mat4x3_scale)(m1, (float) scale); + + for (i = 0; i < 4; i++) { + for (j = 0; j < 3; j++) { + ASSERT(test_eq(m1[i][j], m2[i][j] * scale)) + } + } + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_noise.h b/cglm/test/src/test_noise.h new file mode 100644 index 0000000..cc7ea3c --- /dev/null +++ b/cglm/test/src/test_noise.h @@ -0,0 +1,118 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +TEST_IMPL(GLM_PREFIX, perlin_vec4) { + vec4 p1[] = { + {0.1f, 0.2f, 0.3f, 0.4f}, + {0.2f, 0.3f, 0.4f, 0.5f}, + {0.3f, 0.4f, 0.5f, 0.6f}, + {0.4f, 0.5f, 0.6f, 0.7f}, + {0.5f, 0.6f, 0.7f, 0.8f}, + {0.6f, 0.7f, 0.8f, 0.9f}, + {0.7f, 0.8f, 0.9f, 1.0f}, + {0.8f, 0.9f, 1.0f, 1.1f}, + {0.9f, 1.0f, 1.1f, 1.2f}, + {1.0f, 1.1f, 1.2f, 1.3f}, + }; + + /* expected values calculated by glm::perlin */ + float e[] = { + -0.5091819763183594f, + -0.4375732541084290f, + -0.3212279379367828f, + -0.2279999703168869f, + -0.1577337533235550f, + -0.0445968918502331f, + 0.1069696992635727f, + 0.2067739963531494f, + 0.2106968611478806f, + 0.1397782564163208f + }; + + for (int i = 0; i < 10; i++) { + ASSERT(test_eq(GLM(perlin_vec4)(p1[i]), e[i])); + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, perlin_vec3) { + vec3 p1[] = { + {0.1f, 0.2f, 0.3f}, + {0.2f, 0.3f, 0.4f}, + {0.3f, 0.4f, 0.5f}, + {0.4f, 0.5f, 0.6f}, + {0.5f, 0.6f, 0.7f}, + {0.6f, 0.7f, 0.8f}, + {0.7f, 0.8f, 0.9f}, + {0.8f, 0.9f, 1.0f}, + {0.9f, 1.0f, 1.1f}, + {1.0f, 1.1f, 1.2f}, + }; + + /* expected values calculated by glm::perlin */ + float e[] = { + -0.2909241318702698f, + -0.4667602181434631f, + -0.4679279625415802f, + -0.2616460621356964f, + 0.0562822706997395f, + 0.3178773224353790f, + 0.3981811404228210f, + 0.3011017739772797f, + 0.1263920217752457f, + -0.0602480024099350f + }; + + for (int i = 0; i < 10; i++) { + ASSERT(test_eq(GLM(perlin_vec3)(p1[i]), e[i])); + } + + TEST_SUCCESS +} + + +TEST_IMPL(GLM_PREFIX, perlin_vec2) { + vec2 p1[] = { + {0.1f, 0.2f}, + {0.2f, 0.3f}, + {0.3f, 0.4f}, + {0.4f, 0.5f}, + {0.5f, 0.6f}, + {0.6f, 0.7f}, + {0.7f, 0.8f}, + {0.8f, 0.9f}, + {0.9f, 1.0f}, + {1.0f, 1.1f}, + }; + + /* expected values calculated by glm::perlin */ + float e[] = { + 0.2841092348098755f, + 0.2328013032674789f, + -0.0017980185803026f, + -0.3300299644470215f, + -0.5998955368995667f, + -0.6914522647857666f, + -0.5896517634391785f, + -0.3778679668903351f, + -0.1557840555906296f, + 0.0453133136034012f + }; + + for (int i = 0; i < 10; i++) { + ASSERT(test_eq(GLM(perlin_vec2)(p1[i]), e[i])); + } + + TEST_SUCCESS +} + + + + diff --git a/cglm/test/src/test_plane.h b/cglm/test/src/test_plane.h new file mode 100644 index 0000000..896c8b5 --- /dev/null +++ b/cglm/test/src/test_plane.h @@ -0,0 +1,39 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +TEST_IMPL(GLM_PREFIX, plane_normalize) { + vec4 p1 = {2.0f, -3.0f, 4.0f, 5.0f}, p2 = {2.0f, -3.0f, 4.0f, 5.0f}; + float s = 1.0f; + float norm; + + GLM(plane_normalize)(p2); + + norm = sqrtf(p1[0] * p1[0] + p1[1] * p1[1] + p1[2] * p1[2]); + if (norm == 0.0f) { + ASSERT(test_eq(p1[0], 0.0f)) + ASSERT(test_eq(p1[1], 0.0f)) + ASSERT(test_eq(p1[2], 0.0f)) + ASSERT(test_eq(p1[3], 0.0f)) + + TEST_SUCCESS + } + + norm = s / norm; + + ASSERT(test_eq(p1[0] * norm, p2[0])) + ASSERT(test_eq(p1[1] * norm, p2[1])) + ASSERT(test_eq(p1[2] * norm, p2[2])) + ASSERT(test_eq(p1[3] * norm, p2[3])) + + glm_vec4_zero(p1); + GLM(plane_normalize)(p1); + ASSERTIFY(test_assert_vec4_eq(p1, GLM_VEC4_ZERO)) + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_project.h b/cglm/test/src/test_project.h new file mode 100644 index 0000000..54fdb91 --- /dev/null +++ b/cglm/test/src/test_project.h @@ -0,0 +1,112 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +TEST_IMPL(GLM_PREFIX, unprojecti) { + mat4 model, view, proj, mvp; + vec4 viewport = {0.0f, 0.0f, 800.0f, 600.0f}; + vec3 pos = {13.0f, 45.0f, 0.74f}; + vec3 projected, unprojected; + + glm_translate_make(model, (vec3){0.0f, 0.0f, -10.0f}); + glm_lookat((vec3){0.0f, 0.0f, 0.0f}, pos, GLM_YUP, view); + + glm_perspective_default(0.5f, proj); + glm_mat4_mulN((mat4 *[]){&proj, &view, &model}, 3, mvp); + + GLM(project)(pos, mvp, viewport, projected); + + glm_mat4_inv(mvp, mvp); + GLM(unprojecti)(projected, mvp, viewport, unprojected); + + /* unprojected of projected vector must be same as original one */ + /* we used 0.01 because of projection floating point errors */ +#ifndef CGLM_FAST_MATH + ASSERT(fabsf(pos[0] - unprojected[0]) < 0.01) + ASSERT(fabsf(pos[1] - unprojected[1]) < 0.01) + ASSERT(fabsf(pos[2] - unprojected[2]) < 0.01) +#else + ASSERT(fabsf(pos[0] - unprojected[0]) < 0.1) + ASSERT(fabsf(pos[1] - unprojected[1]) < 0.1) + ASSERT(fabsf(pos[2] - unprojected[2]) < 0.1) +#endif + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, unproject) { + mat4 model, view, proj, mvp; + vec4 viewport = {0.0f, 0.0f, 800.0f, 600.0f}; + vec3 pos = {13.0f, 45.0f, 0.74f}; + vec3 projected, unprojected; + + glm_translate_make(model, (vec3){0.0f, 0.0f, -10.0f}); + glm_lookat((vec3){0.0f, 0.0f, 0.0f}, pos, GLM_YUP, view); + + glm_perspective_default(0.5f, proj); + glm_mat4_mulN((mat4 *[]){&proj, &view, &model}, 3, mvp); + + GLM(project)(pos, mvp, viewport, projected); + GLM(unproject)(projected, mvp, viewport, unprojected); + + /* unprojected of projected vector must be same as original one */ + /* we used 0.01 because of projection floating point errors */ + +#ifndef CGLM_FAST_MATH + ASSERT(fabsf(pos[0] - unprojected[0]) < 0.01) + ASSERT(fabsf(pos[1] - unprojected[1]) < 0.01) + ASSERT(fabsf(pos[2] - unprojected[2]) < 0.01) +#else + ASSERT(fabsf(pos[0] - unprojected[0]) < 0.1) + ASSERT(fabsf(pos[1] - unprojected[1]) < 0.1) + ASSERT(fabsf(pos[2] - unprojected[2]) < 0.1) +#endif + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, project) { + mat4 model, view, proj, mvp; + vec4 viewport = {0.0f, 0.0f, 800.0f, 600.0f}; + vec3 pos = {13.0f, 45.0f, 0.74f}; + vec3 projected, unprojected; + + glm_translate_make(model, (vec3){0.0f, 0.0f, -10.0f}); + glm_lookat((vec3){0.0f, 0.0f, 0.0f}, pos, GLM_YUP, view); + + glm_perspective_default(0.5f, proj); + glm_mat4_mulN((mat4 *[]){&proj, &view, &model}, 3, mvp); + + GLM(project)(pos, mvp, viewport, projected); + GLM(unproject)(projected, mvp, viewport, unprojected); + + /* unprojected of projected vector must be same as original one */ + /* we used 0.01 because of projection floating point errors */ + +#ifndef CGLM_FAST_MATH + ASSERT(fabsf(pos[0] - unprojected[0]) < 0.01) + ASSERT(fabsf(pos[1] - unprojected[1]) < 0.01) + ASSERT(fabsf(pos[2] - unprojected[2]) < 0.01) +#else + ASSERT(fabsf(pos[0] - unprojected[0]) < 0.1) + ASSERT(fabsf(pos[1] - unprojected[1]) < 0.1) + ASSERT(fabsf(pos[2] - unprojected[2]) < 0.1) +#endif + + /* test with no projection */ + glm_mat4_identity(mvp); + + GLM(project)(pos, mvp, viewport, projected); + GLM(unproject)(projected, mvp, viewport, unprojected); + + ASSERT(test_eq(pos[0], unprojected[0])) + ASSERT(test_eq(pos[1], unprojected[1])) + ASSERT(test_eq(pos[2], unprojected[2])) + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_quat.h b/cglm/test/src/test_quat.h new file mode 100644 index 0000000..da831fe --- /dev/null +++ b/cglm/test/src/test_quat.h @@ -0,0 +1,1108 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +#ifndef CGLM_TEST_QUAT_ONCE +#define CGLM_TEST_QUAT_ONCE + +/* Macros */ + +TEST_IMPL(MACRO_GLM_QUAT_IDENTITY_INIT) { + versor v = GLM_QUAT_IDENTITY_INIT; + + ASSERT(test_eq(v[0], 0.0f)) + ASSERT(test_eq(v[1], 0.0f)) + ASSERT(test_eq(v[2], 0.0f)) + ASSERT(test_eq(v[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_QUAT_IDENTITY) { + ASSERT(test_eq(GLM_QUAT_IDENTITY[0], 0.0f)) + ASSERT(test_eq(GLM_QUAT_IDENTITY[1], 0.0f)) + ASSERT(test_eq(GLM_QUAT_IDENTITY[2], 0.0f)) + ASSERT(test_eq(GLM_QUAT_IDENTITY[3], 1.0f)) + + TEST_SUCCESS +} + +#endif /* CGLM_TEST_QUAT_ONCE */ + +TEST_IMPL(GLM_PREFIX, quat_identity) { + versor a = GLM_QUAT_IDENTITY_INIT; + versor b = GLM_QUAT_IDENTITY_INIT; + versor c; + mat4 r; + + GLM(quat_identity)(c); + + ASSERTIFY(test_assert_quat_eq_identity(a)) + ASSERTIFY(test_assert_quat_eq_identity(b)) + ASSERTIFY(test_assert_quat_eq_identity(c)) + + glm_quat_identity(c); + ASSERT(test_eq(glm_quat_real(c), cosf(glm_rad(0.0f) * 0.5f))) + + glm_quat_mat4(c, r); + ASSERTIFY(test_assert_mat4_eq2(r, GLM_MAT4_IDENTITY, 0.000009f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_identity_array) { + size_t i, count; + versor quats[4] = { + {1.0f, 2.0f, 3.0f, 4.0f}, + {1.0f, 2.0f, 3.0f, 4.0f}, + {1.0f, 2.0f, 3.0f, 4.0f}, + {1.0f, 2.0f, 3.0f, 4.0f}, + }; + + count = 4; + + GLM(quat_identity_array)(quats, count); + + for (i = 0; i < count; i++) { + ASSERTIFY(test_assert_quat_eq_identity(quats[i])) + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_init) { + versor q1 = {1.0f, 2.0f, 3.0f, 4.0f}; + versor q2 = {1.0f, 2.0f, 3.0f, 4.0f}; + versor q3 = {1.0f, 2.0f, 3.0f, 4.0f}; + + GLM(quat_init)(q1, 10.0f, 11.0f, 12.0f, 13.0f); + GLM(quat_init)(q2, 100.0f, 110.0f, 120.0f, 130.0f); + GLM(quat_init)(q3, 1000.0f, 1100.0f, 1200.0f, 1300.0f); + + ASSERT(q1[0] == 10.0f) + ASSERT(q1[1] == 11.0f) + ASSERT(q1[2] == 12.0f) + ASSERT(q1[3] == 13.0f) + + ASSERT(q2[0] == 100.0f) + ASSERT(q2[1] == 110.0f) + ASSERT(q2[2] == 120.0f) + ASSERT(q2[3] == 130.0f) + + ASSERT(q3[0] == 1000.0f) + ASSERT(q3[1] == 1100.0f) + ASSERT(q3[2] == 1200.0f) + ASSERT(q3[3] == 1300.0f) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quatv) { + versor q1 = {1.0f, 2.0f, 3.0f, 4.0f}; + vec3 v1, v2; + float a1; + + test_rand_vec3(v1); + GLM(quatv)(q1, glm_rad(60.0f), v1); + + glm_quat_axis(q1, v2); + a1 = glm_quat_angle(q1); + + ASSERT(test_eq(a1, glm_rad(60.0f))) + + glm_vec3_normalize(v1); + ASSERTIFY(test_assert_vec3_eq(v1, v2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat) { + versor q1 = {1.0f, 2.0f, 3.0f, 4.0f}; + vec3 v1, v2; + float a1; + + test_rand_vec3(v1); + GLM(quat)(q1, glm_rad(60.0f), v1[0], v1[1], v1[2]); + + glm_quat_axis(q1, v2); + a1 = glm_quat_angle(q1); + + ASSERT(test_eq(a1, glm_rad(60.0f))) + + glm_vec3_normalize(v1); + ASSERTIFY(test_assert_vec3_eq(v1, v2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_copy) { + versor v1 = {10.0f, 9.0f, 8.0f, 78.0f}; + versor v2 = {1.0f, 2.0f, 3.0f, 4.0f}; + + GLM(quat_copy)(v1, v2); + + ASSERTIFY(test_assert_vec4_eq(v1, v2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_from_vecs) { + versor q1, q2, q3, q4, q5, q6, q7; + vec3 v1 = {1.f, 0.f, 0.f}, v2 = {1.f, 0.f, 0.f}; /* parallel */ + vec3 v3 = {0.f, 1.f, 0.f}, v4 = {1.f, 0.f, 0.f}; /* perpendicular */ + vec3 v5 = {0.f, 0.f, 1.f}, v6 = {0.f, 0.f, -1.f}; /* straight */ + vec3 v7, v8; /* random */ + vec3 v9 = {0.57735026f, 0.57735026f, 0.57735026f}, /* acute */ + v10 = {0.70710678f, 0.70710678f, 0.f}; + vec3 v11 = {0.87287156f, 0.21821789f, 0.43643578f}, /* obtuse */ + v12 = {-0.87287156f, 0.21821789f, 0.43643578f}; + vec3 v13 = GLM_VEC3_ZERO_INIT; /* zero */ + + GLM(quat_from_vecs)(v1, v2, q1); + ASSERTIFY(test_assert_quat_eq_identity(q1)) + + GLM(quat_from_vecs)(v3, v4, q2); + GLM(quat_rotatev)(q2, v3, v3); + ASSERT(test_eq(GLM(vec3_dot)(v3, v4), 1.f)) + ASSERT(test_eq(q2[0], 0.f)) + ASSERT(test_eq(q2[1], 0.f)) + ASSERT(test_eq(q2[2], -0.707106781187f)) + ASSERT(test_eq(q2[3], 0.707106781187f)) + + GLM(quat_from_vecs)(v5, v6, q3); + GLM(quat_rotatev)(q3, v5, v5); + ASSERT(test_eq(GLM(vec3_dot)(v5, v6), 1.f)) + ASSERT(test_eq(q3[0], 0.f)) + ASSERT(test_eq(q3[1], -1.f)) + ASSERT(test_eq(q3[2], 0.f)) + ASSERT(test_eq(q3[3], 0.f)) + + test_rand_vec3(v7); + test_rand_vec3(v8); + GLM(vec3_normalize(v7)); + GLM(vec3_normalize(v8)); + GLM(quat_from_vecs)(v7, v8, q4); + GLM(quat_rotatev)(q4, v7, v7); + ASSERT(test_eq(GLM(vec3_dot)(v7, v8), 1.f)) + + GLM(quat_from_vecs)(v9, v10, q5); + GLM(quat_rotatev)(q5, v9, v9); + ASSERT(test_eq(GLM(vec3_dot)(v9, v10), 1.f)) + + GLM(quat_from_vecs)(v11, v12, q6); + GLM(quat_rotatev)(q6, v11, v11); + ASSERT(test_eq(GLM(vec3_dot)(v11, v12), 1.f)) + + GLM(quat_from_vecs)(v13, v1, q7); + ASSERTIFY(test_assert_quat_eq_identity(q7)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_norm) { + versor a = {10.0f, 9.0f, 8.0f, 78.0f}; + float n1, n2; + + n1 = GLM(quat_norm)(a); + n2 = sqrtf(a[0] * a[0] + a[1] * a[1] + a[2] * a[2] + a[3] * a[3]); + + ASSERT(test_eq(n1, n2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_normalize_to) { + versor v1 = {2.0f, -3.0f, 4.0f, 5.0f}, v2; + float s = 1.0f; + float norm; + + GLM(quat_normalize_to)(v1, v2); + + norm = sqrtf(v1[0] * v1[0] + v1[1] * v1[1] + v1[2] * v1[2] + v1[3] * v1[3]); + if (norm <= 0.0f) { + ASSERTIFY(test_assert_quat_eq_identity(v1)) + + TEST_SUCCESS + } + + norm = s / norm; + + ASSERT(test_eq(v1[0] * norm, v2[0])) + ASSERT(test_eq(v1[1] * norm, v2[1])) + ASSERT(test_eq(v1[2] * norm, v2[2])) + ASSERT(test_eq(v1[3] * norm, v2[3])) + + glm_vec4_zero(v1); + GLM(quat_normalize_to)(v1, v2); + ASSERTIFY(test_assert_quat_eq_identity(v2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_normalize) { + versor v1 = {2.0f, -3.0f, 4.0f, 5.0f}, v2 = {2.0f, -3.0f, 4.0f, 5.0f}; + float s = 1.0f; + float norm; + + GLM(quat_normalize)(v2); + + norm = sqrtf(v1[0] * v1[0] + v1[1] * v1[1] + v1[2] * v1[2] + v1[3] * v1[3]); + if (norm <= 0.0f) { + ASSERTIFY(test_assert_quat_eq_identity(v1)) + + TEST_SUCCESS + } + + norm = s / norm; + + ASSERT(test_eq(v1[0] * norm, v2[0])) + ASSERT(test_eq(v1[1] * norm, v2[1])) + ASSERT(test_eq(v1[2] * norm, v2[2])) + ASSERT(test_eq(v1[3] * norm, v2[3])) + + glm_vec4_zero(v1); + GLM(quat_normalize)(v1); + ASSERTIFY(test_assert_quat_eq_identity(v1)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_dot) { + versor a = {10.0f, 9.0f, 8.0f, 78.0f}; + versor b = {1.0f, 2.0f, 3.0f, 4.0f}; + float dot1, dot2; + + dot1 = GLM(quat_dot)(a, b); + dot2 = a[0] * b[0] + a[1] * b[1] + a[2] * b[2] + a[3] * b[3]; + + ASSERT(test_eq(dot1, dot2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_conjugate) { + versor a = {10.0f, 9.0f, 8.0f, 78.0f}; + versor b = {1.0f, 2.0f, 3.0f, 4.0f}; + versor d, e; + + GLM(quat_conjugate)(a, d); + GLM(quat_conjugate)(b, e); + + ASSERT(test_eq(d[0], -a[0])) + ASSERT(test_eq(d[1], -a[1])) + ASSERT(test_eq(d[2], -a[2])) + ASSERT(test_eq(d[3], a[3])) + + ASSERT(test_eq(e[0], -b[0])) + ASSERT(test_eq(e[1], -b[1])) + ASSERT(test_eq(e[2], -b[2])) + ASSERT(test_eq(e[3], b[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_inv) { + versor a = {10.0f, 9.0f, 8.0f, 78.0f}; + versor b = {1.0f, 2.0f, 3.0f, 4.0f}; + versor d, e; + float n1, n2; + + n1 = 1.0f / glm_vec4_norm2(a); + n2 = 1.0f / glm_vec4_norm2(b); + + GLM(quat_inv)(a, d); + GLM(quat_inv)(b, e); + + ASSERT(test_eq(d[0], -a[0] * n1)) + ASSERT(test_eq(d[1], -a[1] * n1)) + ASSERT(test_eq(d[2], -a[2] * n1)) + ASSERT(test_eq(d[3], a[3] * n1)) + + ASSERT(test_eq(e[0], -b[0] * n2)) + ASSERT(test_eq(e[1], -b[1] * n2)) + ASSERT(test_eq(e[2], -b[2] * n2)) + ASSERT(test_eq(e[3], b[3] * n2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_add) { + versor a = {-10.0f, 9.0f, -8.0f, 56.0f}; + versor b = {12.0f, 19.0f, -18.0f, 1.0f}; + versor c, d; + + c[0] = a[0] + b[0]; + c[1] = a[1] + b[1]; + c[2] = a[2] + b[2]; + c[3] = a[3] + b[3]; + + GLM(quat_add)(a, b, d); + + ASSERTIFY(test_assert_quat_eq(c, d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_sub) { + vec4 a = {-10.0f, 9.0f, -8.0f, 56.0f}; + vec4 b = {12.0f, 19.0f, -18.0f, 1.0f}; + vec4 c, d; + + c[0] = a[0] - b[0]; + c[1] = a[1] - b[1]; + c[2] = a[2] - b[2]; + c[3] = a[3] - b[3]; + + GLM(quat_sub)(a, b, d); + + ASSERTIFY(test_assert_quat_eq(c, d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_real) { + versor a = {10.0f, 9.0f, 8.0f, 78.0f}; + versor b = {1.0f, 2.0f, 3.0f, 4.0f}; + + ASSERT(test_eq(GLM(quat_real)(a), 78.0f)) + ASSERT(test_eq(GLM(quat_real)(b), 4.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_imag) { + versor a = {10.0f, 9.0f, 8.0f, 78.0f}; + versor b = {1.0f, 2.0f, 3.0f, 4.0f}; + vec3 d, e; + + GLM(quat_imag)(a, d); + GLM(quat_imag)(b, e); + + ASSERT(test_eq(d[0], a[0])) + ASSERT(test_eq(d[1], a[1])) + ASSERT(test_eq(d[2], a[2])) + + ASSERT(test_eq(e[0], b[0])) + ASSERT(test_eq(e[1], b[1])) + ASSERT(test_eq(e[2], b[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_imagn) { + versor a = {10.0f, 9.0f, 8.0f, 78.0f}; + versor b = {1.0f, 2.0f, 3.0f, 4.0f}; + vec3 d, e; + + GLM(quat_imagn)(a, d); + GLM(quat_imagn)(b, e); + + glm_vec3_normalize(a); + glm_vec3_normalize(b); + glm_vec3_normalize(d); + glm_vec3_normalize(e); + + ASSERT(test_eq(d[0], a[0])) + ASSERT(test_eq(d[1], a[1])) + ASSERT(test_eq(d[2], a[2])) + + ASSERT(test_eq(e[0], b[0])) + ASSERT(test_eq(e[1], b[1])) + ASSERT(test_eq(e[2], b[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_imaglen) { + versor a = {10.0f, 9.0f, 8.0f, 78.0f}; + versor b = {1.0f, 2.0f, 3.0f, 4.0f}; + + ASSERT(test_eq(GLM(quat_imaglen)(a), glm_vec3_norm(a))); + ASSERT(test_eq(GLM(quat_imaglen)(b), glm_vec3_norm(b))); + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_angle) { + versor q1 = {1.0f, 2.0f, 3.0f, 4.0f}, q2, q3; + vec3 v1; + float a1, a2, a3; + + test_rand_vec3(v1); + GLM(quatv)(q1, glm_rad(60.140f), v1); + GLM(quatv)(q2, glm_rad(160.04f), v1); + GLM(quatv)(q3, glm_rad(20.350f), v1); + + a1 = GLM(quat_angle)(q1); + a2 = GLM(quat_angle)(q2); + a3 = GLM(quat_angle)(q3); + + ASSERT(test_eq(a1, glm_rad(60.140f))) + ASSERT(test_eq(a2, glm_rad(160.04f))) + ASSERT(test_eq(a3, glm_rad(20.350f))) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_axis) { + versor q1 = {1.0f, 2.0f, 3.0f, 4.0f}, q2, q3; + vec3 v1, v2; + + test_rand_vec3(v1); + GLM(quatv)(q1, glm_rad(60.0f), v1); + + glm_quat_axis(q1, v2); + glm_vec3_normalize(v1); + ASSERTIFY(test_assert_vec3_eq(v1, v2)) + + test_rand_vec3(v1); + GLM(quatv)(q2, glm_rad(60.0f), v1); + + glm_quat_axis(q2, v2); + glm_vec3_normalize(v1); + ASSERTIFY(test_assert_vec3_eq(v1, v2)) + + test_rand_vec3(v1); + GLM(quatv)(q3, glm_rad(60.0f), v1); + + glm_quat_axis(q3, v2); + glm_vec3_normalize(v1); + ASSERTIFY(test_assert_vec3_eq(v1, v2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_mul) { + versor q1 = {2.0f, 3.0f, 4.0f, 5.0f}; + versor q2 = {6.0f, 7.0f, 8.0f, 9.0f}; + versor q3; + versor q4; + vec3 v1 = {1.5f, 2.5f, 3.5f}; + + GLM(quat_mul)(q1, q2, q3); + + ASSERT(test_eq(q3[0], q1[3] * q2[0] + q1[0] * q2[3] + q1[1] * q2[2] - q1[2] * q2[1])) + ASSERT(test_eq(q3[1], q1[3] * q2[1] - q1[0] * q2[2] + q1[1] * q2[3] + q1[2] * q2[0])) + ASSERT(test_eq(q3[2], q1[3] * q2[2] + q1[0] * q2[1] - q1[1] * q2[0] + q1[2] * q2[3])) + ASSERT(test_eq(q3[3], q1[3] * q2[3] - q1[0] * q2[0] - q1[1] * q2[1] - q1[2] * q2[2])) + + glm_quatv(q1, glm_rad(30.0f), v1); + glm_quatv(q2, glm_rad(20.0f), v1); + glm_quatv(q3, glm_rad(50.0f), v1); + + GLM(quat_mul)(q1, q2, q4); + + ASSERTIFY(test_assert_quat_eq(q3, q4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_mat4) { + mat4 m1, m2; + versor q1, q2, q3; + vec3 axis1; + vec3 axis2 = {1.9f, 2.3f, 4.5f}; + int i; + + GLM(quat)(q1, GLM_PI_4f, 1.9f, 2.3f, 4.5f); + GLM(quat_mat4)(q1, m1); + GLM(mat4_quat)(m1, q2); + + GLM(rotate_make)(m2, GLM_PI_4f, axis2); + GLM(mat4_quat)(m1, q3); + + GLM(quat_axis)(q3, axis1); + + GLM(vec3_normalize)(axis1); + GLM(vec3_normalize)(axis2); + + ASSERT(test_eq(glm_quat_angle(q3), GLM_PI_4f)) + ASSERTIFY(test_assert_vec3_eq(axis1, axis2)) + ASSERTIFY(test_assert_vec4_eq(q1, q2)) + ASSERTIFY(test_assert_mat4_eq(m1, m2)) + ASSERTIFY(test_assert_vec4_eq(q1, q3)) + + /* 1. test quat to mat and mat to quat */ + for (i = 0; i < 1000; i++) { + test_rand_quat(q1); + + GLM(quat_mat4)(q1, m1); + GLM(mat4_quat)(m1, q2); + GLM(quat_mat4)(q2, m2); + + /* 2. test first quat and generated one equality */ + ASSERTIFY(test_assert_quat_eq_abs(q1, q2)); + + /* 3. test first rot and second rotation */ + /* almost equal */ + ASSERTIFY(test_assert_mat4_eq2(m1, m2, 0.000009f)); + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_mat4t) { + mat4 m1, m2; + versor q1, q2, q3; + vec3 axis1; + vec3 axis2 = {1.9f, 2.3f, 4.5f}; + int i; + + GLM(quat)(q1, GLM_PI_4f, 1.9f, 2.3f, 4.5f); + + GLM(quat_mat4t)(q1, m1); + glm_mat4_transpose(m1); + + GLM(mat4_quat)(m1, q2); + + GLM(rotate_make)(m2, GLM_PI_4f, axis2); + GLM(mat4_quat)(m1, q3); + + GLM(quat_axis)(q3, axis1); + + GLM(vec3_normalize)(axis1); + GLM(vec3_normalize)(axis2); + + ASSERT(test_eq(glm_quat_angle(q3), GLM_PI_4f)) + ASSERTIFY(test_assert_vec3_eq(axis1, axis2)) + ASSERTIFY(test_assert_vec4_eq(q1, q2)) + ASSERTIFY(test_assert_mat4_eq(m1, m2)) + ASSERTIFY(test_assert_vec4_eq(q1, q3)) + + /* 1. test quat to mat and mat to quat */ + for (i = 0; i < 1000; i++) { + test_rand_quat(q1); + + GLM(quat_mat4t)(q1, m1); + glm_mat4_transpose(m1); + + GLM(mat4_quat)(m1, q2); + + GLM(quat_mat4t)(q2, m2); + glm_mat4_transpose(m2); + + /* 2. test first quat and generated one equality */ + ASSERTIFY(test_assert_quat_eq_abs(q1, q2)); + + /* 3. test first rot and second rotation */ + /* almost equal */ + ASSERTIFY(test_assert_mat4_eq2(m1, m2, 0.000009f)); + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_mat3) { + mat4 m1, m2; + mat3 m3; + versor q1, q2, q3; + vec3 axis1; + vec3 axis2 = {1.9f, 2.3f, 4.5f}; + int i; + + GLM(quat)(q1, GLM_PI_4f, 1.9f, 2.3f, 4.5f); + GLM(quat_mat3)(q1, m3); + glm_mat4_identity(m1); + glm_mat4_ins3(m3, m1); + + GLM(mat4_quat)(m1, q2); + + GLM(rotate_make)(m2, GLM_PI_4f, axis2); + GLM(mat4_quat)(m1, q3); + + GLM(quat_axis)(q3, axis1); + + GLM(vec3_normalize)(axis1); + GLM(vec3_normalize)(axis2); + + ASSERT(test_eq(glm_quat_angle(q3), GLM_PI_4f)) + ASSERTIFY(test_assert_vec3_eq(axis1, axis2)) + ASSERTIFY(test_assert_vec4_eq(q1, q2)) + ASSERTIFY(test_assert_mat4_eq(m1, m2)) + ASSERTIFY(test_assert_vec4_eq(q1, q3)) + + /* 1. test quat to mat and mat to quat */ + for (i = 0; i < 1000; i++) { + test_rand_quat(q1); + + GLM(quat_mat3)(q1, m3); + glm_mat4_identity(m1); + glm_mat4_ins3(m3, m1); + + GLM(mat4_quat)(m1, q2); + + GLM(quat_mat3)(q2, m3); + glm_mat4_identity(m2); + glm_mat4_ins3(m3, m2); + + /* 2. test first quat and generated one equality */ + ASSERTIFY(test_assert_quat_eq_abs(q1, q2)); + + /* 3. test first rot and second rotation */ + /* almost equal */ + ASSERTIFY(test_assert_mat4_eq2(m1, m2, 0.000009f)); + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_mat3t) { + mat4 m1, m2; + mat3 m3; + versor q1, q2, q3; + vec3 axis1; + vec3 axis2 = {1.9f, 2.3f, 4.5f}; + int i; + + GLM(quat)(q1, GLM_PI_4f, 1.9f, 2.3f, 4.5f); + + GLM(quat_mat3t)(q1, m3); + glm_mat3_transpose(m3); + glm_mat4_identity(m1); + glm_mat4_ins3(m3, m1); + + GLM(mat4_quat)(m1, q2); + + GLM(rotate_make)(m2, GLM_PI_4f, axis2); + GLM(mat4_quat)(m1, q3); + + GLM(quat_axis)(q3, axis1); + + GLM(vec3_normalize)(axis1); + GLM(vec3_normalize)(axis2); + + ASSERT(test_eq(glm_quat_angle(q3), GLM_PI_4f)) + ASSERTIFY(test_assert_vec3_eq(axis1, axis2)) + ASSERTIFY(test_assert_vec4_eq(q1, q2)) + ASSERTIFY(test_assert_mat4_eq(m1, m2)) + ASSERTIFY(test_assert_vec4_eq(q1, q3)) + + /* 1. test quat to mat and mat to quat */ + for (i = 0; i < 1000; i++) { + test_rand_quat(q1); + + GLM(quat_mat3t)(q1, m3); + glm_mat3_transpose(m3); + glm_mat4_identity(m1); + glm_mat4_ins3(m3, m1); + + GLM(mat4_quat)(m1, q2); + + GLM(quat_mat3t)(q2, m3); + glm_mat3_transpose(m3); + glm_mat4_identity(m2); + glm_mat4_ins3(m3, m2); + + /* 2. test first quat and generated one equality */ + ASSERTIFY(test_assert_quat_eq_abs(q1, q2)); + + /* 3. test first rot and second rotation */ + /* almost equal */ + ASSERTIFY(test_assert_mat4_eq2(m1, m2, 0.000009f)); + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_lerp) { + versor v1 = {-100.0f, -200.0f, -10.0f, -10.0f}; + versor v2 = {100.0f, 200.0f, 10.0f, 10.0f}; + versor v3; + + GLM(quat_lerp)(v1, v2, 0.5f, v3); + ASSERT(test_eq(v3[0], 0.0f)) + ASSERT(test_eq(v3[1], 0.0f)) + ASSERT(test_eq(v3[2], 0.0f)) + ASSERT(test_eq(v3[3], 0.0f)) + + GLM(quat_lerp)(v1, v2, 0.75f, v3); + ASSERT(test_eq(v3[0], 50.0f)) + ASSERT(test_eq(v3[1], 100.0f)) + ASSERT(test_eq(v3[2], 5.0f)) + ASSERT(test_eq(v3[3], 5.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_lerpc) { + versor v1 = {-100.0f, -200.0f, -10.0f, -10.0f}; + versor v2 = {100.0f, 200.0f, 10.0f, 10.0f}; + versor v3; + + GLM(quat_lerpc)(v1, v2, 0.5f, v3); + ASSERT(test_eq(v3[0], 0.0f)) + ASSERT(test_eq(v3[1], 0.0f)) + ASSERT(test_eq(v3[2], 0.0f)) + ASSERT(test_eq(v3[3], 0.0f)) + + GLM(quat_lerpc)(v1, v2, 0.75f, v3); + ASSERT(test_eq(v3[0], 50.0f)) + ASSERT(test_eq(v3[1], 100.0f)) + ASSERT(test_eq(v3[2], 5.0f)) + ASSERT(test_eq(v3[3], 5.0f)) + + GLM(quat_lerpc)(v1, v2, -1.75f, v3); + ASSERT(test_eq(v3[0], -100.0f)) + ASSERT(test_eq(v3[1], -200.0f)) + ASSERT(test_eq(v3[2], -10.0f)) + ASSERT(test_eq(v3[3], -10.0f)) + + GLM(quat_lerpc)(v1, v2, 1.75f, v3); + ASSERT(test_eq(v3[0], 100.0f)) + ASSERT(test_eq(v3[1], 200.0f)) + ASSERT(test_eq(v3[2], 10.0f)) + ASSERT(test_eq(v3[3], 10.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_nlerp) { + versor q1, q2, q3, q4; + vec3 v1 = {10.0f, 0.0f, 0.0f}, v2; + + glm_quatv(q1, glm_rad(30.0f), v1); + glm_quatv(q2, glm_rad(90.0f), v1); + + GLM(quat_nlerp)(q1, q2, 1.0f, q3); + glm_quat_normalize(q2); + ASSERTIFY(test_assert_quat_eq(q2, q3)); + + glm_quatv(q1, glm_rad(30.001f), v1); + glm_quatv(q2, glm_rad(30.002f), v1); + GLM(quat_nlerp)(q1, q2, 0.7f, q3); + glm_quat_lerp(q1, q2, 0.7f, q4); + ASSERTIFY(test_assert_quat_eq(q3, q4)); + + glm_quatv(q1, glm_rad(30.0f), v1); + glm_quatv(q2, glm_rad(90.0f), v1); + GLM(quat_nlerp)(q1, q2, 0.5f, q3); + + glm_quat_axis(q3, v2); + glm_vec3_normalize(v1); + glm_vec3_normalize(v2); + + ASSERT(glm_quat_angle(q3) > glm_rad(30.0f)); + ASSERT(glm_quat_angle(q3) < glm_rad(90.0f)); + ASSERTIFY(test_assert_vec3_eq(v1, v2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_slerp) { + versor q1, q2, q3, q4; + vec3 v1 = {10.0f, 0.0f, 0.0f}, v2; + + glm_quatv(q1, glm_rad(30.0f), v1); + glm_quatv(q2, glm_rad(90.0f), v1); + + q1[0] = 10.0f; + GLM(quat_slerp)(q1, q2, 1.0f, q3); + ASSERTIFY(test_assert_quat_eq(q1, q3)); + + glm_quatv(q1, glm_rad(30.001f), v1); + glm_quatv(q2, glm_rad(30.002f), v1); + GLM(quat_slerp)(q1, q2, 0.7f, q3); + glm_quat_lerp(q1, q2, 0.7f, q4); + ASSERTIFY(test_assert_quat_eq(q3, q4)); + + glm_quatv(q1, glm_rad(30.0f), v1); + glm_quatv(q2, glm_rad(90.0f), v1); + GLM(quat_slerp)(q1, q2, 0.5f, q3); + + glm_quat_axis(q3, v2); + glm_vec3_normalize(v1); + glm_vec3_normalize(v2); + + ASSERT(glm_quat_angle(q3) > glm_rad(30.0f)); + ASSERT(glm_quat_angle(q3) < glm_rad(90.0f)); + ASSERTIFY(test_assert_vec3_eq(v1, v2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_look) { + versor q1; + vec3 v1 = {0.0f, 1.0f, 0.0f}; + mat4 m1, m2; + + glm_quat(q1, glm_rad(90.0f), 0.0f, 1.0f, 0.0f); + GLM(quat_look)(v1, q1, m1); + + glm_look(v1, (vec3){-1.0f, 0.0f, 0.0f}, GLM_YUP, m2); + ASSERTIFY(test_assert_mat4_eq(m1, m2)); + + glm_quat(q1, glm_rad(180.0f), 1.0f, 0.0f, 0.0f); + GLM(quat_look)(v1, q1, m1); + + glm_look(v1, (vec3){0.0f, 0.0f, 1.0f}, (vec3){0.0f, -1.0f, 0.0f}, m2); + + ASSERTIFY(test_assert_mat4_eq(m1, m2)); + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_for) { + versor q1, q2; + + glm_quat(q1, glm_rad(90.0f), 0.0f, 1.0f, 0.0f); + GLM(quat_for)((vec3){-1.0f, 0.0f, 0.0f}, (vec3){0.0f, 1.0f, 0.0f}, q2); + ASSERTIFY(test_assert_quat_eq(q1, q2)); + + glm_quat(q2, glm_rad(90.0f), 1.0f, 0.0f, 0.0f); + GLM(quat_for)((vec3){0.0f, 1.0f, 0.0f}, (vec3){0.0f, 0.0f, 1.0f}, q1); + ASSERTIFY(test_assert_quat_eq(q1, q2)); + + glm_quat(q2, glm_rad(180.0f), 1.0f, 0.0f, 0.0f); + GLM(quat_for)((vec3){0.0f, 0.0f, 1.0f}, (vec3){0.0f, -1.0f, 0.0f}, q1); + ASSERTIFY(test_assert_quat_eq(q1, q2)); + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_forp) { + versor q1, q2; + + glm_quat(q1, glm_rad(90.0f), 0.0f, 1.0f, 0.0f); + GLM(quat_forp)((vec3){2.0f, 0.0f, 0.0f}, + (vec3){1.0f, 0.0f, 0.0f}, + (vec3){0.0f, 1.0f, 0.0f}, + q2); + ASSERTIFY(test_assert_quat_eq(q1, q2)); + + glm_quat(q2, glm_rad(90.0f), 1.0f, 0.0f, 0.0f); + GLM(quat_forp)((vec3){0.0f, 1.0f, 0.0f}, + (vec3){0.0f, 2.0f, 0.0f}, + (vec3){0.0f, 0.0f, 1.0f}, + q1); + ASSERTIFY(test_assert_quat_eq(q1, q2)); + + glm_quat(q2, glm_rad(180.0f), 1.0f, 0.0f, 0.0f); + GLM(quat_forp)((vec3){0.0f, 1.0f, 1.0f}, + (vec3){0.0f, 1.0f, 2.0f}, + (vec3){0.0f, -1.0f, 0.0f}, + q1); + ASSERTIFY(test_assert_quat_eq(q1, q2)); + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_rotatev) { + vec3 v1 = {1.0f, 0.0f, 0.0f}, v2 = {1.0f, 1.0f, 1.0f}; + versor q; + + /* rotate X around Y = -Z */ + glm_quatv(q, GLM_PI_2f, GLM_YUP); + GLM(quat_rotatev)(q, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], -1.0f)) + + /* rotate -Z around X = Y */ + glm_quatv(q, GLM_PI_2f, GLM_XUP); + GLM(quat_rotatev)(q, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 1.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + /* rotate Y around Z = -X */ + glm_quatv(q, GLM_PI_2f, GLM_ZUP); + GLM(quat_rotatev)(q, v1, v1); + + ASSERT(test_eq(v1[0], -1.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + /* rotate v2 around Y by 90deg */ + glm_quatv(q, GLM_PI_2f, GLM_YUP); + GLM(quat_rotatev)(q, v2, v2); + + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], -1.0f)) + + /* rotate v2 around Y by 90deg */ + glm_quatv(q, GLM_PI_2f, GLM_YUP); + GLM(quat_rotatev)(q, v2, v2); + + ASSERT(test_eq(v2[0], -1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], -1.0f)) + + /* rotate v2 around Y by 90deg */ + glm_quatv(q, GLM_PI_2f, GLM_YUP); + GLM(quat_rotatev)(q, v2, v2); + + ASSERT(test_eq(v2[0], -1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + /* rotate v2 around X by 90deg */ + glm_quatv(q, GLM_PI_2f, GLM_XUP); + GLM(quat_rotatev)(q, v2, v2); + + ASSERT(test_eq(v2[0], -1.0f)) + ASSERT(test_eq(v2[1], -1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + /* rotate v2 around Z by 90deg */ + glm_quatv(q, GLM_PI_2f, GLM_ZUP); + GLM(quat_rotatev)(q, v2, v2); + + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], -1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_rotate) { + mat4 m1 = GLM_MAT4_IDENTITY_INIT, m2; + versor q1; + vec4 v1 = {1.0f, 0.0f, 0.0f, 1.0f}; + + /* rotate X around Y = -Z */ + glm_quatv(q1, GLM_PI_2f, GLM_YUP); + GLM(quat_rotate)(m1, q1, m1); + glm_rotate_make(m2, GLM_PI_2f, GLM_YUP); + ASSERTIFY(test_assert_mat4_eq(m1, m2)) + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], -1.0f)) + + glm_mat4_identity(m1); + glm_mat4_identity(m2); + + /* rotate -Z around X = Y */ + glm_quatv(q1, GLM_PI_2f, GLM_XUP); + GLM(quat_rotate)(m1, q1, m1); + glm_rotate(m2, GLM_PI_2f, GLM_XUP); + ASSERTIFY(test_assert_mat4_eq(m1, m2)) + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 1.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + glm_mat4_identity(m1); + glm_mat4_identity(m2); + + /* rotate Y around X = +Z */ + glm_quatv(q1, GLM_PI_2f, GLM_XUP); + GLM(quat_rotate)(m1, q1, m1); + glm_rotate(m2, GLM_PI_2f, GLM_XUP); + ASSERTIFY(test_assert_mat4_eq(m1, m2)) + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_rotate_at) { + mat4 m1 = GLM_MAT4_IDENTITY_INIT; + versor q1; + vec4 v1 = {1.0f, 0.0f, 0.0f, 1.0f}; + + glm_quatv(q1, GLM_PI_2f, GLM_YUP); + GLM(quat_rotate_at)(m1, q1, (vec3){0.5f, 0.0f, 0.0f}); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.5f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], -0.5f)) + + glm_mat4_identity(m1); + + glm_quatv(q1, GLM_PI_2f, GLM_ZUP); + GLM(quat_rotate_at)(m1, q1, (vec3){0.0f, 0.0f, 0.0f}); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.5f)) + ASSERT(test_eq(v1[2], -0.5f)) + + glm_mat4_identity(m1); + + v1[0] = 1.0f; + v1[1] = 1.0f; + v1[2] = 1.0f; + + glm_quatv(q1, GLM_PI_2f, GLM_XUP); + GLM(quat_rotate_at)(m1, q1, GLM_VEC3_ZERO); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 1.0f)) + ASSERT(test_eq(v1[1], -1.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_rotate_atm) { + mat4 m1 = GLM_MAT4_IDENTITY_INIT; + versor q1; + vec4 v1 = {1.0f, 0.0f, 0.0f, 1.0f}; + + glm_quatv(q1, GLM_PI_2f, GLM_YUP); + GLM(quat_rotate_atm)(m1, q1, (vec3){0.5f, 0.0f, 0.0f}); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.5f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], -0.5f)) + + glm_quatv(q1, GLM_PI_2f, GLM_ZUP); + GLM(quat_rotate_atm)(m1, q1, (vec3){0.0f, 0.0f, 0.0f}); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.5f)) + ASSERT(test_eq(v1[2], -0.5f)) + + v1[0] = 1.0f; + v1[1] = 1.0f; + v1[2] = 1.0f; + + glm_quatv(q1, GLM_PI_2f, GLM_XUP); + GLM(quat_rotate_atm)(m1, q1, GLM_VEC3_ZERO); + glm_mat4_mulv(m1, v1, v1); + + ASSERT(test_eq(v1[0], 1.0f)) + ASSERT(test_eq(v1[1], -1.0f)) + ASSERT(test_eq(v1[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, quat_make) { + versor dest[3]; + float src[12] = { + 7.2f, 1.0f, 2.5f, 6.1f, + 0.2f, 2.8f, 17.3f, 5.1f, + 4.2f, 7.3f, 6.6f, 8.8f + }; + + float *srcp = src; + unsigned int i, j; + + for (i = 0, j = 0; i < sizeof(src) / sizeof(float); i+=4,j++) { + GLM(quat_make)(srcp + i, dest[j]); + ASSERT(test_eq(src[ i ], dest[j][0])); + ASSERT(test_eq(src[i+1], dest[j][1])); + ASSERT(test_eq(src[i+2], dest[j][2])); + ASSERT(test_eq(src[i+3], dest[j][3])); + } + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_ray.h b/cglm/test/src/test_ray.h new file mode 100644 index 0000000..ac47685 --- /dev/null +++ b/cglm/test/src/test_ray.h @@ -0,0 +1,73 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +TEST_IMPL(GLM_PREFIX, ray_triangle) { + /* Check whether a simple hit is recognized with the right distance */ + vec3 origin = { 0.0f, 0.0f, 0.0f}; + vec3 direction = { 1.0f, 0.0f, 0.0f}; + vec3 opposite = {-1.0f, 0.0f, 0.0f}; + vec3 v0 = { 5.0f, -1.0f, 1.0f}; + vec3 v1 = { 5.0f, -1.0f, -1.0f}; + vec3 v2 = { 5.0f, 1.0f, 0.0f}; + float d; + bool hit; + + hit = GLM(ray_triangle)(origin, direction, v0, v1, v2, &d); + ASSERT(hit); + ASSERT(fabsf(d - 5.0f) <= 0.0000009); + + /* Check whether a simple miss works */ + hit = GLM(ray_triangle)(origin, opposite, v0, v1, v2, &d); + ASSERT(!hit); + + /* Check that we can disregard distance and pass NULL pointer instead */ + hit = GLM(ray_triangle)(origin, direction, v0, v1, v2, NULL); + ASSERT(hit); + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ray_sphere) { + vec4 sphere = {5.0f, 0.0f, 0.0f, 1.0f}; /* Sphere: center at (5, 0, 0) with radius 1 */ + float t1, t2; + bool hit; + + /* Case 1: Ray misses the sphere */ + hit = GLM(ray_sphere)((vec3){10.0f, 3.0f, 0.0f}, (vec3){1.0f, 0.0f, 0.0f}, sphere, &t1, &t2); + ASSERT(!hit); /* Expect no intersection */ + + /* Case 2: Ray starts inside the sphere */ + hit = GLM(ray_sphere)((vec3){5.0f, 0.5f, 0.0f}, (vec3){1.0f, 0.0f, 0.0f}, sphere, &t1, &t2); + ASSERT(hit); /* Expect an intersection */ + ASSERT(t1 < 0 && t2 > 0); /* Ray exits at t2 */ + + /* Case 3: Ray intersects the sphere from outside */ + hit = GLM(ray_sphere)((vec3){0.0f, 0.0f, 0.0f}, (vec3){1.0f, 0.0f, 0.0f}, sphere, &t1, &t2); + ASSERT(hit); /* Expect an intersection */ + ASSERT(t1 > 0 && t2 > 0); /* Intersections at t1 and t2 */ + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, ray_at) { + vec3 origin = {0.0f, 0.0f, 0.0f}; + vec3 direction = {1.0f, 1.0f, 1.0f}; /* Diagonal direction */ + float distance = sqrtf(3.0f); /* Distance along the ray; sqrt(3) for unit length due to direction normalization */ + vec3 result; + + /* Normalize the direction to ensure accurate distance measurement */ + glm_vec3_normalize(direction); + + GLM(ray_at)(origin, direction, distance, result); + ASSERT(fabsf(result[0] - 1.0f) <= 0.0000009); /* Expecting to be 1 unit along the x-axis */ + ASSERT(fabsf(result[1] - 1.0f) <= 0.0000009); /* Expecting to be 1 unit along the y-axis */ + ASSERT(fabsf(result[2] - 1.0f) <= 0.0000009); /* Expecting to be 1 unit along the z-axis */ + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_struct.c b/cglm/test/src/test_struct.c new file mode 100644 index 0000000..d040225 --- /dev/null +++ b/cglm/test/src/test_struct.c @@ -0,0 +1,150 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +TEST_IMPL(mat2x3s_zero_init) { + mat2x3s mat2x3_zero = GLMS_MAT2X3_ZERO_INIT; + test_assert_mat2x3_eq_zero(mat2x3_zero.raw); + TEST_SUCCESS +} + +TEST_IMPL(mat2x3s_zero) { + mat2x3s mat2x3_zero = GLMS_MAT2X3_ZERO; + test_assert_mat2x3_eq_zero(mat2x3_zero.raw); + TEST_SUCCESS +} + +TEST_IMPL(mat2x4s_zero_init) { + mat2x4s mat2x4_zero = GLMS_MAT2X4_ZERO_INIT; + test_assert_mat2x4_eq_zero(mat2x4_zero.raw); + TEST_SUCCESS +} + +TEST_IMPL(mat2x4s_zero) { + mat2x4s mat2x4_zero = GLMS_MAT2X4_ZERO; + test_assert_mat2x4_eq_zero(mat2x4_zero.raw); + TEST_SUCCESS +} + +TEST_IMPL(mat3s_identity_init) { + mat3s mat3_identity = GLMS_MAT3_IDENTITY_INIT; + mat3 mat3_identity_a = GLM_MAT3_IDENTITY_INIT; + test_assert_mat3_eq(mat3_identity.raw, mat3_identity_a); + TEST_SUCCESS +} + +TEST_IMPL(mat3s_zero_init) { + mat3s mat3_zero = GLMS_MAT3_ZERO_INIT; + mat3 mat3_zero_a = GLM_MAT3_ZERO_INIT; + test_assert_mat3_eq(mat3_zero.raw, mat3_zero_a); + TEST_SUCCESS +} + +TEST_IMPL(mat3x2s_zero_init) { + mat3x2s mat3x2_zero = GLMS_MAT3X2_ZERO_INIT; + test_assert_mat3x2_eq_zero(mat3x2_zero.raw); + TEST_SUCCESS +} + +TEST_IMPL(mat3x2s_zero) { + mat3x2s mat3x2_zero = GLMS_MAT3X2_ZERO; + test_assert_mat3x2_eq_zero(mat3x2_zero.raw); + TEST_SUCCESS +} + +TEST_IMPL(mat3x4s_zero_init) { + mat3x4s mat3x4_zero = GLMS_MAT3X4_ZERO_INIT; + test_assert_mat3x4_eq_zero(mat3x4_zero.raw); + TEST_SUCCESS +} + +TEST_IMPL(mat3x4s_zero) { + mat3x4s mat3x4_zero = GLMS_MAT3X4_ZERO; + test_assert_mat3x4_eq_zero(mat3x4_zero.raw); + TEST_SUCCESS +} + +TEST_IMPL(mat4s_identity_init) { + mat4s mat4_identity = GLMS_MAT4_IDENTITY_INIT; + mat4 mat4_identity_a = GLM_MAT4_IDENTITY_INIT; + test_assert_mat4_eq(mat4_identity.raw, mat4_identity_a); + TEST_SUCCESS +} + +TEST_IMPL(mat4s_zero_init) { + mat4s mat4_zero = GLMS_MAT4_ZERO_INIT; + mat4 mat4_zero_a = GLM_MAT4_ZERO_INIT; + test_assert_mat4_eq(mat4_zero.raw, mat4_zero_a); + TEST_SUCCESS +} + +TEST_IMPL(mat4x2s_zero_init) { + mat4x2s mat4x2_zero = GLMS_MAT4X2_ZERO_INIT; + test_assert_mat4x2_eq_zero(mat4x2_zero.raw); + TEST_SUCCESS +} + +TEST_IMPL(mat4x2s_zero) { + mat4x2s mat4x2_zero = GLMS_MAT4X2_ZERO; + test_assert_mat4x2_eq_zero(mat4x2_zero.raw); + TEST_SUCCESS +} + +TEST_IMPL(mat4x3s_zero_init) { + mat4x3s mat4x3_zero = GLMS_MAT4X3_ZERO_INIT; + test_assert_mat4x3_eq_zero(mat4x3_zero.raw); + TEST_SUCCESS +} + +TEST_IMPL(mat4x3s_zero) { + mat4x3s mat4x3_zero = GLMS_MAT4X3_ZERO; + test_assert_mat4x3_eq_zero(mat4x3_zero.raw); + TEST_SUCCESS +} + +TEST_IMPL(quats_zero_init) { + versors quat_zero = GLMS_QUAT_IDENTITY_INIT; + versor quat_zero_a = GLM_QUAT_IDENTITY_INIT; + test_assert_quat_eq(quat_zero.raw, quat_zero_a); + TEST_SUCCESS +} + +TEST_IMPL(vec3s_one_init) { + vec3s vec3_one = GLMS_VEC3_ONE_INIT; + vec3 vec3_one_a = GLM_VEC3_ONE_INIT; + test_assert_vec3_eq(vec3_one.raw, vec3_one_a); + TEST_SUCCESS +} + +TEST_IMPL(vec3s_zero_init) { + vec3s vec3_zero = GLMS_VEC3_ZERO_INIT; + vec3 vec3_zero_a = GLM_VEC3_ZERO_INIT; + test_assert_vec3_eq(vec3_zero.raw, vec3_zero_a); + TEST_SUCCESS +} + +TEST_IMPL(vec4s_black_init) { + vec4s vec4_black = GLMS_VEC4_BLACK_INIT; + vec4 vec4_black_a = GLM_VEC4_BLACK_INIT; + test_assert_vec4_eq(vec4_black.raw, vec4_black_a); + TEST_SUCCESS +} + +TEST_IMPL(vec4s_one_init) { + vec4s vec4_one = GLMS_VEC4_ONE_INIT; + vec4 vec4_one_a = GLM_VEC4_ONE_INIT; + test_assert_vec4_eq(vec4_one.raw, vec4_one_a); + TEST_SUCCESS +} + +TEST_IMPL(vec4s_zero_init) { + vec4s vec4_zero = GLMS_VEC4_ZERO_INIT; + vec4 vec4_zero_a = GLM_VEC4_ZERO_INIT; + test_assert_vec4_eq(vec4_zero.raw, vec4_zero_a); + TEST_SUCCESS +} diff --git a/cglm/test/src/test_vec2.h b/cglm/test/src/test_vec2.h new file mode 100644 index 0000000..026c387 --- /dev/null +++ b/cglm/test/src/test_vec2.h @@ -0,0 +1,951 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +#ifndef CGLM_TEST_VEC2_ONCE +#define CGLM_TEST_VEC2_ONCE + +/* Macros */ + +TEST_IMPL(MACRO_GLM_VEC2_ONE_INIT) { + vec2 v = GLM_VEC2_ONE_INIT; + + ASSERT(test_eq(v[0], 1.0f)) + ASSERT(test_eq(v[1], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_VEC2_ZERO_INIT) { + vec2 v = GLM_VEC2_ZERO_INIT; + + ASSERT(test_eq(v[0], 0.0f)) + ASSERT(test_eq(v[1], 0.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_VEC2_ONE) { + ASSERT(test_eq(GLM_VEC2_ONE[0], 1.0f)) + ASSERT(test_eq(GLM_VEC2_ONE[1], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_VEC2_ZERO) { + ASSERT(test_eq(GLM_VEC2_ZERO[0], 0.0f)) + ASSERT(test_eq(GLM_VEC2_ZERO[0], 0.0f)) + + TEST_SUCCESS +} + +#endif /* CGLM_TEST_VEC2_ONCE */ + +TEST_IMPL(GLM_PREFIX, vec2) { + vec4 v4 = {10.0f, 9.0f, 8.0f, 7.0f}; + vec3 v3 = {11.0f, 12.0f, 13.0f}; + vec2 v2; + + GLM(vec2)(v4, v2); + ASSERT(test_eq(v2[0], v4[0])) + ASSERT(test_eq(v2[1], v4[1])) + + GLM(vec2)(v3, v2); + ASSERT(test_eq(v2[0], v3[0])) + ASSERT(test_eq(v2[1], v3[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_copy) { + vec2 v1 = {10.0f, 9.0f}; + vec2 v2 = {1.0f, 2.0f}; + + GLM(vec2_copy)(v1, v2); + + ASSERTIFY(test_assert_vec2_eq(v1, v2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_zero) { + vec2 v1 = {10.0f, 9.0f}; + vec2 v2 = {1.0f, 2.0f}; + + GLM(vec2_zero)(v1); + GLM(vec2_zero)(v2); + + ASSERTIFY(test_assert_vec2_eq(v1, GLM_VEC2_ZERO)) + ASSERTIFY(test_assert_vec2_eq(v2, GLM_VEC2_ZERO)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_one) { + vec2 v1 = {10.0f, 9.0f}; + vec2 v2 = {1.0f, 2.0f}; + + GLM(vec2_one)(v1); + GLM(vec2_one)(v2); + + ASSERTIFY(test_assert_vec2_eq(v1, GLM_VEC2_ONE)) + ASSERTIFY(test_assert_vec2_eq(v2, GLM_VEC2_ONE)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_dot) { + vec2 a = {10.0f, 9.0f}; + vec2 b = {1.0f, 2.0f}; + float dot1, dot2; + + dot1 = GLM(vec2_dot)(a, b); + dot2 = a[0] * b[0] + a[1] * b[1]; + + ASSERT(test_eq(dot1, dot2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_cross) { + vec2 a = {10.0f, 9.0f}; + vec2 b = {1.0f, 2.0f}; + float cprod; + + cprod = a[0] * b[1] - a[1] * b[0]; + + ASSERT(test_eq(glm_vec2_cross(a, b), cprod)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_norm2) { + vec2 a = {10.0f, 9.0f}; + float n1, n2; + + n1 = GLM(vec2_norm2)(a); + n2 = a[0] * a[0] + a[1] * a[1]; + + ASSERT(test_eq(n1, n2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_norm) { + vec2 a = {10.0f, 9.0f}; + float n1, n2; + + n1 = GLM(vec2_norm)(a); + n2 = sqrtf(a[0] * a[0] + a[1] * a[1]); + + ASSERT(test_eq(n1, n2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_add) { + vec2 a = {-10.0f, 9.0f}; + vec2 b = {12.0f, 19.0f}; + vec2 c, d; + + c[0] = a[0] + b[0]; + c[1] = a[1] + b[1]; + + GLM(vec2_add)(a, b, d); + + ASSERTIFY(test_assert_vec2_eq(c, d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_adds) { + vec4 a = {-10.0f, 9.0f}; + vec4 c, d; + float s = 7.0f; + + c[0] = a[0] + s; + c[1] = a[1] + s; + + GLM(vec2_adds)(a, s, d); + + ASSERTIFY(test_assert_vec2_eq(c, d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_sub) { + vec2 a = {-10.0f, 9.0f}; + vec2 b = {12.0f, 19.0f}; + vec2 c, d; + + c[0] = a[0] - b[0]; + c[1] = a[1] - b[1]; + + GLM(vec2_sub)(a, b, d); + + ASSERTIFY(test_assert_vec2_eq(c, d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_subs) { + vec2 a = {-10.0f, 9.0f}; + vec2 c, d; + float s = 7.0f; + + c[0] = a[0] - s; + c[1] = a[1] - s; + + GLM(vec2_subs)(a, s, d); + + ASSERTIFY(test_assert_vec2_eq(c, d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_mul) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {-3.0f, 4.0f}, + v3; + + GLM(vec2_mul)(v1, v2, v3); + + ASSERT(test_eq(v1[0] * v2[0], v3[0])) + ASSERT(test_eq(v1[1] * v2[1], v3[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_scale) { + vec2 v1 = {2.0f, -3.0f}, v2; + float s = 7.0f; + + GLM(vec2_scale)(v1, s, v2); + + ASSERT(test_eq(v1[0] * s, v2[0])) + ASSERT(test_eq(v1[1] * s, v2[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_scale_as) { + vec2 v1 = {2.0f, -3.0f}, v2; + float s = 7.0f; + float norm; + + GLM(vec2_scale_as)(v1, s, v2); + + norm = sqrtf(v1[0] * v1[0] + v1[1] * v1[1]); + if (norm < FLT_EPSILON) { + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + + TEST_SUCCESS + } + + norm = s / norm; + + ASSERT(test_eq(v1[0] * norm, v2[0])) + ASSERT(test_eq(v1[1] * norm, v2[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_div) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {-3.0f, 4.0f}, + v3; + + GLM(vec2_div)(v1, v2, v3); + + ASSERT(test_eq(v1[0] / v2[0], v3[0])) + ASSERT(test_eq(v1[1] / v2[1], v3[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_divs) { + vec2 v1 = {2.0f, -3.0f}, v2; + float s = 7.0f; + + GLM(vec2_divs)(v1, s, v2); + + ASSERT(test_eq(v1[0] / s, v2[0])) + ASSERT(test_eq(v1[1] / s, v2[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_addadd) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {-3.0f, 4.0f}, + v3 = {1.0f, 2.0f}, + v4 = {1.0f, 2.0f}; + + GLM(vec2_addadd)(v1, v2, v4); + + ASSERT(test_eq(v3[0] + v1[0] + v2[0], v4[0])) + ASSERT(test_eq(v3[1] + v1[1] + v2[1], v4[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_subadd) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {-3.0f, 4.0f}, + v3 = {1.0f, 2.0f}, + v4 = {1.0f, 2.0f}; + + GLM(vec2_subadd)(v1, v2, v4); + + ASSERT(test_eq(v3[0] + v1[0] - v2[0], v4[0])) + ASSERT(test_eq(v3[1] + v1[1] - v2[1], v4[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_muladd) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {-3.0f, 4.0f}, + v3 = {1.0f, 2.0f}, + v4 = {1.0f, 2.0f}; + + GLM(vec2_muladd)(v1, v2, v4); + + ASSERT(test_eq(v3[0] + v1[0] * v2[0], v4[0])) + ASSERT(test_eq(v3[1] + v1[1] * v2[1], v4[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_muladds) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {1.0f, 2.0f}, + v3 = {1.0f, 2.0f}; + float s = 9.0f; + + GLM(vec2_muladds)(v1, s, v3); + + ASSERT(test_eq(v2[0] + v1[0] * s, v3[0])) + ASSERT(test_eq(v2[1] + v1[1] * s, v3[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_maxadd) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {-3.0f, 4.0f}, + v3 = {1.0f, 2.0f}, + v4 = {1.0f, 2.0f}; + + GLM(vec2_maxadd)(v1, v2, v4); + + ASSERT(test_eq(v3[0] + glm_max(v1[0], v2[0]), v4[0])) + ASSERT(test_eq(v3[1] + glm_max(v1[1], v2[1]), v4[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_minadd) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {-3.0f, 4.0f}, + v3 = {1.0f, 2.0f}, + v4 = {1.0f, 2.0f}; + + GLM(vec2_minadd)(v1, v2, v4); + + ASSERT(test_eq(v3[0] + glm_min(v1[0], v2[0]), v4[0])) + ASSERT(test_eq(v3[1] + glm_min(v1[1], v2[1]), v4[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_subsub) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {-3.0f, 4.0f}, + v3 = {1.0f, 2.0f}, + v4 = {1.0f, 2.0f}; + + GLM(vec2_subsub)(v1, v2, v4); + + ASSERT(test_eq(v3[0] - (v1[0] - v2[0]), v4[0])) + ASSERT(test_eq(v3[1] - (v1[1] - v2[1]), v4[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_addsub) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {-3.0f, 4.0f}, + v3 = {1.0f, 2.0f}, + v4 = {1.0f, 2.0f}; + + GLM(vec2_addsub)(v1, v2, v4); + + ASSERT(test_eq(v3[0] - (v1[0] + v2[0]), v4[0])) + ASSERT(test_eq(v3[1] - (v1[1] + v2[1]), v4[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_mulsub) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {-3.0f, 4.0f}, + v3 = {1.0f, 2.0f}, + v4 = {1.0f, 2.0f}; + + GLM(vec2_mulsub)(v1, v2, v4); + + ASSERT(test_eq(v3[0] - v1[0] * v2[0], v4[0])) + ASSERT(test_eq(v3[1] - v1[1] * v2[1], v4[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_mulsubs) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {1.0f, 2.0f}, + v3 = {1.0f, 2.0f}; + float s = 9.0f; + + GLM(vec2_mulsubs)(v1, s, v3); + + ASSERT(test_eq(v2[0] - v1[0] * s, v3[0])) + ASSERT(test_eq(v2[1] - v1[1] * s, v3[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_maxsub) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {-3.0f, 4.0f}, + v3 = {1.0f, 2.0f}, + v4 = {1.0f, 2.0f}; + + GLM(vec2_maxsub)(v1, v2, v4); + + ASSERT(test_eq(v3[0] - glm_max(v1[0], v2[0]), v4[0])) + ASSERT(test_eq(v3[1] - glm_max(v1[1], v2[1]), v4[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_minsub) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {-3.0f, 4.0f}, + v3 = {1.0f, 2.0f}, + v4 = {1.0f, 2.0f}; + + GLM(vec2_minsub)(v1, v2, v4); + + ASSERT(test_eq(v3[0] - glm_min(v1[0], v2[0]), v4[0])) + ASSERT(test_eq(v3[1] - glm_min(v1[1], v2[1]), v4[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_negate_to) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {-3.0f, 4.0f}, + v3, v4; + + GLM(vec2_negate_to)(v1, v3); + GLM(vec2_negate_to)(v2, v4); + + ASSERT(test_eq(-v1[0], v3[0])) + ASSERT(test_eq(-v1[1], v3[1])) + + ASSERT(test_eq(-v2[0], v4[0])) + ASSERT(test_eq(-v2[1], v4[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_negate) { + vec2 v1 = {2.0f, -3.0f}, + v2 = {-3.0f, 4.0f}, + v3 = {2.0f, -3.0f}, + v4 = {-3.0f, 4.0f}; + + GLM(vec2_negate)(v1); + GLM(vec2_negate)(v2); + + ASSERT(test_eq(-v1[0], v3[0])) + ASSERT(test_eq(-v1[1], v3[1])) + + ASSERT(test_eq(-v2[0], v4[0])) + ASSERT(test_eq(-v2[1], v4[1])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_normalize) { + vec2 v1 = {2.0f, -3.0f}, v2 = {2.0f, -3.0f}; + float s = 1.0f; + float norm; + + GLM(vec2_normalize)(v2); + + norm = sqrtf(v1[0] * v1[0] + v1[1] * v1[1]); + if (norm < FLT_EPSILON) { + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + + TEST_SUCCESS + } + + norm = s / norm; + + ASSERT(test_eq(v1[0] * norm, v2[0])) + ASSERT(test_eq(v1[1] * norm, v2[1])) + + glm_vec2_zero(v1); + GLM(vec2_normalize)(v1); + ASSERTIFY(test_assert_vec2_eq(v1, GLM_VEC2_ZERO)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_normalize_to) { + vec2 v1 = {2.0f, -3.0f}, v2; + float s = 1.0f; + float norm; + + GLM(vec2_normalize_to)(v1, v2); + + norm = sqrtf(v1[0] * v1[0] + v1[1] * v1[1]); + if (norm < FLT_EPSILON) { + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + + TEST_SUCCESS + } + + norm = s / norm; + + ASSERT(test_eq(v1[0] * norm, v2[0])) + ASSERT(test_eq(v1[1] * norm, v2[1])) + + glm_vec2_zero(v1); + GLM(vec2_normalize_to)(v1, v2); + ASSERTIFY(test_assert_vec2_eq(v2, GLM_VEC2_ZERO)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_rotate) { + vec2 v1 = {1.0f, 0.0f}; + + GLM(vec2_rotate)(v1, GLM_PI_2f, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 1.0f)) + + GLM(vec2_rotate)(v1, GLM_PI_2f, v1); + + ASSERT(test_eq(v1[0], -1.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + + GLM(vec2_rotate)(v1, GLM_PI_2f, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], -1.0f)) + + GLM(vec2_rotate)(v1, GLM_PI_2f, v1); + + ASSERT(test_eq(v1[0], 1.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_center) { + vec2 v1 = {1.0f, 1.0f}, + v2 = {0.0f, 0.0f}; + vec2 dest; + GLM(vec2_center)(v1, v2, dest); + + ASSERTIFY(test_assert_vec2_eq(dest, (vec2){ 0.5f, 0.5f })) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_distance2) { + vec2 v1 = {30.0f, 0.0f}, + v2 = {0.0f, 0.0f}, + v3 = {3.0f, 10.0f}, + v4 = {0.46f, 4.0f}; + float d; + + d = GLM(vec2_distance2)(v1, v2); + ASSERT(test_eq(d, 30.0f * 30.0f)) + + d = GLM(vec2_distance2)(v3, v4); + ASSERT(test_eq(powf(v3[0] - v4[0], 2.0f) + + powf(v3[1] - v4[1], 2.0f), d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_distance) { + vec2 v1 = {30.0f, 0.0f}, + v2 = {0.0f, 0.0f}, + v3 = {3.0f, 10.0f}, + v4 = {0.46f, 4.0f}; + float d; + + d = GLM(vec2_distance)(v1, v2); + ASSERT(test_eq(d, 30.0f)) + + d = GLM(vec2_distance)(v3, v4); + ASSERT(test_eq(sqrtf(powf(v3[0] - v4[0], 2.0f) + + powf(v3[1] - v4[1], 2.0f)), d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_maxv) { + vec2 v1, v2, v3; + vec2 v5 = {-1.456f, -1.456f}; + vec2 v6 = {11.0f, 11.0f}; + vec2 v7 = {78.0f, -78.0f}; + + GLM(vec2_maxv)(v5, v6, v1); + GLM(vec2_maxv)(v5, v7, v2); + GLM(vec2_maxv)(v6, v7, v3); + + ASSERT(test_eq(v1[0], 11.0f)) + ASSERT(test_eq(v1[1], 11.0f)) + + ASSERT(test_eq(v2[0], 78.0f)) + ASSERT(test_eq(v2[1], -1.456f)) + + ASSERT(test_eq(v3[0], 78.0f)) + ASSERT(test_eq(v3[1], 11.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_minv) { + vec2 v1, v2, v3; + vec2 v5 = {-1.456f, -1.456f}; + vec2 v6 = {11.0f, 11.0f}; + vec2 v7 = {78.0f, -78.0f}; + + GLM(vec2_minv)(v5, v6, v1); + GLM(vec2_minv)(v5, v7, v2); + GLM(vec2_minv)(v6, v7, v3); + + ASSERT(test_eq(v1[0], -1.456f)) + ASSERT(test_eq(v1[1], -1.456f)) + + ASSERT(test_eq(v2[0], -1.456f)) + ASSERT(test_eq(v2[1], -78.0f)) + + ASSERT(test_eq(v3[0], 11.0f)) + ASSERT(test_eq(v3[1], -78.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_clamp) { + vec2 v1 = {-1.456f, -11.456f}; + vec2 v2 = {0.110f, 111.0f}; + vec2 v3 = {78.0f, 32.0f}; + + GLM(vec2_clamp)(v1, -1.03f, 30.0f); + GLM(vec2_clamp)(v2, 0.11f, 111.0f); + GLM(vec2_clamp)(v3, -88.0f, 70.0f); + + ASSERT(test_eq(v1[0], -1.03f)) + ASSERT(test_eq(v1[1], -1.03f)) + + ASSERT(test_eq(v2[0], 0.11f)) + ASSERT(test_eq(v2[1], 111.0f)) + + ASSERT(test_eq(v3[0], 70.0f)) + ASSERT(test_eq(v3[1], 32.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_abs) { + vec2 v1 = {2, -3}, v2 = {-12, -31}; + vec2 v3, v4; + vec2 v5 = {2, 3}, v6 = {12, 31}; + + GLM(vec2_abs)(v1, v3); + GLM(vec2_abs)(v2, v4); + + ASSERTIFY(test_assert_vec2_eq(v3, v5)) + ASSERTIFY(test_assert_vec2_eq(v4, v6)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_fract) { + vec2 v1 = {2.104f, 3.012f}, v2 = {12.35f, 31.140f}, v3, v4; + vec2 v5 = {0.104f, 0.012f}, v6 = {0.35f, 0.140f}; + + GLM(vec2_fract)(v1, v3); + GLM(vec2_fract)(v2, v4); + + ASSERTIFY(test_assert_vec2_eq(v3, v5)) + ASSERTIFY(test_assert_vec2_eq(v4, v6)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_floor) { + vec2 v1 = {2.104f, 3.012f}, v2 = {12.35f, 31.140f}, v3, v4; + vec2 v5 = {2.0f, 3.0f}, v6 = {12.0f, 31.0f}; + + GLM(vec2_floor)(v1, v3); + GLM(vec2_floor)(v2, v4); + + ASSERTIFY(test_assert_vec2_eq(v3, v5)) + ASSERTIFY(test_assert_vec2_eq(v4, v6)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_mods) { + vec2 v1 = {2.104f, 3.012f}, v2 = {12.35f, 31.140f}, v3, v4; + vec2 v5 = {0.104f, 0.012f}, v6 = {0.35f, 0.140f}; + + /* Mod 1 - leaves just the fractional part */ + GLM(vec2_mods)(v1, 1.0f, v3); + GLM(vec2_mods)(v2, 1.0f, v4); + + ASSERTIFY(test_assert_vec2_eq(v3, v5)) + ASSERTIFY(test_assert_vec2_eq(v4, v6)) + + /* Mod 2 - parity + fractional part */ + GLM(vec2_mods)(v1, 2.0f, v3); + GLM(vec2_mods)(v2, 2.0f, v4); + + vec2 v7 = {0.104f, 1.012f}, v8 = {0.35f, 1.140f}; + + ASSERTIFY(test_assert_vec2_eq(v3, v7)) + ASSERTIFY(test_assert_vec2_eq(v4, v8)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_swizzle) { + vec2 v; + + v[0] = 1; + v[1] = 2; + + GLM(vec2_swizzle)(v, GLM_SHUFFLE2(1, 0), v); + ASSERTIFY(test_assert_vec2_eq(v, (vec2){2, 1})) + + GLM(vec2_swizzle)(v, GLM_SHUFFLE2(0, 0), v); + ASSERTIFY(test_assert_vec2_eq(v, (vec2){1, 1})) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_lerp) { + vec2 v1 = {-100.0f, -200.0f}; + vec2 v2 = {100.0f, 200.0f}; + vec2 v3; + + GLM(vec2_lerp)(v1, v2, 0.5f, v3); + ASSERT(test_eq(v3[0], 0.0f)) + ASSERT(test_eq(v3[1], 0.0f)) + + GLM(vec2_lerp)(v1, v2, 0.75f, v3); + ASSERT(test_eq(v3[0], 50.0f)) + ASSERT(test_eq(v3[1], 100.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_steps) { + vec2 v1 = {-100.0f, -200.0f}; + vec2 v2; + + GLM(vec2_steps)(-2.5f, v1, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + + GLM(vec2_steps)(-150.0f, v1, v2); + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + + GLM(vec2_steps)(-300.0f, v1, v2); + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_stepr) { + vec2 v1 = {-2.5f, -150.0f}; + vec2 v2; + + GLM(vec2_stepr)(v1, -200.0f, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + + GLM(vec2_stepr)(v1, -150.0f, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + + GLM(vec2_stepr)(v1, 0.0f, v2); + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_step) { + vec2 v1 = {-100.0f, -200.0f}; + vec2 s1 = {-100.0f, 0.0f}; + vec2 s2 = {100.0f, -220.0f}; + vec2 s3 = {100.0f, 200.0f}; + vec2 v2; + + GLM(vec2_step)(s1, v1, v2); + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + + GLM(vec2_step)(s2, v1, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + + GLM(vec2_step)(s3, v1, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_complex_mul) { + vec2 v1 = { 3.0f, 5.0f }, + v2 = { 7.0f, 11.0f }, + v3 = { cosf(GLM_PIf/4.0f), sinf(GLM_PIf/4.0f) }; + + GLM(vec2_complex_mul)(v1, v2, v2); + ASSERTIFY(test_assert_vec2_eq(v2, (vec2){ -34, 68 })) + + GLM(vec2_complex_mul)(v3, v3, v3); + ASSERTIFY(test_assert_vec2_eq(v3, (vec2){ 0.0f, 1.0f })) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_complex_div) { + vec2 v1 = { -34.0f, 68.0f }, + v2 = { 3.0f, 5.0f }, + v3 = { cosf(GLM_PIf/4.0f), sinf(GLM_PIf/4.0f) }, + v4 = { cosf(GLM_PIf/4.0f), -sinf(GLM_PIf/4.0f) }; + + GLM(vec2_complex_div)(v1, v2, v2); + ASSERTIFY(test_assert_vec2_eq(v2, (vec2){ 7.0f, 11.0f })) + + GLM(vec2_complex_div)(v3, v4, v4); + ASSERTIFY(test_assert_vec2_eq(v4, (vec2){ 0.0f, 1.0f })) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_make) { + float src[6] = { + 7.2f, 1.0f, + 2.5f, 6.1f, + 17.7f, 4.3f + }; + vec2 dest[3]; + + float *srcp = src; + unsigned int i, j; + + for (i = 0, j = 0; i < sizeof(src) / sizeof(float); i+=2,j++) { + GLM(vec2_make)(srcp + i, dest[j]); + ASSERT(test_eq(src[ i ], dest[j][0])); + ASSERT(test_eq(src[i+1], dest[j][1])); + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_reflect) { + vec2 dest; + + /* Reflecting off a "horizontal" surface in 2D */ + vec2 I1 = {1.0f, -1.0f}; /* Incoming vector */ + vec2 N1 = {0.0f, 1.0f}; /* Normal vector */ + GLM(vec2_reflect)(I1, N1, dest); + ASSERT(fabsf(dest[0] - 1.0f) < 0.00001f && + fabsf(dest[1] - 1.0f) < 0.00001f); /* Expect reflection upwards */ + + /* Reflecting at an angle in 2D */ + vec2 I2 = {sqrtf(2)/2, -sqrtf(2)/2}; /* Incoming vector at 45 degrees */ + vec2 N2 = {0.0f, 1.0f}; /* Upwards normal vector */ + GLM(vec2_reflect)(I2, N2, dest); + ASSERT(fabsf(dest[0] - sqrtf(2)/2) < 0.00001f && + fabsf(dest[1] - sqrtf(2)/2) < 0.00001f); /* Expect reflection upwards */ + + /* Reflecting off a line in 2D representing a "vertical" surface analogy */ + vec2 I3 = {1.0f, 0.0f}; /* Incoming vector */ + vec2 N3 = {-1.0f, 0.0f}; /* Normal vector representing a "vertical" line */ + GLM(vec2_reflect)(I3, N3, dest); + ASSERT(fabsf(dest[0] + 1.0f) < 0.00001f && + fabsf(dest[1]) < 0.00001f); /* Expect reflection to the left */ + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec2_refract) { + vec2 v = {sqrtf(0.5f), -sqrtf(0.5f)}; /* Incoming vector at 45 degrees to normal */ + vec2 N = {0.0f, 1.0f}; /* Surface normal */ + vec2 dest; + float eta; + float r; + + /* Water to Air (eta = 1.33/1.0) */ + eta = 1.33f / 1.0f; + r = GLM(vec2_refract)(v, N, eta, dest); + // In 2D, we expect a similar bending behavior as in 3D, so we check dest[1] + if (!(dest[0] == 0.0f && dest[1] == 0.0f)) { + ASSERT(dest[1] < -0.3f); // Refracted ray bends away from the normal + ASSERT(r == true); + } else { + ASSERT(dest[0] == 0.0f && dest[1] == 0.0f); // Total internal reflection + ASSERT(r == false); + } + + /* Air to Glass (eta = 1.0 / 1.5) */ + eta = 1.0f / 1.5f; + r = GLM(vec2_refract)(v, N, eta, dest); + ASSERT(r == true); + ASSERT(dest[1] < -sqrtf(0.5f)); // Expect bending towards the normal + + /* Glass to Water (eta = 1.5 / 1.33) */ + eta = 1.5f / 1.33f; + r = GLM(vec2_refract)(v, N, eta, dest); + ASSERT(r == true); + ASSERT(dest[1] < -0.6f); // Expect bending towards the normal, less bending than air to glass + + /* Diamond to Air (eta = 2.42 / 1.0) */ + eta = 2.42f / 1.0f; + r = GLM(vec2_refract)(v, N, eta, dest); + if (!(dest[0] == 0.0f && dest[1] == 0.0f)) { + /* High potential for total internal reflection, but if it occurs, expect significant bending */ + ASSERT(dest[1] < -sqrtf(0.5f)); + ASSERT(r == true); + } else { + ASSERT(dest[0] == 0.0f && dest[1] == 0.0f); // Total internal reflection + ASSERT(r == false); + } + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_vec3.h b/cglm/test/src/test_vec3.h new file mode 100644 index 0000000..4d963d5 --- /dev/null +++ b/cglm/test/src/test_vec3.h @@ -0,0 +1,2032 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +#define TEST_GLM_SHUFFLE3(z, y, x) (((z) << 4) | ((y) << 2) | (x)) + +#ifndef CGLM_TEST_VEC3_ONCE +#define CGLM_TEST_VEC3_ONCE + +/* Macros */ + +TEST_IMPL(MACRO_GLM_VEC3_ONE_INIT) { + vec3 v = GLM_VEC3_ONE_INIT; + + ASSERT(test_eq(v[0], 1.0f)) + ASSERT(test_eq(v[1], 1.0f)) + ASSERT(test_eq(v[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_VEC3_ZERO_INIT) { + vec3 v = GLM_VEC3_ZERO_INIT; + + ASSERT(test_eq(v[0], 0.0f)) + ASSERT(test_eq(v[1], 0.0f)) + ASSERT(test_eq(v[2], 0.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_VEC3_ONE) { + ASSERT(test_eq(GLM_VEC3_ONE[0], 1.0f)) + ASSERT(test_eq(GLM_VEC3_ONE[1], 1.0f)) + ASSERT(test_eq(GLM_VEC3_ONE[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_VEC3_ZERO) { + ASSERT(test_eq(GLM_VEC3_ZERO[0], 0.0f)) + ASSERT(test_eq(GLM_VEC3_ZERO[1], 0.0f)) + ASSERT(test_eq(GLM_VEC3_ZERO[2], 0.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_YUP) { + ASSERT(test_eq(GLM_YUP[0], 0.0f)) + ASSERT(test_eq(GLM_YUP[1], 1.0f)) + ASSERT(test_eq(GLM_YUP[2], 0.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_ZUP) { + ASSERT(test_eq(GLM_ZUP[0], 0.0f)) + ASSERT(test_eq(GLM_ZUP[1], 0.0f)) + ASSERT(test_eq(GLM_ZUP[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_XUP) { + ASSERT(test_eq(GLM_XUP[0], 1.0f)) + ASSERT(test_eq(GLM_XUP[1], 0.0f)) + ASSERT(test_eq(GLM_XUP[2], 0.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_FORWARD_RH) { + ASSERT(test_eq(GLM_FORWARD[0], 0.0f)) + ASSERT(test_eq(GLM_FORWARD[1], 0.0f)) + ASSERT(test_eq(GLM_FORWARD[2], -1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_SHUFFLE3) { + ASSERT(TEST_GLM_SHUFFLE3(1, 0, 0) == GLM_SHUFFLE3(1, 0, 0)) + ASSERT(TEST_GLM_SHUFFLE3(0, 1, 0) == GLM_SHUFFLE3(0, 1, 0)) + ASSERT(TEST_GLM_SHUFFLE3(0, 0, 1) == GLM_SHUFFLE3(0, 0, 1)) + ASSERT(TEST_GLM_SHUFFLE3(1, 0, 0) == GLM_SHUFFLE3(1, 0, 0)) + ASSERT(TEST_GLM_SHUFFLE3(1, 0, 1) == GLM_SHUFFLE3(1, 0, 1)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_XXX) { + ASSERT(TEST_GLM_SHUFFLE3(0, 0, 0) == GLM_XXX) + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_YYY) { + ASSERT(TEST_GLM_SHUFFLE3(1, 1, 1) == GLM_YYY) + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_ZZZ) { + ASSERT(TEST_GLM_SHUFFLE3(2, 2, 2) == GLM_ZZZ) + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_ZYX) { + ASSERT(TEST_GLM_SHUFFLE3(0, 1, 2) == GLM_ZYX) + TEST_SUCCESS +} + +/* Deprecated */ + +TEST_IMPL(MACRO_glm_vec3_dup) { + vec3 v1 = {13.0f, 12.0f, 11.0f}, v2; + + glm_vec3_dup(v1, v2); + + ASSERTIFY(test_assert_vec3_eq(v1, v2)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_glm_vec3_flipsign) { + vec3 v1 = {13.0f, -12.0f, 11.0f}, + v2 = {13.0f, -12.0f, 11.0f}, + v3 = {-13.0f, 12.0f, -11.0f}; + + glm_vec3_flipsign(v1); + glmc_vec3_flipsign(v2); + + ASSERTIFY(test_assert_vec3_eq(v1, v3)) + ASSERTIFY(test_assert_vec3_eq(v2, v3)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_glm_vec3_flipsign_to) { + vec3 v1 = {13.0f, -12.0f, 11.0f}, + v2 = {-13.0f, 12.0f, -11.0f}, + v3, v4; + + glm_vec3_flipsign_to(v1, v3); + glmc_vec3_flipsign_to(v1, v4); + + ASSERTIFY(test_assert_vec3_eq(v2, v3)) + ASSERTIFY(test_assert_vec3_eq(v2, v4)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_glm_vec3_inv) { + vec3 v1 = {13.0f, -12.0f, 11.0f}, + v2 = {13.0f, -12.0f, 11.0f}, + v3 = {-13.0f, 12.0f, -11.0f}; + + glm_vec3_inv(v1); + glmc_vec3_inv(v2); + + ASSERTIFY(test_assert_vec3_eq(v1, v3)) + ASSERTIFY(test_assert_vec3_eq(v2, v3)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_glm_vec3_inv_to) { + vec3 v1 = {13.0f, -12.0f, 11.0f}, + v2 = {-13.0f, 12.0f, -11.0f}, + v3, v4; + + glm_vec3_inv_to(v1, v3); + glmc_vec3_inv_to(v1, v4); + + ASSERTIFY(test_assert_vec3_eq(v3, v4)) + ASSERTIFY(test_assert_vec3_eq(v2, v3)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_glm_vec3_mulv) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f}, + v3, v4; + + glm_vec3_mulv(v1, v2, v3); + glmc_vec3_mulv(v1, v2, v4); + + ASSERTIFY(test_assert_vec3_eq(v3, v4)) + + ASSERT(test_eq(v1[0] * v2[0], v3[0])) + ASSERT(test_eq(v1[1] * v2[1], v3[1])) + ASSERT(test_eq(v1[2] * v2[2], v3[2])) + + TEST_SUCCESS +} + +#endif /* CGLM_TEST_VEC3_ONCE */ + +/* --- */ + +TEST_IMPL(GLM_PREFIX, vec3) { + vec4 v4 = {10.0f, 9.0f, 8.0f, 7.0f}; + vec3 v3; + + GLM(vec3)(v4, v3); + + ASSERT(test_eq(v3[0], v4[0])) + ASSERT(test_eq(v3[1], v4[1])) + ASSERT(test_eq(v3[2], v4[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_copy) { + vec3 v1 = {10.0f, 9.0f, 8.0f}; + vec3 v2 = {1.0f, 2.0f, 3.0f}; + + GLM(vec3_copy)(v1, v2); + + ASSERTIFY(test_assert_vec3_eq(v1, v2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_zero) { + vec3 v1 = {10.0f, 9.0f, 8.0f}; + vec3 v2 = {1.0f, 2.0f, 3.0f}; + + GLM(vec3_zero)(v1); + GLM(vec3_zero)(v2); + + ASSERTIFY(test_assert_vec3_eq(v1, GLM_VEC3_ZERO)) + ASSERTIFY(test_assert_vec3_eq(v2, GLM_VEC3_ZERO)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_one) { + vec3 v1 = {10.0f, 9.0f, 8.0f}; + vec3 v2 = {1.0f, 2.0f, 3.0f}; + + GLM(vec3_one)(v1); + GLM(vec3_one)(v2); + + ASSERTIFY(test_assert_vec3_eq(v1, GLM_VEC3_ONE)) + ASSERTIFY(test_assert_vec3_eq(v1, GLM_VEC3_ONE)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_dot) { + vec3 a = {10.0f, 9.0f, 8.0f}; + vec3 b = {1.0f, 2.0f, 3.0f}; + float dot1, dot2; + + dot1 = GLM(vec3_dot)(a, b); + dot2 = a[0] * b[0] + a[1] * b[1] + a[2] * b[2]; + + ASSERT(test_eq(dot1, dot2)) + + TEST_SUCCESS +} + + +TEST_IMPL(GLM_PREFIX, dot) { + /* SAME AS VEC3_DOT */ + + vec3 a = {10.0f, 9.0f, 8.0f}; + vec3 b = {1.0f, 2.0f, 3.0f}; + float dot1, dot2; + + dot1 = GLM(vec3_dot)(a, b); + dot2 = a[0] * b[0] + a[1] * b[1] + a[2] * b[2]; + + ASSERT(test_eq(dot1, dot2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_norm2) { + vec3 a = {10.0f, 9.0f, 8.0f}; + float n1, n2; + + n1 = GLM(vec3_norm2)(a); + n2 = a[0] * a[0] + a[1] * a[1] + a[2] * a[2]; + + ASSERT(test_eq(n1, n2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_norm) { + vec3 a = {10.0f, 9.0f, 8.0f}; + float n1, n2; + + n1 = GLM(vec3_norm)(a); + n2 = sqrtf(a[0] * a[0] + a[1] * a[1] + a[2] * a[2]); + + ASSERT(test_eq(n1, n2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_norm_one) { + vec3 a = {-10.0f, 9.0f, -8.0f}; + float n1, n2; + + n1 = GLM(vec3_norm_one)(a); + n2 = fabsf(a[0]) + fabsf(a[1]) + fabsf(a[2]); + + ASSERT(test_eq(n1, n2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_norm_inf) { + vec3 a = {-10.0f, 9.0f, -8.0f}; + float n1, n2; + + n1 = GLM(vec3_norm_inf)(a); + n2 = fabsf(a[0]); + + if (n2 < fabsf(a[1])) + n2 = fabsf(a[1]); + + if (n2 < fabsf(a[2])) + n2 = fabsf(a[2]); + + ASSERT(test_eq(n1, n2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_add) { + vec3 a = {-10.0f, 9.0f, -8.0f}; + vec3 b = {12.0f, 19.0f, -18.0f}; + vec3 c, d; + + c[0] = a[0] + b[0]; + c[1] = a[1] + b[1]; + c[2] = a[2] + b[2]; + + GLM(vec3_add)(a, b, d); + + ASSERTIFY(test_assert_vec3_eq(c, d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_adds) { + vec3 a = {-10.0f, 9.0f, -8.0f}; + vec3 c, d; + float s = 7.0f; + + c[0] = a[0] + s; + c[1] = a[1] + s; + c[2] = a[2] + s; + + GLM(vec3_adds)(a, s, d); + + ASSERTIFY(test_assert_vec3_eq(c, d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_sub) { + vec3 a = {-10.0f, 9.0f, -8.0f}; + vec3 b = {12.0f, 19.0f, -18.0f}; + vec3 c, d; + + c[0] = a[0] - b[0]; + c[1] = a[1] - b[1]; + c[2] = a[2] - b[2]; + + GLM(vec3_sub)(a, b, d); + + ASSERTIFY(test_assert_vec3_eq(c, d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_subs) { + vec3 a = {-10.0f, 9.0f, -8.0f}; + vec3 c, d; + float s = 7.0f; + + c[0] = a[0] - s; + c[1] = a[1] - s; + c[2] = a[2] - s; + + GLM(vec3_subs)(a, s, d); + + ASSERTIFY(test_assert_vec3_eq(c, d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_mul) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f}, + v3; + + GLM(vec3_mul)(v1, v2, v3); + + ASSERT(test_eq(v1[0] * v2[0], v3[0])) + ASSERT(test_eq(v1[1] * v2[1], v3[1])) + ASSERT(test_eq(v1[2] * v2[2], v3[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_scale) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, v2; + float s = 7.0f; + + GLM(vec3_scale)(v1, s, v2); + + ASSERT(test_eq(v1[0] * s, v2[0])) + ASSERT(test_eq(v1[1] * s, v2[1])) + ASSERT(test_eq(v1[2] * s, v2[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_scale_as) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, v2; + float s = 7.0f; + float norm; + + GLM(vec3_scale_as)(v1, s, v2); + + norm = sqrtf(v1[0] * v1[0] + v1[1] * v1[1] + v1[2] * v1[2]); + if (norm < FLT_EPSILON) { + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + TEST_SUCCESS + } + + norm = s / norm; + + ASSERT(test_eq(v1[0] * norm, v2[0])) + ASSERT(test_eq(v1[1] * norm, v2[1])) + ASSERT(test_eq(v1[2] * norm, v2[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_div) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f}, + v3; + + GLM(vec3_div)(v1, v2, v3); + + ASSERT(test_eq(v1[0] / v2[0], v3[0])) + ASSERT(test_eq(v1[1] / v2[1], v3[1])) + ASSERT(test_eq(v1[2] / v2[2], v3[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_divs) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, v2; + float s = 7.0f; + + GLM(vec3_divs)(v1, s, v2); + + ASSERT(test_eq(v1[0] / s, v2[0])) + ASSERT(test_eq(v1[1] / s, v2[1])) + ASSERT(test_eq(v1[2] / s, v2[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_addadd) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f}, + v3 = {1.0f, 2.0f, 3.0f}, + v4 = {1.0f, 2.0f, 3.0f}; + + GLM(vec3_addadd)(v1, v2, v4); + + ASSERT(test_eq(v3[0] + v1[0] + v2[0], v4[0])) + ASSERT(test_eq(v3[1] + v1[1] + v2[1], v4[1])) + ASSERT(test_eq(v3[2] + v1[2] + v2[2], v4[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_subadd) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f}, + v3 = {1.0f, 2.0f, 3.0f}, + v4 = {1.0f, 2.0f, 3.0f}; + + GLM(vec3_subadd)(v1, v2, v4); + + ASSERT(test_eq(v3[0] + v1[0] - v2[0], v4[0])) + ASSERT(test_eq(v3[1] + v1[1] - v2[1], v4[1])) + ASSERT(test_eq(v3[2] + v1[2] - v2[2], v4[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_muladd) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f}, + v3 = {1.0f, 2.0f, 3.0f}, + v4 = {1.0f, 2.0f, 3.0f}; + + GLM(vec3_muladd)(v1, v2, v4); + + ASSERT(test_eq(v3[0] + v1[0] * v2[0], v4[0])) + ASSERT(test_eq(v3[1] + v1[1] * v2[1], v4[1])) + ASSERT(test_eq(v3[2] + v1[2] * v2[2], v4[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_muladds) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {1.0f, 2.0f, 3.0f}, + v3 = {1.0f, 2.0f, 3.0f}; + float s = 9.0f; + + GLM(vec3_muladds)(v1, s, v3); + + ASSERT(test_eq(v2[0] + v1[0] * s, v3[0])) + ASSERT(test_eq(v2[1] + v1[1] * s, v3[1])) + ASSERT(test_eq(v2[2] + v1[2] * s, v3[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_maxadd) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f}, + v3 = {1.0f, 2.0f, 3.0f}, + v4 = {1.0f, 2.0f, 3.0f}; + + GLM(vec3_maxadd)(v1, v2, v4); + + ASSERT(test_eq(v3[0] + glm_max(v1[0], v2[0]), v4[0])) + ASSERT(test_eq(v3[1] + glm_max(v1[1], v2[1]), v4[1])) + ASSERT(test_eq(v3[2] + glm_max(v1[2], v2[2]), v4[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_minadd) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f}, + v3 = {1.0f, 2.0f, 3.0f}, + v4 = {1.0f, 2.0f, 3.0f}; + + GLM(vec3_minadd)(v1, v2, v4); + + ASSERT(test_eq(v3[0] + glm_min(v1[0], v2[0]), v4[0])) + ASSERT(test_eq(v3[1] + glm_min(v1[1], v2[1]), v4[1])) + ASSERT(test_eq(v3[2] + glm_min(v1[2], v2[2]), v4[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_subsub) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f}, + v3 = {1.0f, 2.0f, 3.0f}, + v4 = {1.0f, 2.0f, 3.0f}; + + GLM(vec3_subsub)(v1, v2, v4); + + ASSERT(test_eq(v3[0] - (v1[0] - v2[0]), v4[0])) + ASSERT(test_eq(v3[1] - (v1[1] - v2[1]), v4[1])) + ASSERT(test_eq(v3[2] - (v1[2] - v2[2]), v4[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_addsub) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f}, + v3 = {1.0f, 2.0f, 3.0f}, + v4 = {1.0f, 2.0f, 3.0f}; + + GLM(vec3_addsub)(v1, v2, v4); + + ASSERT(test_eq(v3[0] - (v1[0] + v2[0]), v4[0])) + ASSERT(test_eq(v3[1] - (v1[1] + v2[1]), v4[1])) + ASSERT(test_eq(v3[2] - (v1[2] + v2[2]), v4[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_mulsub) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f}, + v3 = {1.0f, 2.0f, 3.0f}, + v4 = {1.0f, 2.0f, 3.0f}; + + GLM(vec3_mulsub)(v1, v2, v4); + + ASSERT(test_eq(v3[0] - v1[0] * v2[0], v4[0])) + ASSERT(test_eq(v3[1] - v1[1] * v2[1], v4[1])) + ASSERT(test_eq(v3[2] - v1[2] * v2[2], v4[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_mulsubs) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {1.0f, 2.0f, 3.0f}, + v3 = {1.0f, 2.0f, 3.0f}; + float s = 9.0f; + + GLM(vec3_mulsubs)(v1, s, v3); + + ASSERT(test_eq(v2[0] - v1[0] * s, v3[0])) + ASSERT(test_eq(v2[1] - v1[1] * s, v3[1])) + ASSERT(test_eq(v2[2] - v1[2] * s, v3[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_maxsub) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f}, + v3 = {1.0f, 2.0f, 3.0f}, + v4 = {1.0f, 2.0f, 3.0f}; + + GLM(vec3_maxsub)(v1, v2, v4); + + ASSERT(test_eq(v3[0] - glm_max(v1[0], v2[0]), v4[0])) + ASSERT(test_eq(v3[1] - glm_max(v1[1], v2[1]), v4[1])) + ASSERT(test_eq(v3[2] - glm_max(v1[2], v2[2]), v4[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_minsub) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f}, + v3 = {1.0f, 2.0f, 3.0f}, + v4 = {1.0f, 2.0f, 3.0f}; + + GLM(vec3_minsub)(v1, v2, v4); + + ASSERT(test_eq(v3[0] - glm_min(v1[0], v2[0]), v4[0])) + ASSERT(test_eq(v3[1] - glm_min(v1[1], v2[1]), v4[1])) + ASSERT(test_eq(v3[2] - glm_min(v1[2], v2[2]), v4[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_negate_to) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f}, + v3, v4; + + GLM(vec3_negate_to)(v1, v3); + GLM(vec3_negate_to)(v2, v4); + + ASSERT(test_eq(-v1[0], v3[0])) + ASSERT(test_eq(-v1[1], v3[1])) + ASSERT(test_eq(-v1[2], v3[2])) + + ASSERT(test_eq(-v2[0], v4[0])) + ASSERT(test_eq(-v2[1], v4[1])) + ASSERT(test_eq(-v2[2], v4[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_negate) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f}, + v3 = {2.0f, -3.0f, 4.0f}, + v4 = {-3.0f, 4.0f, -5.0f}; + + GLM(vec3_negate)(v1); + GLM(vec3_negate)(v2); + + ASSERT(test_eq(-v1[0], v3[0])) + ASSERT(test_eq(-v1[1], v3[1])) + ASSERT(test_eq(-v1[2], v3[2])) + + ASSERT(test_eq(-v2[0], v4[0])) + ASSERT(test_eq(-v2[1], v4[1])) + ASSERT(test_eq(-v2[2], v4[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_normalize) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, v2 = {2.0f, -3.0f, 4.0f}; + float s = 1.0f; + float norm; + + GLM(vec3_normalize)(v2); + + norm = sqrtf(v1[0] * v1[0] + v1[1] * v1[1] + v1[2] * v1[2]); + if (norm < FLT_EPSILON) { + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + TEST_SUCCESS + } + + norm = s / norm; + + ASSERT(test_eq(v1[0] * norm, v2[0])) + ASSERT(test_eq(v1[1] * norm, v2[1])) + ASSERT(test_eq(v1[2] * norm, v2[2])) + + glm_vec3_zero(v1); + GLM(vec3_normalize)(v1); + ASSERTIFY(test_assert_vec3_eq(v1, GLM_VEC3_ZERO)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_normalize_to) { + vec3 v1 = {2.0f, -3.0f, 4.0f}, v2; + float s = 1.0f; + float norm; + + GLM(vec3_normalize_to)(v1, v2); + + norm = sqrtf(v1[0] * v1[0] + v1[1] * v1[1] + v1[2] * v1[2]); + if (norm < FLT_EPSILON) { + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + TEST_SUCCESS + } + + norm = s / norm; + + ASSERT(test_eq(v1[0] * norm, v2[0])) + ASSERT(test_eq(v1[1] * norm, v2[1])) + ASSERT(test_eq(v1[2] * norm, v2[2])) + + glm_vec3_zero(v1); + GLM(vec3_normalize_to)(v1, v2); + ASSERTIFY(test_assert_vec3_eq(v2, GLM_VEC3_ZERO)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, normalize) { + /* SAME AS VEC3_NORMALIZE */ + + vec3 v1 = {2.0f, -3.0f, 4.0f}, v2 = {2.0f, -3.0f, 4.0f}; + float s = 1.0f; + float norm; + + GLM(vec3_normalize)(v2); + + norm = sqrtf(v1[0] * v1[0] + v1[1] * v1[1] + v1[2] * v1[2]); + if (norm < FLT_EPSILON) { + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + TEST_SUCCESS + } + + norm = s / norm; + + ASSERT(test_eq(v1[0] * norm, v2[0])) + ASSERT(test_eq(v1[1] * norm, v2[1])) + ASSERT(test_eq(v1[2] * norm, v2[2])) + + glm_vec3_zero(v1); + GLM(vec3_normalize)(v1); + ASSERTIFY(test_assert_vec3_eq(v1, GLM_VEC3_ZERO)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, normalize_to) { + /* SAME AS VEC3_NORMALIZE_TO */ + + vec3 v1 = {2.0f, -3.0f, 4.0f}, v2; + float s = 1.0f; + float norm; + + GLM(vec3_normalize_to)(v1, v2); + + norm = sqrtf(v1[0] * v1[0] + v1[1] * v1[1] + v1[2] * v1[2]); + if (norm < FLT_EPSILON) { + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + TEST_SUCCESS + } + + norm = s / norm; + + ASSERT(test_eq(v1[0] * norm, v2[0])) + ASSERT(test_eq(v1[1] * norm, v2[1])) + ASSERT(test_eq(v1[2] * norm, v2[2])) + + glm_vec3_zero(v1); + GLM(vec3_normalize_to)(v1, v2); + ASSERTIFY(test_assert_vec3_eq(v2, GLM_VEC3_ZERO)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_cross) { + /* (u2.v3 - u3.v2, u3.v1 - u1.v3, u1.v2 - u2.v1) */ + vec3 v1 = {2.0f, -3.0f, 4.0f}, v2 = {12.0f, -31.0f, 43.0f}, v3, v4; + + GLM(vec3_cross)(v1, v2, v3); + + v4[0] = v1[1] * v2[2] - v1[2] * v2[1]; + v4[1] = v1[2] * v2[0] - v1[0] * v2[2]; + v4[2] = v1[0] * v2[1] - v1[1] * v2[0]; + + ASSERTIFY(test_assert_vec3_eq(v3, v4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_crossn) { + /* (u2.v3 - u3.v2, u3.v1 - u1.v3, u1.v2 - u2.v1) */ + vec3 v1 = {2.0f, -3.0f, 4.0f}, v2 = {12.0f, -31.0f, 43.0f}, v3, v4; + + GLM(vec3_crossn)(v1, v2, v3); + + v4[0] = v1[1] * v2[2] - v1[2] * v2[1]; + v4[1] = v1[2] * v2[0] - v1[0] * v2[2]; + v4[2] = v1[0] * v2[1] - v1[1] * v2[0]; + + glm_normalize(v4); + + ASSERTIFY(test_assert_vec3_eq(v3, v4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, cross) { + /* SAME AS VEC3_CROSS */ + + /* (u2.v3 - u3.v2, u3.v1 - u1.v3, u1.v2 - u2.v1) */ + vec3 v1 = {2.0f, -3.0f, 4.0f}, v2 = {12.0f, -31.0f, 43.0f}, v3, v4; + + GLM(vec3_cross)(v1, v2, v3); + + v4[0] = v1[1] * v2[2] - v1[2] * v2[1]; + v4[1] = v1[2] * v2[0] - v1[0] * v2[2]; + v4[2] = v1[0] * v2[1] - v1[1] * v2[0]; + + ASSERTIFY(test_assert_vec3_eq(v3, v4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_angle) { + vec3 v1 = {1.0f, 0.0f, 0.0f}, + v2 = {1.0f, 0.0f, 1.0f}, + v3 = {0.0f, 1.0f, 0.0f}; + float a; + + a = GLM(vec3_angle)(v1, v1); + +#ifndef CGLM_FAST_MATH + ASSERT(!isinf(a)) + ASSERT(!isnan(a)) +#endif + + ASSERT(test_eq(a, 0.0f)) + + a = GLM(vec3_angle)(v1, v2); + +#ifndef CGLM_FAST_MATH + ASSERT(!isinf(a)) + ASSERT(!isnan(a)) +#endif + + ASSERT(test_eq(a, GLM_PI_4f)) + + a = GLM(vec3_angle)(v1, v3); + +#ifndef CGLM_FAST_MATH + ASSERT(!isinf(a)) + ASSERT(!isnan(a)) +#endif + + ASSERT(test_eq(a, GLM_PI_2f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_rotate) { + vec3 v1 = {1.0f, 0.0f, 0.0f}, v2 = {1.0f, 1.0f, 1.0f}; + + /* rotate X around Y = -Z */ + GLM(vec3_rotate)(v1, GLM_PI_2f, GLM_YUP); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], -1.0f)) + + /* rotate -Z around X = Y */ + GLM(vec3_rotate)(v1, GLM_PI_2f, GLM_XUP); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 1.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + /* rotate Y around Z = -X */ + GLM(vec3_rotate)(v1, GLM_PI_2f, GLM_ZUP); + + ASSERT(test_eq(v1[0], -1.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + /* rotate v2 around Y by 90deg */ + GLM(vec3_rotate)(v2, GLM_PI_2f, GLM_YUP); + + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], -1.0f)) + + /* rotate v2 around Y by 90deg */ + GLM(vec3_rotate)(v2, GLM_PI_2f, GLM_YUP); + + ASSERT(test_eq(v2[0], -1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], -1.0f)) + + /* rotate v2 around Y by 90deg */ + GLM(vec3_rotate)(v2, GLM_PI_2f, GLM_YUP); + + ASSERT(test_eq(v2[0], -1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + /* rotate v2 around X by 90deg */ + GLM(vec3_rotate)(v2, GLM_PI_2f, GLM_XUP); + + ASSERT(test_eq(v2[0], -1.0f)) + ASSERT(test_eq(v2[1], -1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + /* rotate v2 around Z by 90deg */ + GLM(vec3_rotate)(v2, GLM_PI_2f, GLM_ZUP); + + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], -1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_rotate_m4) { + vec3 v1 = {1.0f, 0.0f, 0.0f}, v2 = {1.0f, 1.0f, 1.0f}; + mat4 x, y, z; + + glm_rotate_make(x, GLM_PI_2f, GLM_XUP); + glm_rotate_make(y, GLM_PI_2f, GLM_YUP); + glm_rotate_make(z, GLM_PI_2f, GLM_ZUP); + + /* rotate X around Y = -Z */ + GLM(vec3_rotate_m4)(y, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], -1.0f)) + + /* rotate -Z around X = Y */ + GLM(vec3_rotate_m4)(x, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 1.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + /* rotate Y around X = -X */ + GLM(vec3_rotate_m4)(z, v1, v1); + + ASSERT(test_eq(v1[0], -1.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + /* rotate v2 around Y by 90deg */ + GLM(vec3_rotate_m4)(y, v2, v2); + + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], -1.0f)) + + /* rotate v2 around Y by 90deg */ + GLM(vec3_rotate_m4)(y, v2, v2); + + ASSERT(test_eq(v2[0], -1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], -1.0f)) + + /* rotate v2 around Y by 90deg */ + GLM(vec3_rotate_m4)(y, v2, v2); + + ASSERT(test_eq(v2[0], -1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + /* rotate v2 around X by 90deg */ + GLM(vec3_rotate_m4)(x, v2, v2); + + ASSERT(test_eq(v2[0], -1.0f)) + ASSERT(test_eq(v2[1], -1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + /* rotate v2 around Z by 90deg */ + GLM(vec3_rotate_m4)(z, v2, v2); + + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], -1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_rotate_m3) { + vec3 v1 = {1.0f, 0.0f, 0.0f}, v2 = {1.0f, 1.0f, 1.0f}; + mat4 x0, y0, z0; + mat3 x, y, z; + + glm_rotate_make(x0, GLM_PI_2f, GLM_XUP); + glm_rotate_make(y0, GLM_PI_2f, GLM_YUP); + glm_rotate_make(z0, GLM_PI_2f, GLM_ZUP); + + glm_mat4_pick3(x0, x); + glm_mat4_pick3(y0, y); + glm_mat4_pick3(z0, z); + + /* rotate X around Y = -Z */ + GLM(vec3_rotate_m3)(y, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], -1.0f)) + + /* rotate -Z around X = Y */ + GLM(vec3_rotate_m3)(x, v1, v1); + + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 1.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + /* rotate Y around Z = -X */ + GLM(vec3_rotate_m3)(z, v1, v1); + + ASSERT(test_eq(v1[0], -1.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + + /* rotate v2 around Y by 90deg */ + GLM(vec3_rotate_m3)(y, v2, v2); + + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], -1.0f)) + + /* rotate v2 around Y by 90deg */ + GLM(vec3_rotate_m3)(y, v2, v2); + + ASSERT(test_eq(v2[0], -1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], -1.0f)) + + /* rotate v2 around Y by 90deg */ + GLM(vec3_rotate_m3)(y, v2, v2); + + ASSERT(test_eq(v2[0], -1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + /* rotate v2 around X by 90deg */ + GLM(vec3_rotate_m3)(x, v2, v2); + + ASSERT(test_eq(v2[0], -1.0f)) + ASSERT(test_eq(v2[1], -1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + /* rotate v2 around Z by 90deg */ + GLM(vec3_rotate_m3)(z, v2, v2); + + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], -1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_proj) { + vec3 v1 = {3.0f, 4.0f, 0.0f}, + v2 = {10.0f, 0.0f, 0.0f}, + v3 = {0.0f, 10.0f, 0.0f}, + v4 = {3.0f, 0.0f, 0.0f}, + v5 = {0.0f, 4.0f, 0.0f}, + v6; + + GLM(vec3_proj)(v1, v2, v6); + ASSERTIFY(test_assert_vec3_eq(v4, v6)) + + GLM(vec3_proj)(v1, v3, v6); + ASSERTIFY(test_assert_vec3_eq(v5, v6)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_center) { + vec3 v1 = {30.0f, 0.0f, 0.0f}, + v2 = {0.0f, 0.0f, 0.0f}, + v3 = {15.0f, 0.0f, 0.0f}, + v4 = {3.0f, 10.0f, 120.0f}, + v5 = {0.46f, 4.0f, 14.0f}, + v6; + + GLM(vec3_center)(v1, v2, v6); + ASSERTIFY(test_assert_vec3_eq(v3, v6)) + + GLM(vec3_center)(v4, v5, v6); + ASSERT(test_eq((v4[0] + v5[0]) * 0.5f, v6[0])) + ASSERT(test_eq((v4[1] + v5[1]) * 0.5f, v6[1])) + ASSERT(test_eq((v4[2] + v5[2]) * 0.5f, v6[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_distance2) { + vec3 v1 = {30.0f, 0.0f, 0.0f}, + v2 = {0.0f, 0.0f, 0.0f}, + v3 = {3.0f, 10.0f, 120.0f}, + v4 = {0.46f, 4.0f, 14.0f}; + float d; + + d = GLM(vec3_distance2)(v1, v2); + ASSERT(test_eq(d, 30.0f * 30.0f)) + + d = GLM(vec3_distance2)(v3, v4); + ASSERT(test_eq(powf(v3[0] - v4[0], 2.0f) + + powf(v3[1] - v4[1], 2.0f) + + powf(v3[2] - v4[2], 2.0f), d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_distance) { + vec3 v1 = {30.0f, 0.0f, 0.0f}, + v2 = {0.0f, 0.0f, 0.0f}, + v3 = {3.0f, 10.0f, 120.0f}, + v4 = {0.46f, 4.0f, 14.0f}; + float d; + + d = GLM(vec3_distance)(v1, v2); + ASSERT(test_eq(d, 30.0f)) + + d = GLM(vec3_distance)(v3, v4); + ASSERT(test_eq(sqrtf(powf(v3[0] - v4[0], 2.0f) + + powf(v3[1] - v4[1], 2.0f) + + powf(v3[2] - v4[2], 2.0f)), d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_maxv) { + vec3 v1, v2, v3; + vec3 v5 = {-1.456f, -1.456f, 241.456f}; + vec3 v6 = {11.0f, 11.0f, 11.0f}; + vec3 v7 = {78.0f, -78.0f, 7.0f}; + + GLM(vec3_maxv)(v5, v6, v1); + GLM(vec3_maxv)(v5, v7, v2); + GLM(vec3_maxv)(v6, v7, v3); + + ASSERT(test_eq(v1[0], 11.0f)) + ASSERT(test_eq(v1[1], 11.0f)) + ASSERT(test_eq(v1[2], 241.456f)) + + ASSERT(test_eq(v2[0], 78.0f)) + ASSERT(test_eq(v2[1], -1.456f)) + ASSERT(test_eq(v2[2], 241.456f)) + + ASSERT(test_eq(v3[0], 78.0f)) + ASSERT(test_eq(v3[1], 11.0f)) + ASSERT(test_eq(v3[2], 11.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_minv) { + vec3 v1, v2, v3; + vec3 v5 = {-1.456f, -1.456f, 241.456f}; + vec3 v6 = {11.0f, 11.0f, 11.0f}; + vec3 v7 = {78.0f, -78.0f, 7.0f}; + + GLM(vec3_minv)(v5, v6, v1); + GLM(vec3_minv)(v5, v7, v2); + GLM(vec3_minv)(v6, v7, v3); + + ASSERT(test_eq(v1[0], -1.456f)) + ASSERT(test_eq(v1[1], -1.456f)) + ASSERT(test_eq(v1[2], 11.0f)) + + ASSERT(test_eq(v2[0], -1.456f)) + ASSERT(test_eq(v2[1], -78.0f)) + ASSERT(test_eq(v2[2], 7.0f)) + + ASSERT(test_eq(v3[0], 11.0f)) + ASSERT(test_eq(v3[1], -78.0f)) + ASSERT(test_eq(v3[2], 7.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_ortho) { + vec3 v1, v2, v3, v4 = {1.f, 1.f, 1.f}; + vec3 v5, v6, v7, v8; + float a; + + test_rand_vec3(v1); + test_rand_vec3(v2); + test_rand_vec3(v3); + + GLM(vec3_ortho)(v1, v5); + GLM(vec3_ortho)(v2, v6); + GLM(vec3_ortho)(v3, v7); + GLM(vec3_ortho)(v4, v8); + + a = glm_vec3_angle(v1, v5); + +#ifndef CGLM_FAST_MATH + ASSERT(!isinf(a)) + ASSERT(!isnan(a)) +#endif + + ASSERT(test_eq(a, GLM_PI_2f)) + + a = glm_vec3_angle(v2, v6); + +#ifndef CGLM_FAST_MATH + ASSERT(!isinf(a)) + ASSERT(!isnan(a)) +#endif + + ASSERT(test_eq(a, GLM_PI_2f)) + + a = glm_vec3_angle(v3, v7); + +#ifndef CGLM_FAST_MATH + ASSERT(!isinf(a)) + ASSERT(!isnan(a)) +#endif + + ASSERT(test_eq(a, GLM_PI_2f)) + + a = glm_vec3_angle(v4, v8); + +#ifndef CGLM_FAST_MATH + ASSERT(!isinf(a)) + ASSERT(!isnan(a)) +#endif + + ASSERT(test_eq(a, GLM_PI_2f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_clamp) { + vec3 v1 = {-1.456f, -11.456f, 31.456f}; + vec3 v2 = {0.110f, 111.0f, 11.0f}; + vec3 v3 = {78.0f, 32.0f, -78.0f}; + + GLM(vec3_clamp)(v1, -1.03f, 30.0f); + GLM(vec3_clamp)(v2, 0.11f, 111.0f); + GLM(vec3_clamp)(v3, -88.0f, 70.0f); + + ASSERT(test_eq(v1[0], -1.03f)) + ASSERT(test_eq(v1[1], -1.03f)) + ASSERT(test_eq(v1[2], 30.0f)) + + ASSERT(test_eq(v2[0], 0.11f)) + ASSERT(test_eq(v2[1], 111.0f)) + ASSERT(test_eq(v2[2], 11.0f)) + + ASSERT(test_eq(v3[0], 70.0f)) + ASSERT(test_eq(v3[1], 32.0f)) + ASSERT(test_eq(v3[2], -78.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_lerp) { + vec3 v1 = {-100.0f, -200.0f, -10.0f}; + vec3 v2 = {100.0f, 200.0f, 10.0f}; + vec3 v3; + + GLM(vec3_lerp)(v1, v2, 0.5f, v3); + ASSERT(test_eq(v3[0], 0.0f)) + ASSERT(test_eq(v3[1], 0.0f)) + ASSERT(test_eq(v3[2], 0.0f)) + + GLM(vec3_lerp)(v1, v2, 0.75f, v3); + ASSERT(test_eq(v3[0], 50.0f)) + ASSERT(test_eq(v3[1], 100.0f)) + ASSERT(test_eq(v3[2], 5.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_lerpc) { + vec3 v1 = {-100.0f, -200.0f, -10.0f}; + vec3 v2 = {100.0f, 200.0f, 10.0f}; + vec3 v3; + + GLM(vec3_lerpc)(v1, v2, 0.5f, v3); + ASSERT(test_eq(v3[0], 0.0f)) + ASSERT(test_eq(v3[1], 0.0f)) + ASSERT(test_eq(v3[2], 0.0f)) + + GLM(vec3_lerpc)(v1, v2, 0.75f, v3); + ASSERT(test_eq(v3[0], 50.0f)) + ASSERT(test_eq(v3[1], 100.0f)) + ASSERT(test_eq(v3[2], 5.0f)) + + GLM(vec3_lerpc)(v1, v2, -1.75f, v3); + ASSERT(test_eq(v3[0], -100.0f)) + ASSERT(test_eq(v3[1], -200.0f)) + ASSERT(test_eq(v3[2], -10.0f)) + + GLM(vec3_lerpc)(v1, v2, 1.75f, v3); + ASSERT(test_eq(v3[0], 100.0f)) + ASSERT(test_eq(v3[1], 200.0f)) + ASSERT(test_eq(v3[2], 10.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_mix) { + vec3 v1 = {-100.0f, -200.0f, -10.0f}; + vec3 v2 = {100.0f, 200.0f, 10.0f}; + vec3 v3; + + GLM(vec3_mix)(v1, v2, 0.5f, v3); + ASSERT(test_eq(v3[0], 0.0f)) + ASSERT(test_eq(v3[1], 0.0f)) + ASSERT(test_eq(v3[2], 0.0f)) + + GLM(vec3_mix)(v1, v2, 0.75f, v3); + ASSERT(test_eq(v3[0], 50.0f)) + ASSERT(test_eq(v3[1], 100.0f)) + ASSERT(test_eq(v3[2], 5.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_mixc) { + vec3 v1 = {-100.0f, -200.0f, -10.0f}; + vec3 v2 = {100.0f, 200.0f, 10.0f}; + vec3 v3; + + GLM(vec3_mixc)(v1, v2, 0.5f, v3); + ASSERT(test_eq(v3[0], 0.0f)) + ASSERT(test_eq(v3[1], 0.0f)) + ASSERT(test_eq(v3[2], 0.0f)) + + GLM(vec3_mixc)(v1, v2, 0.75f, v3); + ASSERT(test_eq(v3[0], 50.0f)) + ASSERT(test_eq(v3[1], 100.0f)) + ASSERT(test_eq(v3[2], 5.0f)) + + GLM(vec3_mixc)(v1, v2, -1.75f, v3); + ASSERT(test_eq(v3[0], -100.0f)) + ASSERT(test_eq(v3[1], -200.0f)) + ASSERT(test_eq(v3[2], -10.0f)) + + GLM(vec3_mixc)(v1, v2, 1.75f, v3); + ASSERT(test_eq(v3[0], 100.0f)) + ASSERT(test_eq(v3[1], 200.0f)) + ASSERT(test_eq(v3[2], 10.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_steps) { + vec3 v1 = {-100.0f, -200.0f, -10.0f}; + vec3 v2; + + GLM(vec3_steps)(-2.5f, v1, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + ASSERT(test_eq(v2[2], 0.0f)) + + GLM(vec3_steps)(-10.0f, v1, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + GLM(vec3_steps)(-1000.0f, v1, v2); + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_stepr) { + vec3 v1 = {-2.5f, -10.0f, -1000.0f}; + vec3 v2; + + GLM(vec3_stepr)(v1, -100.0f, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + GLM(vec3_stepr)(v1, -5.0f, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + GLM(vec3_stepr)(v1, -1.0f, v2); + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_step) { + vec3 v1 = {-100.0f, -200.0f, -10.0f}; + vec3 s1 = {-100.0f, 0.0f, 10.0f}; + vec3 s2 = {100.0f, -220.0f, -10.0f}; + vec3 s3 = {100.0f, 200.0f, 10.0f}; + vec3 v2; + + GLM(vec3_step)(s1, v1, v2); + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + ASSERT(test_eq(v2[2], 0.0f)) + + GLM(vec3_step)(s2, v1, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + + GLM(vec3_step)(s3, v1, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + ASSERT(test_eq(v2[2], 0.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_smoothstep_uni) { + vec3 v1 = {-100.0f, -200.0f, -10.0f}; + vec3 v2; + + GLM(vec3_smoothstep_uni)(-200.0f, -100.0f, v1, v2); + ASSERT(test_eq_th(v2[0], 1.0f, 1e-5f)) + ASSERT(test_eq_th(v2[1], 0.0f, 1e-5f)) + ASSERT(test_eq_th(v2[2], 1.0f, 1e-5f)) + + GLM(vec3_smoothstep_uni)(-250.0f, -200.0f, v1, v2); + ASSERT(test_eq_th(v2[0], 1.0f, 1e-5f)) + ASSERT(test_eq_th(v2[1], 1.0f, 1e-5f)) + ASSERT(test_eq_th(v2[2], 1.0f, 1e-5f)) + + GLM(vec3_smoothstep_uni)(-200.0f, 200, v1, v2); + ASSERT(v2[0] > 0.0f && v2[0] < 0.25f) + ASSERT(test_eq(v2[1], 0.0f)) + ASSERT(v2[2] > 0.0f && v2[2] < 0.5f) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_smoothstep) { + vec3 v1 = {-100.0f, -200.0f, -10.0f}; + vec3 e1_0 = {-100.0f, 0.0f, -11.0f}; + vec3 e1_1 = {50.0f, 10.0f, 20.0f}; + vec3 e2_0 = {-180.0f, -300.0f, -93.0f}; + vec3 e2_1 = {100.0f, 120.0f, -10.0f}; + vec3 e3_0 = {-12.0f, 100.0f, 0.0f}; + vec3 e3_1 = {100.0f, 200.0f, 10.0f}; + vec3 v2; + + GLM(vec3_smoothstep)(e1_0, e1_1, v1, v2); + ASSERT(test_eq_th(v2[0], 0.0f, 1e-5f)) + ASSERT(test_eq_th(v2[1], 0.0f, 1e-5f)) + ASSERT(v2[2] > 0.0f && v2[2] < 0.1f) + + GLM(vec3_smoothstep)(e2_0, e2_1, v1, v2); + ASSERT(v2[0] > 0.0f && v2[0] < 0.25f) + ASSERT(v2[1] > 0.0f && v2[1] < 0.15f) + ASSERT(test_eq_th(v2[2], 1.0f, 1e-5f)) + + GLM(vec3_smoothstep)(e3_0, e3_1, v1, v2); + ASSERT(test_eq_th(v2[0], 0.0f, 1e-5f)) + ASSERT(test_eq_th(v2[1], 0.0f, 1e-5f)) + ASSERT(test_eq_th(v2[2], 0.0f, 1e-5f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_smoothinterp) { + vec3 e1_0 = {-100.0f, 0.0f, -11.0f}; + vec3 e1_1 = {50.0f, 10.0f, 20.0f}; + vec3 e2_0 = {80.0f, -220.0f, -19.0f}; + vec3 e2_1 = {100.0f, -200.0f, -10.0f}; + vec3 e3_0 = {-12.0f, 100.0f, 0.0f}; + vec3 e3_1 = {100.0f, 200.0f, 10.0f}; + vec3 v2; + + GLM(vec3_smoothinterp)(e1_0, e1_1, 0.5f, v2); + ASSERT(v2[0] >= e1_0[0] && v2[0] <= e1_1[0]) + ASSERT(v2[1] >= e1_0[1] && v2[1] <= e1_1[1]) + ASSERT(v2[2] >= e1_0[2] && v2[2] <= e1_1[2]) + + GLM(vec3_smoothinterp)(e2_0, e2_1, 0.5, v2); + ASSERT(v2[0] >= e2_0[0] && v2[0] <= e2_1[0]) + ASSERT(v2[1] >= e2_0[1] && v2[1] <= e2_1[1]) + ASSERT(v2[2] >= e2_0[2] && v2[2] <= e2_1[2]) + + GLM(vec3_smoothinterp)(e3_0, e3_1, 1.0, v2); + ASSERT(v2[0] >= e3_0[0] && v2[0] <= e3_1[0]) + ASSERT(v2[1] >= e3_0[1] && v2[1] <= e3_1[1]) + ASSERT(v2[2] >= e3_0[2] && v2[2] <= e3_1[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_smoothinterpc) { + vec3 e1_0 = {-100.0f, 0.0f, -11.0f}; + vec3 e1_1 = {50.0f, 10.0f, 20.0f}; + vec3 e2_0 = {80.0f, -220.0f, -19.0f}; + vec3 e2_1 = {100.0f, -200.0f, -10.0f}; + vec3 e3_0 = {-12.0f, 100.0f, 0.0f}; + vec3 e3_1 = {100.0f, 200.0f, 10.0f}; + vec3 v2; + + GLM(vec3_smoothinterpc)(e1_0, e1_1, -0.5f, v2); + ASSERT(v2[0] >= e1_0[0] && v2[0] <= e1_1[0]) + ASSERT(v2[1] >= e1_0[1] && v2[1] <= e1_1[1]) + ASSERT(v2[2] >= e1_0[2] && v2[2] <= e1_1[2]) + + GLM(vec3_smoothinterpc)(e2_0, e2_1, 0.5f, v2); + ASSERT(v2[0] >= e2_0[0] && v2[0] <= e2_1[0]) + ASSERT(v2[1] >= e2_0[1] && v2[1] <= e2_1[1]) + ASSERT(v2[2] >= e2_0[2] && v2[2] <= e2_1[2]) + + GLM(vec3_smoothinterpc)(e3_0, e3_1, 2.0f, v2); + ASSERT(v2[0] >= e3_0[0] && v2[0] <= e3_1[0]) + ASSERT(v2[1] >= e3_0[1] && v2[1] <= e3_1[1]) + ASSERT(v2[2] >= e3_0[2] && v2[2] <= e3_1[2]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_swizzle) { + vec3 v; + + /* ZYX */ + v[0] = 1; + v[1] = 2; + v[2] = 3; + + GLM(vec3_swizzle)(v, GLM_ZYX, v); + ASSERTIFY(test_assert_vec3_eq(v, (vec3){3, 2, 1})) + + GLM(vec3_swizzle)(v, GLM_XXX, v); + ASSERTIFY(test_assert_vec3_eq(v, (vec3){3, 3, 3})) + + v[0] = 1; + v[1] = 2; + v[2] = 3; + + GLM(vec3_swizzle)(v, GLM_YYY, v); + ASSERTIFY(test_assert_vec3_eq(v, (vec3){2, 2, 2})) + + v[0] = 1; + v[1] = 2; + v[2] = 3; + + GLM(vec3_swizzle)(v, GLM_ZZZ, v); + ASSERTIFY(test_assert_vec3_eq(v, (vec3){3, 3, 3})) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_broadcast) { + vec3 v1, v2, v3; + vec3 v5 = {-1.456f, -1.456f, -1.456f}; + vec3 v6 = {11.0f, 11.0f, 11.0f}; + vec3 v7 = {78.0f, 78.0f, 78.0f}; + + GLM(vec3_broadcast)(-1.456f, v1); + GLM(vec3_broadcast)(11.0f, v2); + GLM(vec3_broadcast)(78.0f, v3); + + ASSERTIFY(test_assert_vec3_eq(v1, v5)) + ASSERTIFY(test_assert_vec3_eq(v2, v6)) + ASSERTIFY(test_assert_vec3_eq(v3, v7)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_fill) { + vec3 v1, v2, v3; + vec3 v5 = {-1.456f, -1.456f, -1.456f}; + vec3 v6 = {11.0f, 11.0f, 11.0f}; + vec3 v7 = {78.0f, 78.0f, 78.0f}; + + GLM(vec3_fill)(v1, -1.456f); + GLM(vec3_fill)(v2, 11.0f); + GLM(vec3_fill)(v3, 78.0f); + + ASSERTIFY(test_assert_vec3_eq(v1, v5)) + ASSERTIFY(test_assert_vec3_eq(v2, v6)) + ASSERTIFY(test_assert_vec3_eq(v3, v7)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_eq) { + vec3 v1, v2, v3; + + GLM(vec3_fill)(v1, -1.456f); + GLM(vec3_fill)(v2, 11.0f); + GLM(vec3_fill)(v3, 78.1f); + + ASSERT(GLM(vec3_eq)(v1, -1.456f)) + ASSERT(GLM(vec3_eq)(v2, 11.0f)) + ASSERT(!GLM(vec3_eq)(v3, 78.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_eq_eps) { + vec3 v1, v2, v3; + + GLM(vec3_fill)(v1, -1.456f); + GLM(vec3_fill)(v2, 11.0f); + GLM(vec3_fill)(v3, 78.1f); + + ASSERT(GLM(vec3_eq_eps)(v1, -1.456f)) + ASSERT(GLM(vec3_eq_eps)(v2, 11.0f)) + ASSERT(!GLM(vec3_eq_eps)(v3, 78.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_eq_all) { + vec3 v1, v2, v3; + vec3 v4 = {2.104f, -3.012f, -4.10f}; + vec3 v5 = {-12.35f, -31.140f, -43.502f}; + + GLM(vec3_fill)(v1, -1.456f); + GLM(vec3_fill)(v2, 11.0f); + GLM(vec3_fill)(v3, 78.0f); + + ASSERT(GLM(vec3_eq_all)(v1)) + ASSERT(GLM(vec3_eq_all)(v2)) + ASSERT(GLM(vec3_eq_all)(v3)) + ASSERT(!GLM(vec3_eq_all)(v4)) + ASSERT(!GLM(vec3_eq_all)(v5)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_eqv) { + vec3 v1, v2, v3, v4, v5; + vec3 v6 = {-1.456f, -1.456f, -1.456f}; + vec3 v7 = {11.0f, 11.0f, 11.0f}; + vec3 v8 = {78.0f, 78.0f, -43.502f}; + + GLM(vec3_fill)(v1, -1.456f); + GLM(vec3_fill)(v2, 11.0f); + GLM(vec3_fill)(v3, 78.0f); + + test_rand_vec3(v4); + test_rand_vec3(v5); + + ASSERT(GLM(vec3_eqv)(v1, v6)) + ASSERT(GLM(vec3_eqv)(v2, v7)) + ASSERT(!GLM(vec3_eqv)(v3, v8)) + ASSERT(!GLM(vec3_eqv)(v4, v5)) + ASSERT(GLM(vec3_eqv)(v5, v5)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_eqv_eps) { + vec3 v1, v2, v3, v4, v5; + vec3 v6 = {-1.456f, -1.456f, -1.456f}; + vec3 v7 = {11.0f, 11.0f, 11.0f}; + vec3 v8 = {78.0f, 78.0f, -43.502f}; + + GLM(vec3_fill)(v1, -1.456f); + GLM(vec3_fill)(v2, 11.0f); + GLM(vec3_fill)(v3, 78.0f); + + test_rand_vec3(v4); + test_rand_vec3(v5); + + ASSERT(GLM(vec3_eqv_eps)(v1, v6)) + ASSERT(GLM(vec3_eqv_eps)(v2, v7)) + ASSERT(!GLM(vec3_eqv_eps)(v3, v8)) + ASSERT(!GLM(vec3_eqv_eps)(v4, v5)) + ASSERT(GLM(vec3_eqv_eps)(v5, v5)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_max) { + vec3 v1 = {2.104f, -3.012f, -4.10f}, v2 = {-12.35f, -31.140f, -43.502f}; +#ifndef CGLM_FAST_MATH + vec3 v3 = {INFINITY, 0.0f, 0.0f}/*, v4 = {NAN, INFINITY, 2.0f}*/; +#endif + vec3 /*v5 = {NAN, -1.0f, -1.0f}, */v6 = {-1.0f, -11.0f, 11.0f}; + + ASSERT(test_eq(GLM(vec3_max)(v1), 2.104f)) + ASSERT(test_eq(GLM(vec3_max)(v2), -12.35f)) +#ifndef CGLM_FAST_MATH + ASSERT(isinf(GLM(vec3_max)(v3))) +#endif +// ASSERT(isnan(GLM(vec3_max)(v4))) +// ASSERT(isnan(GLM(vec3_max)(v5))) + ASSERT(test_eq(GLM(vec3_max)(v6), 11.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_min) { + vec3 v1 = {2.104f, -3.012f, -4.10f}, v2 = {-12.35f, -31.140f, -43.502f}; +#ifndef CGLM_FAST_MATH + vec3 v3 = {INFINITY, 0.0f, 0.0f}/*, v4 = {NAN, INFINITY, 2.0f}*/; +#endif + vec3 /*v5 = {NAN, -1.0f, -1.0f},*/ v6 = {-1.0f, -11.0f, 11.0f}; + + ASSERT(test_eq(GLM(vec3_min)(v1), -4.10f)) + ASSERT(test_eq(GLM(vec3_min)(v2), -43.502f)) +#ifndef CGLM_FAST_MATH + ASSERT(test_eq(GLM(vec3_min)(v3), 0.0f)) +#endif +// ASSERT(isnan(GLM(vec3_min)(v4))) +// ASSERT(isnan(GLM(vec3_min)(v5))) + ASSERT(test_eq(GLM(vec3_min)(v6), -11.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_isnan) { +#ifndef CGLM_FAST_MATH + vec3 v1 = {2.104f, -3.012f, -4.10f}, v2 = {-12.35f, -31.140f, -43.502f}; + vec3 v3 = {INFINITY, 0.0f, 0.0f}, v4 = {NAN, INFINITY, 2.0f}; + vec3 v5 = {NAN, -1.0f, -1.0f}, v6 = {-1.0f, -1.0f, 11.0f}; + + ASSERT(!GLM(vec3_isnan)(v1)) + ASSERT(!GLM(vec3_isnan)(v2)) + ASSERT(!GLM(vec3_isnan)(v3)) + ASSERT(GLM(vec3_isnan)(v4)) + ASSERT(GLM(vec3_isnan)(v5)) + ASSERT(!GLM(vec3_isnan)(v6)) +#endif + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_isinf) { +#ifndef CGLM_FAST_MATH + vec3 v1 = {2.104f, -3.012f, -4.10f}, v2 = {-12.35f, -31.140f, -43.502f}; + vec3 v3 = {INFINITY, 0.0f, 0.0f}, v4 = {NAN, INFINITY, 2.0f}; + vec3 v5 = {NAN, -1.0f, -1.0f}, v6 = {-1.0f, -1.0f, 11.0f}; + + ASSERT(!GLM(vec3_isinf)(v1)) + ASSERT(!GLM(vec3_isinf)(v2)) + ASSERT(GLM(vec3_isinf)(v3)) + ASSERT(GLM(vec3_isinf)(v4)) + ASSERT(!GLM(vec3_isinf)(v5)) + ASSERT(!GLM(vec3_isinf)(v6)) +#endif + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_isvalid) { +#ifndef CGLM_FAST_MATH + vec3 v1 = {2.104f, -3.012f, -4.10f}, v2 = {-12.35f, -31.140f, -43.502f}; + vec3 v3 = {INFINITY, 0.0f, 0.0f}, v4 = {NAN, INFINITY, 2.0f}; + vec3 v5 = {NAN, -1.0f, -1.0f}, v6 = {-1.0f, -1.0f, 11.0f}; + + ASSERT(GLM(vec3_isvalid)(v1)) + ASSERT(GLM(vec3_isvalid)(v2)) + ASSERT(!GLM(vec3_isvalid)(v3)) + ASSERT(!GLM(vec3_isvalid)(v4)) + ASSERT(!GLM(vec3_isvalid)(v5)) + ASSERT(GLM(vec3_isvalid)(v6)) +#endif + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_sign) { + vec3 v1 = {2.104f, -3.012f, -4.10f}, v2 = {-12.35f, -31.140f, -43.502f}; + vec3 v3, v4; + vec3 v5 = {1.0f, -1.0f, -1.0f}, v6 = {-1.0f, -1.0f, -1.0f}; + + GLM(vec3_sign)(v1, v3); + GLM(vec3_sign)(v2, v4); + + ASSERTIFY(test_assert_vec3_eq(v3, v5)) + ASSERTIFY(test_assert_vec3_eq(v4, v6)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_abs) { + vec3 v1 = {2.104f, -3.012f, -4.10f}, v2 = {-12.35f, -31.140f, -43.502f}; + vec3 v3, v4; + vec3 v5 = {2.104f, 3.012f, 4.10f}, v6 = {12.35f, 31.140f, 43.502f}; + + GLM(vec3_abs)(v1, v3); + GLM(vec3_abs)(v2, v4); + + ASSERTIFY(test_assert_vec3_eq(v3, v5)) + ASSERTIFY(test_assert_vec3_eq(v4, v6)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_fract) { + vec3 v1 = {2.104f, 3.012f, 4.10f}, v2 = {12.35f, 31.140f, 43.502f}, v3, v4; + vec3 v5 = {0.104f, 0.012f, 0.10f}, v6 = {0.35f, 0.140f, 0.502f}; + + GLM(vec3_fract)(v1, v3); + GLM(vec3_fract)(v2, v4); + + ASSERTIFY(test_assert_vec3_eq(v3, v5)) + ASSERTIFY(test_assert_vec3_eq(v4, v6)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_floor) { + vec3 v1 = {2.104f, 3.012f, 4.10f}, v2 = {12.35f, 31.140f, 43.502f}, v3, v4; + vec3 v5 = {2.0f, 3.0f, 4.0f}, v6 = {12.0f, 31.0f, 43.0f}; + + GLM(vec3_floor)(v1, v3); + GLM(vec3_floor)(v2, v4); + + ASSERTIFY(test_assert_vec3_eq(v3, v5)) + ASSERTIFY(test_assert_vec3_eq(v4, v6)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_mods) { + vec3 v1 = {2.104f, 3.012f, 4.10f}, v2 = {12.35f, 31.140f, 43.502f}, v3, v4; + vec3 v5 = {0.104f, 0.012f, 0.10f}, v6 = {0.35f, 0.140f, 0.502f}; + + /* Mod 1 - leaves just the fractional part */ + GLM(vec3_mods)(v1, 1.0f, v3); + GLM(vec3_mods)(v2, 1.0f, v4); + + ASSERTIFY(test_assert_vec3_eq(v3, v5)) + ASSERTIFY(test_assert_vec3_eq(v4, v6)) + + /* Mod 2 - parity + fractional part */ + GLM(vec3_mods)(v1, 2.0f, v3); + GLM(vec3_mods)(v2, 2.0f, v4); + + vec3 v7 = {0.104f, 1.012f, 0.10f}, v8 = {0.35f, 1.140f, 1.502f}; + + ASSERTIFY(test_assert_vec3_eq(v3, v7)) + ASSERTIFY(test_assert_vec3_eq(v4, v8)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_hadd) { + vec3 v1 = {2.0f, 3.0f, 4.0f}, v2 = {12.0f, 31.0f, 43.0f}; + float r1, r2, r3, r4; + + r1 = GLM(vec3_hadd)(v1); + r2 = GLM(vec3_hadd)(v2); + + r3 = v1[0] + v1[1] + v1[2]; + r4 = v2[0] + v2[1] + v2[2]; + + ASSERT(test_eq(r1, r3)) + ASSERT(test_eq(r2, r4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_sqrt) { + vec3 v1 = {2.0f, 3.0f, 4.0f}, v2 = {12.0f, 31.0f, 43.0f}, v3, v4; + + GLM(vec3_sqrt)(v1, v3); + GLM(vec3_sqrt)(v2, v4); + + ASSERT(test_eq(sqrtf(v1[0]), v3[0])) + ASSERT(test_eq(sqrtf(v1[1]), v3[1])) + ASSERT(test_eq(sqrtf(v1[2]), v3[2])) + + ASSERT(test_eq(sqrtf(v2[0]), v4[0])) + ASSERT(test_eq(sqrtf(v2[1]), v4[1])) + ASSERT(test_eq(sqrtf(v2[2]), v4[2])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_make) { + float src[9] = { + 7.2f, 1.0f, 5.8f, + 2.5f, 6.1f, 9.9f, + 17.7f, 4.3f, 3.2f + }; + vec3 dest[3]; + + float *srcp = src; + unsigned int i, j; + + for (i = 0, j = 0; i < sizeof(src) / sizeof(float); i+=3,j++) { + GLM(vec3_make)(srcp + i, dest[j]); + ASSERT(test_eq(src[ i ], dest[j][0])); + ASSERT(test_eq(src[i+1], dest[j][1])); + ASSERT(test_eq(src[i+2], dest[j][2])); + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_faceforward) { + vec3 N = {0.0f, 1.0f, 0.0f}; + vec3 v = {1.0f, -1.0f, 0.0f}; + vec3 Nref = {0.0f, -1.0f, 0.0f}; + vec3 dest; + + GLM(vec3_faceforward)(N, v, Nref, dest); + ASSERT(dest[0] == 0.0f + && dest[1] == -1.0f + && dest[2] == 0.0f); /* Expect N flipped */ + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_reflect) { + vec3 dest; + + /* Original test: Reflecting off a horizontal surface */ + vec3 I1 = {1.0f, -1.0f, 0.0f}; /* Incoming vector */ + vec3 N1 = {0.0f, 1.0f, 0.0f}; /* Normal vector */ + GLM(vec3_reflect)(I1, N1, dest); + ASSERT(fabsf(dest[0] - 1.0f) < 0.00001f + && fabsf(dest[1] - 1.0f) < 0.00001f + && fabsf(dest[2] - 0.0f) < 0.00001f); /* Expect reflection */ + + /* Scenario 2: Reflecting off a vertical surface */ + vec3 I2 = {1.0f, 0.0f, 0.0f}; /* Incoming vector */ + vec3 N2 = {-1.0f, 0.0f, 0.0f}; /* Normal vector */ + GLM(vec3_reflect)(I2, N2, dest); + ASSERT(fabsf(dest[0] + 1.0f) < 0.00001f + && fabsf(dest[1]) < 0.00001f + && fabsf(dest[2]) < 0.00001f); /* Expect reflection to the left */ + + /* Scenario 3: Reflecting at an angle */ + vec3 I3 = {sqrtf(2)/2, -sqrtf(2)/2, 0.0f}; /* Incoming vector at 45 degrees */ + vec3 N3 = {0.0f, 1.0f, 0.0f}; /* Upwards normal vector */ + GLM(vec3_reflect)(I3, N3, dest); + ASSERT(fabsf(dest[0] - sqrtf(2)/2) < 0.00001f + && fabsf(dest[1] - sqrtf(2)/2) < 0.00001f + && fabsf(dest[2]) < 0.00001f); /* Expect reflection upwards */ + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec3_refract) { + vec3 v = {sqrtf(0.5f), -sqrtf(0.5f), 0.0f}; /* Incoming vector at 45 degrees to normal */ + vec3 N = {0.0f, 1.0f, 0.0f}; /* Surface normal */ + vec3 dest; + float eta; + bool r; + + /* Water to Air (eta = 1.33/1.0) */ + eta = 1.33f / 1.0f; + r = GLM(vec3_refract)(v, N, eta, dest); + if (!(dest[0] == 0.0f && dest[1] == 0.0f && dest[2] == 0.0f)) { + ASSERT(dest[1] < -0.3f); + ASSERT(r == true); + } else { + ASSERT(dest[0] == 0.0f && dest[1] == 0.0f && dest[2] == 0.0f); + ASSERT(r == false); + } + + /* Air to Glass (eta = 1.0 / 1.5) */ + eta = 1.0f / 1.5f; + r = GLM(vec3_refract)(v, N, eta, dest); + + /* Expect bending towards the normal */ + ASSERT(r == true); + ASSERT(dest[1] < -sqrtf(0.5f)); + + /* Glass to Water (eta = 1.5 / 1.33) */ + eta = 1.5f / 1.33f; + r = GLM(vec3_refract)(v, N, eta, dest); + + /* Expect bending towards the normal, less bending than air to glass */ + ASSERT(r == true); + ASSERT(dest[1] < -0.6f); + + /* Diamond to Air (eta = 2.42 / 1.0) */ + eta = 2.42f / 1.0f; + r = GLM(vec3_refract)(v, N, eta, dest); + if (!(dest[0] == 0.0f && dest[1] == 0.0f && dest[2] == 0.0f)) { + /* High potential for total internal reflection, but if it occurs, expect significant bending */ + ASSERT(dest[1] < -sqrtf(0.5f)); + ASSERT(r == true); + } else { + ASSERT(dest[0] == 0.0f && dest[1] == 0.0f && dest[2] == 0.0f); + ASSERT(r == false); + } + + TEST_SUCCESS +} diff --git a/cglm/test/src/test_vec4.h b/cglm/test/src/test_vec4.h new file mode 100644 index 0000000..417babb --- /dev/null +++ b/cglm/test/src/test_vec4.h @@ -0,0 +1,1705 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#include "test_common.h" + +#define TEST_GLM_SHUFFLE4(z, y, x, w) (((z) << 6) | ((y) << 4) | ((x) << 2)|(w)) + +#ifndef CGLM_TEST_VEC4_ONCE +#define CGLM_TEST_VEC4_ONCE + +/* Macros */ + +TEST_IMPL(MACRO_GLM_VEC4_ONE_INIT) { + vec4 v = GLM_VEC4_ONE_INIT; + + ASSERT(test_eq(v[0], 1.0f)) + ASSERT(test_eq(v[1], 1.0f)) + ASSERT(test_eq(v[2], 1.0f)) + ASSERT(test_eq(v[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_VEC4_ZERO_INIT) { + vec4 v = GLM_VEC4_ZERO_INIT; + + ASSERT(test_eq(v[0], 0.0f)) + ASSERT(test_eq(v[1], 0.0f)) + ASSERT(test_eq(v[2], 0.0f)) + ASSERT(test_eq(v[3], 0.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_VEC4_ONE) { + ASSERT(test_eq(GLM_VEC4_ONE[0], 1.0f)) + ASSERT(test_eq(GLM_VEC4_ONE[1], 1.0f)) + ASSERT(test_eq(GLM_VEC4_ONE[2], 1.0f)) + ASSERT(test_eq(GLM_VEC4_ONE[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_VEC4_ZERO) { + ASSERT(test_eq(GLM_VEC4_ZERO[0], 0.0f)) + ASSERT(test_eq(GLM_VEC4_ZERO[1], 0.0f)) + ASSERT(test_eq(GLM_VEC4_ZERO[2], 0.0f)) + ASSERT(test_eq(GLM_VEC4_ZERO[3], 0.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_XXXX) { + ASSERT(TEST_GLM_SHUFFLE4(0, 0, 0, 0) == GLM_XXXX) + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_YYYY) { + ASSERT(TEST_GLM_SHUFFLE4(1, 1, 1, 1) == GLM_YYYY) + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_ZZZZ) { + ASSERT(TEST_GLM_SHUFFLE4(2, 2, 2, 2) == GLM_ZZZZ) + TEST_SUCCESS +} + +TEST_IMPL(MACRO_GLM_WZYX) { + ASSERT(TEST_GLM_SHUFFLE4(0, 1, 2, 3) == GLM_WZYX) + TEST_SUCCESS +} + +/* Deprecated */ + +TEST_IMPL(MACRO_glm_vec4_dup) { + vec4 v1 = {13.0f, 12.0f, 11.0f, 56.0f}, v2; + + glm_vec4_dup(v1, v2); + + ASSERTIFY(test_assert_vec4_eq(v1, v2)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_glm_vec4_flipsign) { + vec4 v1 = {13.0f, -12.0f, 11.0f, 56.0f}, + v2 = {13.0f, -12.0f, 11.0f, 56.0f}, + v3 = {-13.0f, 12.0f, -11.0f, -56.0f}; + + glm_vec4_flipsign(v1); + glmc_vec4_flipsign(v2); + + ASSERTIFY(test_assert_vec4_eq(v1, v3)) + ASSERTIFY(test_assert_vec4_eq(v2, v3)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_glm_vec4_flipsign_to) { + vec4 v1 = {13.0f, -12.0f, 11.0f, 56.0f}, + v2 = {-13.0f, 12.0f, -11.0f, -56.0f}, + v3, v4; + + glm_vec4_flipsign_to(v1, v3); + glmc_vec4_flipsign_to(v1, v4); + + ASSERTIFY(test_assert_vec4_eq(v2, v3)) + ASSERTIFY(test_assert_vec4_eq(v2, v4)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_glm_vec4_inv) { + vec4 v1 = {13.0f, -12.0f, 11.0f, 56.0f}, + v2 = {13.0f, -12.0f, 11.0f, 56.0f}, + v3 = {-13.0f, 12.0f, -11.0f, -56.0f}; + + glm_vec4_inv(v1); + glmc_vec4_inv(v2); + + ASSERTIFY(test_assert_vec4_eq(v1, v3)) + ASSERTIFY(test_assert_vec4_eq(v2, v3)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_glm_vec4_inv_to) { + vec4 v1 = {13.0f, -12.0f, 11.0f, 56.0f}, + v2 = {-13.0f, 12.0f, -11.0f, -56.0f}, + v3, v4; + + glm_vec4_inv_to(v1, v3); + glmc_vec4_inv_to(v1, v4); + + ASSERTIFY(test_assert_vec4_eq(v3, v4)) + ASSERTIFY(test_assert_vec4_eq(v2, v3)) + + TEST_SUCCESS +} + +TEST_IMPL(MACRO_glm_vec4_mulv) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 56.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 56.0f}, + v3, v4; + + glm_vec4_mulv(v1, v2, v3); + glmc_vec4_mulv(v1, v2, v4); + + ASSERTIFY(test_assert_vec4_eq(v3, v4)) + + ASSERT(test_eq(v1[0] * v2[0], v3[0])) + ASSERT(test_eq(v1[1] * v2[1], v3[1])) + ASSERT(test_eq(v1[2] * v2[2], v3[2])) + ASSERT(test_eq(v1[3] * v2[3], v3[3])) + + TEST_SUCCESS +} + +#endif /* CGLM_TEST_VEC4_ONCE */ + +/* --- */ + +TEST_IMPL(GLM_PREFIX, vec4) { + vec4 v1 = {10.0f, 9.0f, 8.0f}; + vec4 v2 = {10.0f, 9.0f, 8.0f, 7.0f}; + vec4 v3; + + GLM(vec4)(v1, 7.0f, v3); + + ASSERTIFY(test_assert_vec4_eq(v2, v3)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_copy3) { + vec4 v4 = {10.0f, 9.0f, 8.0f, 7.0f}; + vec3 v3; + + GLM(vec4_copy3)(v4, v3); + + ASSERTIFY(test_assert_vec3_eq(v3, v4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_copy) { + vec4 v1 = {10.0f, 9.0f, 8.0f, 78.0f}; + vec4 v2 = {1.0f, 2.0f, 3.0f, 4.0f}; + + GLM(vec4_copy)(v1, v2); + + ASSERTIFY(test_assert_vec4_eq(v1, v2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_ucopy) { + vec4 v1 = {10.0f, 9.0f, 8.0f, 78.0f}; + vec4 v2 = {1.0f, 2.0f, 3.0f, 4.0f}; + + GLM(vec4_ucopy)(v1, v2); + + ASSERTIFY(test_assert_vec4_eq(v1, v2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_zero) { + vec4 v1 = {10.0f, 9.0f, 8.0f, 78.0f}; + vec4 v2 = {1.0f, 2.0f, 3.0f, 4.0f}; + + GLM(vec4_zero)(v1); + GLM(vec4_zero)(v2); + + ASSERTIFY(test_assert_vec4_eq(v1, GLM_VEC4_ZERO)) + ASSERTIFY(test_assert_vec4_eq(v2, GLM_VEC4_ZERO)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_one) { + vec4 v1 = {10.0f, 9.0f, 8.0f, 78.0f}; + vec4 v2 = {1.0f, 2.0f, 3.0f, 4.0f}; + + GLM(vec4_one)(v1); + GLM(vec4_one)(v2); + + ASSERTIFY(test_assert_vec4_eq(v1, GLM_VEC4_ONE)) + ASSERTIFY(test_assert_vec4_eq(v2, GLM_VEC4_ONE)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_dot) { + vec4 a = {10.0f, 9.0f, 8.0f, 78.0f}; + vec4 b = {1.0f, 2.0f, 3.0f, 4.0f}; + float dot1, dot2; + + dot1 = GLM(vec4_dot)(a, b); + dot2 = a[0] * b[0] + a[1] * b[1] + a[2] * b[2] + a[3] * b[3]; + + ASSERT(test_eq(dot1, dot2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_norm2) { + vec4 a = {10.0f, 9.0f, 8.0f, 78.0f}; + float n1, n2; + + n1 = GLM(vec4_norm2)(a); + n2 = a[0] * a[0] + a[1] * a[1] + a[2] * a[2] + a[3] * a[3]; + + ASSERT(test_eq(n1, n2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_norm) { + vec4 a = {10.0f, 9.0f, 8.0f, 78.0f}; + float n1, n2; + + n1 = GLM(vec4_norm)(a); + n2 = sqrtf(a[0] * a[0] + a[1] * a[1] + a[2] * a[2] + a[3] * a[3]); + + ASSERT(test_eq(n1, n2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_norm_one) { + vec4 a = {-10.0f, 9.0f, -8.0f, 78.0f}; + float n1, n2; + + n1 = GLM(vec4_norm_one)(a); + n2 = fabsf(a[0]) + fabsf(a[1]) + fabsf(a[2]) + fabsf(a[3]); + + ASSERT(test_eq(n1, n2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_norm_inf) { + vec4 a = {-10.0f, 9.0f, -8.0f, 78.0f}; + float n1, n2; + + n1 = GLM(vec4_norm_inf)(a); + n2 = fabsf(a[0]); + + if (n2 < fabsf(a[1])) + n2 = fabsf(a[1]); + + if (n2 < fabsf(a[2])) + n2 = fabsf(a[2]); + + if (n2 < fabsf(a[3])) + n2 = fabsf(a[3]); + + ASSERT(test_eq(n1, n2)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_add) { + vec4 a = {-10.0f, 9.0f, -8.0f, 56.0f}; + vec4 b = {12.0f, 19.0f, -18.0f, 1.0f}; + vec4 c, d; + + c[0] = a[0] + b[0]; + c[1] = a[1] + b[1]; + c[2] = a[2] + b[2]; + c[3] = a[3] + b[3]; + + GLM(vec4_add)(a, b, d); + + ASSERTIFY(test_assert_vec4_eq(c, d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_adds) { + vec4 a = {-10.0f, 9.0f, -8.0f, 56.0f}; + vec4 c, d; + float s = 7.0f; + + c[0] = a[0] + s; + c[1] = a[1] + s; + c[2] = a[2] + s; + c[3] = a[3] + s; + + GLM(vec4_adds)(a, s, d); + + ASSERTIFY(test_assert_vec4_eq(c, d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_sub) { + vec4 a = {-10.0f, 9.0f, -8.0f, 56.0f}; + vec4 b = {12.0f, 19.0f, -18.0f, 1.0f}; + vec4 c, d; + + c[0] = a[0] - b[0]; + c[1] = a[1] - b[1]; + c[2] = a[2] - b[2]; + c[3] = a[3] - b[3]; + + GLM(vec4_sub)(a, b, d); + + ASSERTIFY(test_assert_vec4_eq(c, d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_subs) { + vec4 a = {-10.0f, 9.0f, -8.0f, 74.0f}; + vec4 c, d; + float s = 7.0f; + + c[0] = a[0] - s; + c[1] = a[1] - s; + c[2] = a[2] - s; + c[3] = a[3] - s; + + GLM(vec4_subs)(a, s, d); + + ASSERTIFY(test_assert_vec4_eq(c, d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_mul) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 56.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 46.0f}, + v3; + + GLM(vec4_mul)(v1, v2, v3); + + ASSERT(test_eq(v1[0] * v2[0], v3[0])) + ASSERT(test_eq(v1[1] * v2[1], v3[1])) + ASSERT(test_eq(v1[2] * v2[2], v3[2])) + ASSERT(test_eq(v1[3] * v2[3], v3[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_scale) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 5.0f}, v2; + float s = 7.0f; + + GLM(vec4_scale)(v1, s, v2); + + ASSERT(test_eq(v1[0] * s, v2[0])) + ASSERT(test_eq(v1[1] * s, v2[1])) + ASSERT(test_eq(v1[2] * s, v2[2])) + ASSERT(test_eq(v1[3] * s, v2[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_scale_as) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 5.0f}, v2; + float s = 7.0f; + float norm; + + GLM(vec4_scale_as)(v1, s, v2); + + norm = sqrtf(v1[0] * v1[0] + v1[1] * v1[1] + v1[2] * v1[2] + v1[3] * v1[3]); + if (norm < FLT_EPSILON) { + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + ASSERT(test_eq(v1[3], 0.0f)) + + TEST_SUCCESS + } + + norm = s / norm; + + ASSERT(test_eq(v1[0] * norm, v2[0])) + ASSERT(test_eq(v1[1] * norm, v2[1])) + ASSERT(test_eq(v1[2] * norm, v2[2])) + ASSERT(test_eq(v1[3] * norm, v2[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_div) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 40.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 2.0f}, + v3; + + GLM(vec4_div)(v1, v2, v3); + + ASSERT(test_eq(v1[0] / v2[0], v3[0])) + ASSERT(test_eq(v1[1] / v2[1], v3[1])) + ASSERT(test_eq(v1[2] / v2[2], v3[2])) + ASSERT(test_eq(v1[3] / v2[3], v3[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_divs) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 40.0f}, v2; + float s = 7.0f; + + GLM(vec4_divs)(v1, s, v2); + + ASSERT(test_eq(v1[0] / s, v2[0])) + ASSERT(test_eq(v1[1] / s, v2[1])) + ASSERT(test_eq(v1[2] / s, v2[2])) + ASSERT(test_eq(v1[3] / s, v2[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_addadd) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 20.0f}, + v3 = {1.0f, 2.0f, 3.0f, 130.0f}, + v4 = {1.0f, 2.0f, 3.0f, 130.0f}; + + GLM(vec4_addadd)(v1, v2, v4); + + ASSERT(test_eq(v3[0] + v1[0] + v2[0], v4[0])) + ASSERT(test_eq(v3[1] + v1[1] + v2[1], v4[1])) + ASSERT(test_eq(v3[2] + v1[2] + v2[2], v4[2])) + ASSERT(test_eq(v3[3] + v1[3] + v2[3], v4[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_subadd) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 20.0f}, + v3 = {1.0f, 2.0f, 3.0f, 130.0f}, + v4 = {1.0f, 2.0f, 3.0f, 130.0f}; + + GLM(vec4_subadd)(v1, v2, v4); + + ASSERT(test_eq(v3[0] + v1[0] - v2[0], v4[0])) + ASSERT(test_eq(v3[1] + v1[1] - v2[1], v4[1])) + ASSERT(test_eq(v3[2] + v1[2] - v2[2], v4[2])) + ASSERT(test_eq(v3[3] + v1[3] - v2[3], v4[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_muladd) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 20.0f}, + v3 = {1.0f, 2.0f, 3.0f, 130.0f}, + v4 = {1.0f, 2.0f, 3.0f, 130.0f}; + + GLM(vec4_muladd)(v1, v2, v4); + + ASSERT(test_eq(v3[0] + v1[0] * v2[0], v4[0])) + ASSERT(test_eq(v3[1] + v1[1] * v2[1], v4[1])) + ASSERT(test_eq(v3[2] + v1[2] * v2[2], v4[2])) + ASSERT(test_eq(v3[3] + v1[3] * v2[3], v4[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_muladds) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 20.0f}, + v3 = {-3.0f, 4.0f, -5.0f, 20.0f}; + float s = 9.0f; + + GLM(vec4_muladds)(v1, s, v3); + + ASSERT(test_eq(v2[0] + v1[0] * s, v3[0])) + ASSERT(test_eq(v2[1] + v1[1] * s, v3[1])) + ASSERT(test_eq(v2[2] + v1[2] * s, v3[2])) + ASSERT(test_eq(v2[3] + v1[3] * s, v3[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_maxadd) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 20.0f}, + v3 = {1.0f, 2.0f, 3.0f, 130.0f}, + v4 = {1.0f, 2.0f, 3.0f, 130.0f}; + + GLM(vec4_maxadd)(v1, v2, v4); + + ASSERT(test_eq(v3[0] + glm_max(v1[0], v2[0]), v4[0])) + ASSERT(test_eq(v3[1] + glm_max(v1[1], v2[1]), v4[1])) + ASSERT(test_eq(v3[2] + glm_max(v1[2], v2[2]), v4[2])) + ASSERT(test_eq(v3[3] + glm_max(v1[3], v2[3]), v4[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_minadd) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 20.0f}, + v3 = {1.0f, 2.0f, 3.0f, 130.0f}, + v4 = {1.0f, 2.0f, 3.0f, 130.0f}; + + GLM(vec4_minadd)(v1, v2, v4); + + ASSERT(test_eq(v3[0] + glm_min(v1[0], v2[0]), v4[0])) + ASSERT(test_eq(v3[1] + glm_min(v1[1], v2[1]), v4[1])) + ASSERT(test_eq(v3[2] + glm_min(v1[2], v2[2]), v4[2])) + ASSERT(test_eq(v3[3] + glm_min(v1[3], v2[3]), v4[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_subsub) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 20.0f}, + v3 = {1.0f, 2.0f, 3.0f, 130.0f}, + v4 = {1.0f, 2.0f, 3.0f, 130.0f}; + + GLM(vec4_subsub)(v1, v2, v4); + + ASSERT(test_eq(v3[0] - (v1[0] - v2[0]), v4[0])) + ASSERT(test_eq(v3[1] - (v1[1] - v2[1]), v4[1])) + ASSERT(test_eq(v3[2] - (v1[2] - v2[2]), v4[2])) + ASSERT(test_eq(v3[3] - (v1[3] - v2[3]), v4[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_addsub) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 20.0f}, + v3 = {1.0f, 2.0f, 3.0f, 130.0f}, + v4 = {1.0f, 2.0f, 3.0f, 130.0f}; + + GLM(vec4_addsub)(v1, v2, v4); + + ASSERT(test_eq(v3[0] - (v1[0] + v2[0]), v4[0])) + ASSERT(test_eq(v3[1] - (v1[1] + v2[1]), v4[1])) + ASSERT(test_eq(v3[2] - (v1[2] + v2[2]), v4[2])) + ASSERT(test_eq(v3[3] - (v1[3] + v2[3]), v4[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_mulsub) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 20.0f}, + v3 = {1.0f, 2.0f, 3.0f, 130.0f}, + v4 = {1.0f, 2.0f, 3.0f, 130.0f}; + + GLM(vec4_mulsub)(v1, v2, v4); + + ASSERT(test_eq(v3[0] - (v1[0] * v2[0]), v4[0])) + ASSERT(test_eq(v3[1] - (v1[1] * v2[1]), v4[1])) + ASSERT(test_eq(v3[2] - (v1[2] * v2[2]), v4[2])) + ASSERT(test_eq(v3[3] - (v1[3] * v2[3]), v4[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_mulsubs) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 20.0f}, + v3 = {-3.0f, 4.0f, -5.0f, 20.0f}; + float s = 9.0f; + + GLM(vec4_mulsubs)(v1, s, v3); + + ASSERT(test_eq(v2[0] - (v1[0] * s), v3[0])) + ASSERT(test_eq(v2[1] - (v1[1] * s), v3[1])) + ASSERT(test_eq(v2[2] - (v1[2] * s), v3[2])) + ASSERT(test_eq(v2[3] - (v1[3] * s), v3[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_maxsub) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 20.0f}, + v3 = {1.0f, 2.0f, 3.0f, 130.0f}, + v4 = {1.0f, 2.0f, 3.0f, 130.0f}; + + GLM(vec4_maxsub)(v1, v2, v4); + + ASSERT(test_eq(v3[0] - glm_max(v1[0], v2[0]), v4[0])) + ASSERT(test_eq(v3[1] - glm_max(v1[1], v2[1]), v4[1])) + ASSERT(test_eq(v3[2] - glm_max(v1[2], v2[2]), v4[2])) + ASSERT(test_eq(v3[3] - glm_max(v1[3], v2[3]), v4[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_minsub) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 4.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 20.0f}, + v3 = {1.0f, 2.0f, 3.0f, 130.0f}, + v4 = {1.0f, 2.0f, 3.0f, 130.0f}; + + GLM(vec4_minsub)(v1, v2, v4); + + ASSERT(test_eq(v3[0] - glm_min(v1[0], v2[0]), v4[0])) + ASSERT(test_eq(v3[1] - glm_min(v1[1], v2[1]), v4[1])) + ASSERT(test_eq(v3[2] - glm_min(v1[2], v2[2]), v4[2])) + ASSERT(test_eq(v3[3] - glm_min(v1[3], v2[3]), v4[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_negate_to) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 60.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 34.0f}, + v3, v4; + + GLM(vec4_negate_to)(v1, v3); + GLM(vec4_negate_to)(v2, v4); + + ASSERT(test_eq(-v1[0], v3[0])) + ASSERT(test_eq(-v1[1], v3[1])) + ASSERT(test_eq(-v1[2], v3[2])) + ASSERT(test_eq(-v1[3], v3[3])) + + ASSERT(test_eq(-v2[0], v4[0])) + ASSERT(test_eq(-v2[1], v4[1])) + ASSERT(test_eq(-v2[2], v4[2])) + ASSERT(test_eq(-v2[3], v4[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_negate) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 60.0f}, + v2 = {-3.0f, 4.0f, -5.0f, 34.0f}, + v3 = {2.0f, -3.0f, 4.0f, 60.0f}, + v4 = {-3.0f, 4.0f, -5.0f, 34.0f}; + + GLM(vec4_negate)(v1); + GLM(vec4_negate)(v2); + + ASSERT(test_eq(-v1[0], v3[0])) + ASSERT(test_eq(-v1[1], v3[1])) + ASSERT(test_eq(-v1[2], v3[2])) + ASSERT(test_eq(-v1[3], v3[3])) + + ASSERT(test_eq(-v2[0], v4[0])) + ASSERT(test_eq(-v2[1], v4[1])) + ASSERT(test_eq(-v2[2], v4[2])) + ASSERT(test_eq(-v2[3], v4[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_normalize) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 5.0f}, v2 = {2.0f, -3.0f, 4.0f, 5.0f}; + float s = 1.0f; + float norm; + + GLM(vec4_normalize)(v2); + + norm = sqrtf(v1[0] * v1[0] + v1[1] * v1[1] + v1[2] * v1[2] + v1[3] * v1[3]); + if (norm < FLT_EPSILON) { + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + ASSERT(test_eq(v1[3], 0.0f)) + + TEST_SUCCESS + } + + norm = s / norm; + + ASSERT(test_eq(v1[0] * norm, v2[0])) + ASSERT(test_eq(v1[1] * norm, v2[1])) + ASSERT(test_eq(v1[2] * norm, v2[2])) + ASSERT(test_eq(v1[3] * norm, v2[3])) + + glm_vec4_zero(v1); + GLM(vec4_normalize)(v1); + ASSERTIFY(test_assert_vec4_eq(v1, GLM_VEC4_ZERO)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_normalize_to) { + vec4 v1 = {2.0f, -3.0f, 4.0f, 5.0f}, v2; + float s = 1.0f; + float norm; + + GLM(vec4_normalize_to)(v1, v2); + + norm = sqrtf(v1[0] * v1[0] + v1[1] * v1[1] + v1[2] * v1[2] + v1[3] * v1[3]); + if (norm < FLT_EPSILON) { + ASSERT(test_eq(v1[0], 0.0f)) + ASSERT(test_eq(v1[1], 0.0f)) + ASSERT(test_eq(v1[2], 0.0f)) + ASSERT(test_eq(v1[3], 0.0f)) + + TEST_SUCCESS + } + + norm = s / norm; + + ASSERT(test_eq(v1[0] * norm, v2[0])) + ASSERT(test_eq(v1[1] * norm, v2[1])) + ASSERT(test_eq(v1[2] * norm, v2[2])) + ASSERT(test_eq(v1[3] * norm, v2[3])) + + glm_vec4_zero(v1); + GLM(vec4_normalize_to)(v1, v2); + ASSERTIFY(test_assert_vec4_eq(v2, GLM_VEC4_ZERO)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_distance2) { + vec4 v1 = {30.0f, 0.0f, 0.0f, 0.0f}, + v2 = {0.0f, 0.0f, 0.0f, 0.0f}, + v3 = {3.0f, 10.0f, 120.0f, 140.0f}, + v4 = {0.46f, 4.0f, 14.0f, 10.0f}; + float d; + + d = GLM(vec4_distance2)(v1, v2); + ASSERT(test_eq(d, 30.0f * 30.0f)) + + d = GLM(vec4_distance2)(v3, v4); + ASSERT(test_eq(powf(v3[0] - v4[0], 2.0f) + + powf(v3[1] - v4[1], 2.0f) + + powf(v3[2] - v4[2], 2.0f) + + powf(v3[3] - v4[3], 2.0f), d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_distance) { + vec4 v1 = {30.0f, 0.0f, 0.0f, 0.0f}, + v2 = {0.0f, 0.0f, 0.0f, 0.0f}, + v3 = {3.0f, 10.0f, 120.0f, 140.0f}, + v4 = {0.46f, 4.0f, 14.0f, 10.0f}; + float d; + + d = GLM(vec4_distance)(v1, v2); + ASSERT(test_eq(d, 30.0f)) + + d = GLM(vec4_distance)(v3, v4); + ASSERT(test_eq(sqrtf(powf(v3[0] - v4[0], 2.0f) + + powf(v3[1] - v4[1], 2.0f) + + powf(v3[2] - v4[2], 2.0f) + + powf(v3[3] - v4[3], 2.0f)), d)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_maxv) { + vec4 v1, v2, v3; + vec4 v5 = {-1.456f, -1.456f, 241.456f, 10.0f}; + vec4 v6 = {11.0f, 11.0f, 11.0f, 90.0f}; + vec4 v7 = {78.0f, -78.0f, 7.0f, 5.0f}; + + GLM(vec4_maxv)(v5, v6, v1); + GLM(vec4_maxv)(v5, v7, v2); + GLM(vec4_maxv)(v6, v7, v3); + + ASSERT(test_eq(v1[0], 11.0f)) + ASSERT(test_eq(v1[1], 11.0f)) + ASSERT(test_eq(v1[2], 241.456f)) + ASSERT(test_eq(v1[3], 90.0f)) + + ASSERT(test_eq(v2[0], 78.0f)) + ASSERT(test_eq(v2[1], -1.456f)) + ASSERT(test_eq(v2[2], 241.456f)) + ASSERT(test_eq(v2[3], 10.0f)) + + ASSERT(test_eq(v3[0], 78.0f)) + ASSERT(test_eq(v3[1], 11.0f)) + ASSERT(test_eq(v3[2], 11.0f)) + ASSERT(test_eq(v3[3], 90.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_minv) { + vec4 v1, v2, v3; + vec4 v5 = {-1.456f, -1.456f, 241.456f, 10.0f}; + vec4 v6 = {11.0f, 11.0f, 11.0f, 90.0f}; + vec4 v7 = {78.0f, -78.0f, 7.0f, 5.0f}; + + GLM(vec4_minv)(v5, v6, v1); + GLM(vec4_minv)(v5, v7, v2); + GLM(vec4_minv)(v6, v7, v3); + + ASSERT(test_eq(v1[0], -1.456f)) + ASSERT(test_eq(v1[1], -1.456f)) + ASSERT(test_eq(v1[2], 11.0f)) + ASSERT(test_eq(v1[3], 10.0f)) + + ASSERT(test_eq(v2[0], -1.456f)) + ASSERT(test_eq(v2[1], -78.0f)) + ASSERT(test_eq(v2[2], 7.0f)) + ASSERT(test_eq(v2[3], 5.0f)) + + ASSERT(test_eq(v3[0], 11.0f)) + ASSERT(test_eq(v3[1], -78.0f)) + ASSERT(test_eq(v3[2], 7.0f)) + ASSERT(test_eq(v3[3], 5.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_clamp) { + vec4 v1 = {-1.456f, -11.456f, 31.456f, 67.04f}; + vec4 v2 = {0.110f, 111.0f, 11.0f, 90.0f}; + vec4 v3 = {78.0f, 32.0f, -78.0f, 3.0f}; + + GLM(vec4_clamp)(v1, -1.03f, 30.0f); + GLM(vec4_clamp)(v2, 0.11f, 111.0f); + GLM(vec4_clamp)(v3, -88.0f, 70.0f); + + ASSERT(test_eq(v1[0], -1.03f)) + ASSERT(test_eq(v1[1], -1.03f)) + ASSERT(test_eq(v1[2], 30.0f)) + ASSERT(test_eq(v1[3], 30.0f)) + + ASSERT(test_eq(v2[0], 0.11f)) + ASSERT(test_eq(v2[1], 111.0f)) + ASSERT(test_eq(v2[2], 11.0f)) + ASSERT(test_eq(v2[3], 90.0f)) + + ASSERT(test_eq(v3[0], 70.0f)) + ASSERT(test_eq(v3[1], 32.0f)) + ASSERT(test_eq(v3[2], -78.0f)) + ASSERT(test_eq(v3[3], 3.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_lerp) { + vec4 v1 = {-100.0f, -200.0f, -10.0f, -10.0f}; + vec4 v2 = {100.0f, 200.0f, 10.0f, 10.0f}; + vec4 v3; + + GLM(vec4_lerp)(v1, v2, 0.5f, v3); + ASSERT(test_eq(v3[0], 0.0f)) + ASSERT(test_eq(v3[1], 0.0f)) + ASSERT(test_eq(v3[2], 0.0f)) + ASSERT(test_eq(v3[3], 0.0f)) + + GLM(vec4_lerp)(v1, v2, 0.75f, v3); + ASSERT(test_eq(v3[0], 50.0f)) + ASSERT(test_eq(v3[1], 100.0f)) + ASSERT(test_eq(v3[2], 5.0f)) + ASSERT(test_eq(v3[3], 5.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_lerpc) { + vec4 v1 = {-100.0f, -200.0f, -10.0f, -10.0f}; + vec4 v2 = {100.0f, 200.0f, 10.0f, 10.0f}; + vec4 v3; + + GLM(vec4_lerpc)(v1, v2, 0.5f, v3); + ASSERT(test_eq(v3[0], 0.0f)) + ASSERT(test_eq(v3[1], 0.0f)) + ASSERT(test_eq(v3[2], 0.0f)) + ASSERT(test_eq(v3[3], 0.0f)) + + GLM(vec4_lerpc)(v1, v2, 0.75f, v3); + ASSERT(test_eq(v3[0], 50.0f)) + ASSERT(test_eq(v3[1], 100.0f)) + ASSERT(test_eq(v3[2], 5.0f)) + ASSERT(test_eq(v3[3], 5.0f)) + + GLM(vec4_lerpc)(v1, v2, -1.75f, v3); + ASSERT(test_eq(v3[0], -100.0f)) + ASSERT(test_eq(v3[1], -200.0f)) + ASSERT(test_eq(v3[2], -10.0f)) + ASSERT(test_eq(v3[3], -10.0f)) + + GLM(vec4_lerpc)(v1, v2, 1.75f, v3); + ASSERT(test_eq(v3[0], 100.0f)) + ASSERT(test_eq(v3[1], 200.0f)) + ASSERT(test_eq(v3[2], 10.0f)) + ASSERT(test_eq(v3[3], 10.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_mix) { + vec4 v1 = {-100.0f, -200.0f, -10.0f, -10.0f}; + vec4 v2 = {100.0f, 200.0f, 10.0f, 10.0f}; + vec4 v3; + + GLM(vec4_mix)(v1, v2, 0.5f, v3); + ASSERT(test_eq(v3[0], 0.0f)) + ASSERT(test_eq(v3[1], 0.0f)) + ASSERT(test_eq(v3[2], 0.0f)) + ASSERT(test_eq(v3[3], 0.0f)) + + GLM(vec4_mix)(v1, v2, 0.75f, v3); + ASSERT(test_eq(v3[0], 50.0f)) + ASSERT(test_eq(v3[1], 100.0f)) + ASSERT(test_eq(v3[2], 5.0f)) + ASSERT(test_eq(v3[3], 5.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_mixc) { + vec4 v1 = {-100.0f, -200.0f, -10.0f, -10.0f}; + vec4 v2 = {100.0f, 200.0f, 10.0f, 10.0f}; + vec4 v3; + + GLM(vec4_mixc)(v1, v2, 0.5f, v3); + ASSERT(test_eq(v3[0], 0.0f)) + ASSERT(test_eq(v3[1], 0.0f)) + ASSERT(test_eq(v3[2], 0.0f)) + ASSERT(test_eq(v3[3], 0.0f)) + + GLM(vec4_mixc)(v1, v2, 0.75f, v3); + ASSERT(test_eq(v3[0], 50.0f)) + ASSERT(test_eq(v3[1], 100.0f)) + ASSERT(test_eq(v3[2], 5.0f)) + ASSERT(test_eq(v3[3], 5.0f)) + + GLM(vec4_mixc)(v1, v2, -1.75f, v3); + ASSERT(test_eq(v3[0], -100.0f)) + ASSERT(test_eq(v3[1], -200.0f)) + ASSERT(test_eq(v3[2], -10.0f)) + ASSERT(test_eq(v3[3], -10.0f)) + + GLM(vec4_mixc)(v1, v2, 1.75f, v3); + ASSERT(test_eq(v3[0], 100.0f)) + ASSERT(test_eq(v3[1], 200.0f)) + ASSERT(test_eq(v3[2], 10.0f)) + ASSERT(test_eq(v3[3], 10.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_steps) { + vec4 v1 = {-100.0f, -200.0f, -10.0f, -10.0f}; + vec4 v2; + + GLM(vec4_steps)(-2.5f, v1, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + ASSERT(test_eq(v2[2], 0.0f)) + ASSERT(test_eq(v2[3], 0.0f)) + + GLM(vec4_steps)(-10.0f, v1, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + GLM(vec4_steps)(-1000.0f, v1, v2); + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_stepr) { + vec4 v1 = {-2.5f, -100.0f, -200.0f, -300.0f}; + vec4 v2; + + GLM(vec4_stepr)(v1, -1000.0f, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + ASSERT(test_eq(v2[2], 0.0f)) + ASSERT(test_eq(v2[3], 0.0f)) + + GLM(vec4_stepr)(v1, -250.0f, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + ASSERT(test_eq(v2[2], 0.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + GLM(vec4_stepr)(v1, -150.0f, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + GLM(vec4_stepr)(v1, -10.0f, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + GLM(vec4_stepr)(v1, 0.0f, v2); + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_step) { + vec4 v1 = {-100.0f, -200.0f, -10.0f, -10.0f}; + vec4 s1 = {-100.0f, 0.0f, 10.0f, 10.0f}; + vec4 s2 = {100.0f, -220.0f, -10.0f, -10.0f}; + vec4 s3 = {100.0f, 200.0f, 10.0f, 10.0f}; + vec4 v2; + + GLM(vec4_step)(s1, v1, v2); + ASSERT(test_eq(v2[0], 1.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + ASSERT(test_eq(v2[2], 0.0f)) + ASSERT(test_eq(v2[3], 0.0f)) + + GLM(vec4_step)(s2, v1, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 1.0f)) + ASSERT(test_eq(v2[2], 1.0f)) + ASSERT(test_eq(v2[3], 1.0f)) + + GLM(vec4_step)(s3, v1, v2); + ASSERT(test_eq(v2[0], 0.0f)) + ASSERT(test_eq(v2[1], 0.0f)) + ASSERT(test_eq(v2[2], 0.0f)) + ASSERT(test_eq(v2[3], 0.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_smoothstep_uni) { + vec4 v1 = {-100.0f, -200.0f, -10.0f, -10.0f}; + vec4 v2; + + GLM(vec4_smoothstep_uni)(-200.0f, -100.0f, v1, v2); + ASSERT(test_eq_th(v2[0], 1.0f, 1e-5f)) + ASSERT(test_eq_th(v2[1], 0.0f, 1e-5f)) + ASSERT(test_eq_th(v2[2], 1.0f, 1e-5f)) + ASSERT(test_eq_th(v2[3], 1.0f, 1e-5f)) + + GLM(vec4_smoothstep_uni)(-250.0f, -200.0f, v1, v2); + ASSERT(test_eq_th(v2[0], 1.0f, 1e-5f)) + ASSERT(test_eq_th(v2[1], 1.0f, 1e-5f)) + ASSERT(test_eq_th(v2[2], 1.0f, 1e-5f)) + ASSERT(test_eq_th(v2[3], 1.0f, 1e-5f)) + + GLM(vec4_smoothstep_uni)(-200.0f, 200.0f, v1, v2); + ASSERT(v2[0] > 0.0f && v2[0] < 0.25f) + ASSERT(test_eq(v2[1], 0.0f)) + ASSERT(v2[2] > 0.0f && v2[2] < 0.5f) + ASSERT(v2[3] > 0.0f && v2[3] < 0.5f) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_smoothstep) { + vec4 v1 = {-100.0f, -200.0f, -10.0f, -10.0f}; + vec4 e1_0 = {-100.0f, 0.0f, -11.0f, -11.0f}; + vec4 e1_1 = {50.0f, 10.0f, 20.0f, 20.0f}; + vec4 e2_0 = {-180.0f, -300.0f, -93.0f, -93.0f}; + vec4 e2_1 = {100.0f, 120.0f, -10.0f, -10.0f}; + vec4 e3_0 = {-12.0f, 100.0f, 0.0f, 0.0f}; + vec4 e3_1 = {100.0f, 200.0f, 10.0f, 10.0f}; + vec4 v2; + + GLM(vec4_smoothstep)(e1_0, e1_1, v1, v2); + ASSERT(test_eq_th(v2[0], 0.0f, 1e-5f)) + ASSERT(test_eq_th(v2[1], 0.0f, 1e-5f)) + ASSERT(v2[2] > 0.0f && v2[2] < 0.1f) + ASSERT(v2[3] > 0.0f && v2[3] < 0.1f) + + GLM(vec4_smoothstep)(e2_0, e2_1, v1, v2); + ASSERT(v2[0] > 0.0f && v2[0] < 0.25f) + ASSERT(v2[1] > 0.0f && v2[1] < 0.15f) + ASSERT(test_eq_th(v2[2], 1.0f, 1e-5f)) + ASSERT(test_eq_th(v2[3], 1.0f, 1e-5f)) + + GLM(vec4_smoothstep)(e3_0, e3_1, v1, v2); + ASSERT(test_eq_th(v2[0], 0.0f, 1e-5f)) + ASSERT(test_eq_th(v2[1], 0.0f, 1e-5f)) + ASSERT(test_eq_th(v2[2], 0.0f, 1e-5f)) + ASSERT(test_eq_th(v2[3], 0.0f, 1e-5f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_smoothinterp) { + vec4 e1_0 = {-100.0f, 0.0f, -11.0f, -11.0f}; + vec4 e1_1 = {50.0f, 10.0f, 20.0f, 20.0f}; + vec4 e2_0 = {80.0f, -220.0f, -19.0f, -19.0f}; + vec4 e2_1 = {100.0f, -200.0f, -10.0f, -10.0f}; + vec4 e3_0 = {-12.0f, 100.0f, 0.0f, 0.0f}; + vec4 e3_1 = {100.0f, 200.0f, 10.0f, 10.0f}; + vec4 v2; + + GLM(vec4_smoothinterp)(e1_0, e1_1, 0.5f, v2); + ASSERT(v2[0] >= e1_0[0] && v2[0] <= e1_1[0]) + ASSERT(v2[1] >= e1_0[1] && v2[1] <= e1_1[1]) + ASSERT(v2[2] >= e1_0[2] && v2[2] <= e1_1[2]) + ASSERT(v2[3] >= e1_0[3] && v2[3] <= e1_1[3]) + + GLM(vec4_smoothinterp)(e2_0, e2_1, 0.5f, v2); + ASSERT(v2[0] >= e2_0[0] && v2[0] <= e2_1[0]) + ASSERT(v2[1] >= e2_0[1] && v2[1] <= e2_1[1]) + ASSERT(v2[2] >= e2_0[2] && v2[2] <= e2_1[2]) + ASSERT(v2[3] >= e2_0[3] && v2[3] <= e2_1[3]) + + GLM(vec4_smoothinterp)(e3_0, e3_1, 1.0f, v2); + ASSERT(v2[0] >= e3_0[0] && v2[0] <= e3_1[0]) + ASSERT(v2[1] >= e3_0[1] && v2[1] <= e3_1[1]) + ASSERT(v2[2] >= e3_0[2] && v2[2] <= e3_1[2]) + ASSERT(v2[3] >= e3_0[3] && v2[3] <= e3_1[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_smoothinterpc) { + vec4 e1_0 = {-100.0f, 0.0f, -11.0f, -11.0f}; + vec4 e1_1 = {50.0f, 10.0f, 20.0f, 20.0f}; + vec4 e2_0 = {80.0f, -220.0f, -19.0f, -19.0f}; + vec4 e2_1 = {100.0f, -200.0f, -10.0f, -10.0f}; + vec4 e3_0 = {-12.0f, 100.0f, 0.0f, 0.0f}; + vec4 e3_1 = {100.0f, 200.0f, 10.0f, 10.0f}; + vec4 v2; + + GLM(vec4_smoothinterpc)(e1_0, e1_1, -0.5f, v2); + ASSERT(v2[0] >= e1_0[0] && v2[0] <= e1_1[0]) + ASSERT(v2[1] >= e1_0[1] && v2[1] <= e1_1[1]) + ASSERT(v2[2] >= e1_0[2] && v2[2] <= e1_1[2]) + ASSERT(v2[3] >= e1_0[3] && v2[3] <= e1_1[3]) + + GLM(vec4_smoothinterpc)(e2_0, e2_1, 0.5f, v2); + ASSERT(v2[0] >= e2_0[0] && v2[0] <= e2_1[0]) + ASSERT(v2[1] >= e2_0[1] && v2[1] <= e2_1[1]) + ASSERT(v2[2] >= e2_0[2] && v2[2] <= e2_1[2]) + ASSERT(v2[3] >= e2_0[3] && v2[3] <= e2_1[3]) + + GLM(vec4_smoothinterpc)(e3_0, e3_1, 2.0f, v2); + ASSERT(v2[0] >= e3_0[0] && v2[0] <= e3_1[0]) + ASSERT(v2[1] >= e3_0[1] && v2[1] <= e3_1[1]) + ASSERT(v2[2] >= e3_0[2] && v2[2] <= e3_1[2]) + ASSERT(v2[3] >= e3_0[3] && v2[3] <= e3_1[3]) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_cubic) { + vec4 v1 = {125.0f, 25.0f, 5.0f, 1.0f}; + vec4 v2 = {216.0f, 36.0f, 6.0f, 1.0f}; + vec4 v3, v4; + + ASSERT(test_eq(v1[0], v1[2] * v1[2] * v1[2])) + ASSERT(test_eq(v1[1], v1[2] * v1[2])) + ASSERT(test_eq(v1[3], 1.0f)) + + ASSERT(test_eq(v2[0], v2[2] * v2[2] * v2[2])) + ASSERT(test_eq(v2[1], v2[2] * v2[2])) + ASSERT(test_eq(v2[3], 1.0f)) + + GLM(vec4_cubic)(test_rand(), v3); + ASSERT(test_eq(v3[0], v3[2] * v3[2] * v3[2])) + ASSERT(test_eq(v3[1], v3[2] * v3[2])) + ASSERT(test_eq(v3[3], 1.0f)) + + GLM(vec4_cubic)(test_rand(), v4); + ASSERT(test_eq(v4[0], v4[2] * v4[2] * v4[2])) + ASSERT(test_eq(v4[1], v4[2] * v4[2])) + ASSERT(test_eq(v4[3], 1.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_swizzle) { + vec4 v; + + /* ZYX */ + v[0] = 1; + v[1] = 2; + v[2] = 3; + v[3] = 4; + + GLM(vec4_swizzle)(v, GLM_WZYX, v); + ASSERTIFY(test_assert_vec4_eq(v, (vec4){4, 3, 2, 1})) + + GLM(vec4_swizzle)(v, GLM_XXXX, v); + ASSERTIFY(test_assert_vec4_eq(v, (vec4){4, 4, 4, 4})) + + v[0] = 1; + v[1] = 2; + v[2] = 3; + v[3] = 4; + + GLM(vec4_swizzle)(v, GLM_YYYY, v); + ASSERTIFY(test_assert_vec4_eq(v, (vec4){2, 2, 2, 2})) + + v[0] = 1; + v[1] = 2; + v[2] = 3; + v[3] = 4; + + GLM(vec4_swizzle)(v, GLM_ZZZZ, v); + ASSERTIFY(test_assert_vec4_eq(v, (vec4){3, 3, 3, 3})) + + v[0] = 1; + v[1] = 2; + v[2] = 3; + v[3] = 4; + + GLM(vec4_swizzle)(v, GLM_WWWW, v); + ASSERTIFY(test_assert_vec4_eq(v, (vec4){4, 4, 4, 4})) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_broadcast) { + vec4 v1, v2, v3; + vec4 v5 = {-1.456f, -1.456f, -1.456f, -1.456f}; + vec4 v6 = {11.0f, 11.0f, 11.0f, 11.0f}; + vec4 v7 = {78.0f, 78.0f, 78.0f, 78.0f}; + + GLM(vec4_broadcast)(-1.456f, v1); + GLM(vec4_broadcast)(11.0f, v2); + GLM(vec4_broadcast)(78.0f, v3); + + ASSERTIFY(test_assert_vec4_eq(v1, v5)) + ASSERTIFY(test_assert_vec4_eq(v2, v6)) + ASSERTIFY(test_assert_vec4_eq(v3, v7)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_fill) { + vec4 v1, v2, v3; + vec4 v5 = {-1.456f, -1.456f, -1.456f, -1.456f}; + vec4 v6 = {11.0f, 11.0f, 11.0f, 11.0f}; + vec4 v7 = {78.0f, 78.0f, 78.0f, 78.0f}; + + GLM(vec4_fill)(v1, -1.456f); + GLM(vec4_fill)(v2, 11.0f); + GLM(vec4_fill)(v3, 78.0f); + + ASSERTIFY(test_assert_vec4_eq(v1, v5)) + ASSERTIFY(test_assert_vec4_eq(v2, v6)) + ASSERTIFY(test_assert_vec4_eq(v3, v7)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_eq) { + vec4 v1, v2, v3; + + GLM(vec4_fill)(v1, -1.456f); + GLM(vec4_fill)(v2, 11.0f); + GLM(vec4_fill)(v3, 78.1f); + + ASSERT(GLM(vec4_eq)(v1, -1.456f)) + ASSERT(GLM(vec4_eq)(v2, 11.0f)) + ASSERT(!GLM(vec4_eq)(v3, 78.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_eq_eps) { + vec4 v1, v2, v3; + + GLM(vec4_fill)(v1, -1.456f); + GLM(vec4_fill)(v2, 11.0f); + GLM(vec4_fill)(v3, 78.1f); + + ASSERT(GLM(vec4_eq_eps)(v1, -1.456f)) + ASSERT(GLM(vec4_eq_eps)(v2, 11.0f)) + ASSERT(!GLM(vec4_eq_eps)(v3, 78.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_eq_all) { + vec4 v1, v2, v3; + vec4 v4 = {2.104f, -3.012f, -4.10f, -4.10f}; + vec4 v5 = {-12.35f, -31.140f, -43.502f, -43.502f}; + + GLM(vec4_fill)(v1, -1.456f); + GLM(vec4_fill)(v2, 11.0f); + GLM(vec4_fill)(v3, 78.0f); + + ASSERT(GLM(vec4_eq_all)(v1)) + ASSERT(GLM(vec4_eq_all)(v2)) + ASSERT(GLM(vec4_eq_all)(v3)) + ASSERT(!GLM(vec4_eq_all)(v4)) + ASSERT(!GLM(vec4_eq_all)(v5)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_eqv) { + vec4 v1, v2, v3, v4, v5; + vec4 v6 = {-1.456f, -1.456f, -1.456f, -1.456f}; + vec4 v7 = {11.0f, 11.0f, 11.0f, 11.0f}; + vec4 v8 = {78.0f, 78.0f, -43.502f, -43.502f}; + + GLM(vec4_fill)(v1, -1.456f); + GLM(vec4_fill)(v2, 11.0f); + GLM(vec4_fill)(v3, 78.0f); + + test_rand_vec4(v4); + test_rand_vec4(v5); + + ASSERT(GLM(vec4_eqv)(v1, v6)) + ASSERT(GLM(vec4_eqv)(v2, v7)) + ASSERT(!GLM(vec4_eqv)(v3, v8)) + ASSERT(!GLM(vec4_eqv)(v4, v5)) + ASSERT(GLM(vec4_eqv)(v5, v5)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_eqv_eps) { + vec4 v1, v2, v3, v4, v5; + vec4 v6 = {-1.456f, -1.456f, -1.456f, -1.456f}; + vec4 v7 = {11.0f, 11.0f, 11.0f, 11.0f}; + vec4 v8 = {78.0f, 78.0f, -43.502f, -43.502f}; + + GLM(vec4_fill)(v1, -1.456f); + GLM(vec4_fill)(v2, 11.0f); + GLM(vec4_fill)(v3, 78.0f); + + test_rand_vec4(v4); + test_rand_vec4(v5); + + ASSERT(GLM(vec4_eqv_eps)(v1, v6)) + ASSERT(GLM(vec4_eqv_eps)(v2, v7)) + ASSERT(!GLM(vec4_eqv_eps)(v3, v8)) + ASSERT(!GLM(vec4_eqv_eps)(v4, v5)) + ASSERT(GLM(vec4_eqv_eps)(v5, v5)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_max) { + vec4 v1 = {2.104f, -3.012f, -4.10f, -4.10f}; + vec4 v2 = {-12.35f, -31.140f, -43.502f, -43.502f}; +#ifndef CGLM_FAST_MATH + vec4 v3 = {INFINITY, 0.0f, 0.0f, 0.0f}; +#endif +// vec4 v4 = {NAN, INFINITY, 2.0f, 2.0f}; +// vec4 v5 = {NAN, -1.0f, -1.0f, -1.0f}; + vec4 v6 = {-1.0f, -11.0f, 11.0f, 11.0f}; + + ASSERT(test_eq(GLM(vec4_max)(v1), 2.104f)) + ASSERT(test_eq(GLM(vec4_max)(v2), -12.35f)) +#ifndef CGLM_FAST_MATH + ASSERT(isinf(GLM(vec4_max)(v3))) +#endif +// ASSERT(isnan(GLM(vec4_max)(v4))) +// ASSERT(isnan(GLM(vec4_max)(v5))) + ASSERT(test_eq(GLM(vec4_max)(v6), 11.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_min) { + vec4 v1 = {2.104f, -3.012f, -4.10f, -4.10f}; + vec4 v2 = {-12.35f, -31.140f, -43.502f, -43.502f}; +#ifndef CGLM_FAST_MATH + vec4 v3 = {INFINITY, 0.0f, 0.0f, 0.0f}; +#endif +// vec4 v4 = {NAN, INFINITY, 2.0f, 2.0f}; +// vec4 v5 = {NAN, -1.0f, -1.0f, -1.0f}; + vec4 v6 = {-1.0f, -11.0f, 11.0f, 11.0f}; + + ASSERT(test_eq(GLM(vec4_min)(v1), -4.10f)) + ASSERT(test_eq(GLM(vec4_min)(v2), -43.502f)) +#ifndef CGLM_FAST_MATH + ASSERT(test_eq(GLM(vec4_min)(v3), 0.0f)) +#endif +// ASSERT(isnan(GLM(vec4_min)(v4))) +// ASSERT(isnan(GLM(vec4_min)(v5))) + ASSERT(test_eq(GLM(vec4_min)(v6), -11.0f)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_isnan) { +#ifndef CGLM_FAST_MATH + vec4 v1 = {2.104f, -3.012f, -4.10f, -4.10f}; + vec4 v2 = {-12.35f, -31.140f, -43.502f, -43.502f}; + vec4 v3 = {INFINITY, 0.0f, 0.0f, 0.0f}; + vec4 v4 = {NAN, INFINITY, 2.0f, 2.0f}; + vec4 v5 = {NAN, -1.0f, -1.0f, -1.0f}; + vec4 v6 = {-1.0f, -1.0f, 11.0f, 11.0f}; + + ASSERT(!GLM(vec4_isnan)(v1)) + ASSERT(!GLM(vec4_isnan)(v2)) + ASSERT(!GLM(vec4_isnan)(v3)) + ASSERT(GLM(vec4_isnan)(v4)) + ASSERT(GLM(vec4_isnan)(v5)) + ASSERT(!GLM(vec4_isnan)(v6)) +#endif + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_isinf) { +#ifndef CGLM_FAST_MATH + vec4 v1 = {2.104f, -3.012f, -4.10f, -4.10f}; + vec4 v2 = {-12.35f, -31.140f, -43.502f, -43.502f}; + vec4 v3 = {INFINITY, 0.0f, 0.0f, 0.0f}; + vec4 v4 = {NAN, INFINITY, 2.0f, 2.0f}; + vec4 v5 = {NAN, -1.0f, -1.0f, -1.0f}; + vec4 v6 = {-1.0f, -1.0f, 11.0f, 11.0f}; + + ASSERT(!GLM(vec4_isinf)(v1)) + ASSERT(!GLM(vec4_isinf)(v2)) + ASSERT(GLM(vec4_isinf)(v3)) + ASSERT(GLM(vec4_isinf)(v4)) + ASSERT(!GLM(vec4_isinf)(v5)) + ASSERT(!GLM(vec4_isinf)(v6)) +#endif + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_isvalid) { +#ifndef CGLM_FAST_MATH + vec4 v1 = {2.104f, -3.012f, -4.10f, -4.10f}; + vec4 v2 = {-12.35f, -31.140f, -43.502f, -43.502f}; + vec4 v3 = {INFINITY, 0.0f, 0.0f, 0.0f}; + vec4 v4 = {NAN, INFINITY, 2.0f, 2.0f}; + vec4 v5 = {NAN, -1.0f, -1.0f, -1.0f}; + vec4 v6 = {-1.0f, -1.0f, 11.0f, 11.0f}; + + ASSERT(GLM(vec4_isvalid)(v1)) + ASSERT(GLM(vec4_isvalid)(v2)) + ASSERT(!GLM(vec4_isvalid)(v3)) + ASSERT(!GLM(vec4_isvalid)(v4)) + ASSERT(!GLM(vec4_isvalid)(v5)) + ASSERT(GLM(vec4_isvalid)(v6)) +#endif + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_sign) { + vec4 v1 = {2.104f, -3.012f, -4.10f, -4.10f}; + vec4 v2 = {-12.35f, -31.140f, -43.502f, -43.502f}; + vec4 v3, v4; + vec4 v5 = {1.0f, -1.0f, -1.0f, -1.0f}; + vec4 v6 = {-1.0f, -1.0f, -1.0f, -1.0f}; + + GLM(vec4_sign)(v1, v3); + GLM(vec4_sign)(v2, v4); + + ASSERTIFY(test_assert_vec4_eq(v3, v5)) + ASSERTIFY(test_assert_vec4_eq(v4, v6)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_abs) { + vec4 v1 = {2.104f, -3.012f, -4.10f, -4.10f}; + vec4 v2 = {-12.35f, -31.140f, -43.502f, -43.502f}; + vec4 v3, v4; + vec4 v5 = {2.104f, 3.012f, 4.10f, 4.10f}; + vec4 v6 = {12.35f, 31.140f, 43.502f, 43.502f}; + + GLM(vec4_abs)(v1, v3); + GLM(vec4_abs)(v2, v4); + + ASSERTIFY(test_assert_vec4_eq(v3, v5)) + ASSERTIFY(test_assert_vec4_eq(v4, v6)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_fract) { + vec4 v1 = {2.104f, 3.012f, 4.10f, 4.10f}; + vec4 v2 = {12.35f, 31.140f, 43.502f, 43.502f}; + vec4 v3, v4; + vec4 v5 = {0.104f, 0.012f, 0.10f, 0.10f}; + vec4 v6 = {0.35f, 0.140f, 0.502f, 0.502f}; + + GLM(vec4_fract)(v1, v3); + GLM(vec4_fract)(v2, v4); + + ASSERTIFY(test_assert_vec4_eq(v3, v5)) + ASSERTIFY(test_assert_vec4_eq(v4, v6)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_floor) { + vec4 v1 = {2.104f, 3.012f, 4.10f, 4.10f}; + vec4 v2 = {12.35f, 31.140f, 43.502f, 43.502f}; + vec4 v3, v4; + vec4 v5 = {2.0f, 3.0f, 4.0f, 4.0f}; + vec4 v6 = {12.0f, 31.0f, 43.0f, 43.0f}; + + GLM(vec4_floor)(v1, v3); + GLM(vec4_floor)(v2, v4); + + ASSERTIFY(test_assert_vec4_eq(v3, v5)) + ASSERTIFY(test_assert_vec4_eq(v4, v6)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_mods) { + vec4 v1 = {2.104f, 3.012f, 4.10f, 5.78f}, v2 = {12.35f, 31.140f, 43.502f, 198.999f}; + vec4 v3, v4; + vec4 v5 = {0.104f, 0.012f, 0.10f, 0.78f}, v6 = {0.35f, 0.140f, 0.502f, 0.999f}; + + /* Mod 1 - leaves just the fractional part */ + GLM(vec4_mods)(v1, 1.0f, v3); + GLM(vec4_mods)(v2, 1.0f, v4); + + ASSERTIFY(test_assert_vec4_eq(v3, v5)) + ASSERTIFY(test_assert_vec4_eq(v4, v6)) + + /* Mod 2 - parity + fractional part */ + GLM(vec4_mods)(v1, 2.0f, v3); + GLM(vec4_mods)(v2, 2.0f, v4); + + vec4 v7 = {0.104f, 1.012f, 0.10f, 1.78f}, v8 = {0.35f, 1.140f, 1.502f, 0.999f}; + + ASSERTIFY(test_assert_vec4_eq(v3, v7)) + ASSERTIFY(test_assert_vec4_eq(v4, v8)) + + TEST_SUCCESS +} + + +TEST_IMPL(GLM_PREFIX, vec4_hadd) { + vec4 v1 = {2.0f, 3.0f, 4.0f, 4.0f}, v2 = {12.0f, 31.0f, 43.0f, 43.0f}; + float r1, r2, r3, r4; + + r1 = GLM(vec4_hadd)(v1); + r2 = GLM(vec4_hadd)(v2); + + r3 = v1[0] + v1[1] + v1[2] + v1[3]; + r4 = v2[0] + v2[1] + v2[2] + v2[3]; + + ASSERT(test_eq(r1, r3)) + ASSERT(test_eq(r2, r4)) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_sqrt) { + vec4 v1 = {2.0f, 3.0f, 4.0f, 4.0f}, v2 = {12.0f, 31.0f, 43.0f, 43.0f}; + vec4 v3, v4; + + GLM(vec4_sqrt)(v1, v3); + GLM(vec4_sqrt)(v2, v4); + + ASSERT(test_eq(sqrtf(v1[0]), v3[0])) + ASSERT(test_eq(sqrtf(v1[1]), v3[1])) + ASSERT(test_eq(sqrtf(v1[2]), v3[2])) + ASSERT(test_eq(sqrtf(v1[3]), v3[3])) + + ASSERT(test_eq(sqrtf(v2[0]), v4[0])) + ASSERT(test_eq(sqrtf(v2[1]), v4[1])) + ASSERT(test_eq(sqrtf(v2[2]), v4[2])) + ASSERT(test_eq(sqrtf(v2[3]), v4[3])) + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_make) { + float src[12] = { + 7.2f, 1.0f, 5.8f, 0.0f, + 2.5f, 6.1f, 9.9f, 1.0f, + 17.7f, 4.3f, 3.2f, 1.0f + }; + vec4 dest[3]; + + float *srcp = src; + unsigned int i, j; + + for (i = 0, j = 0; i < sizeof(src) / sizeof(float); i+=4,j++) { + GLM(vec4_make)(srcp + i, dest[j]); + ASSERT(test_eq(src[ i ], dest[j][0])); + ASSERT(test_eq(src[i+1], dest[j][1])); + ASSERT(test_eq(src[i+2], dest[j][2])); + ASSERT(test_eq(src[i+3], dest[j][3])); + } + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_reflect) { + vec4 dest; + + /* Original test: Reflecting off a horizontal surface */ + vec4 I1 = {1.0f, -1.0f, 0.0f, 0.0f}; /* Incoming vector */ + vec4 N1 = {0.0f, 1.0f, 0.0f, 0.0f}; /* Normal vector */ + GLM(vec4_reflect)(I1, N1, dest); + ASSERT(fabsf(dest[0] - 1.0f) < 0.00001f && + fabsf(dest[1] - 1.0f) < 0.00001f && + fabsf(dest[2] - 0.0f) < 0.00001f && + fabsf(dest[3] - 0.0f) < 0.00001f); /* Expect reflection */ + + /* Scenario 2: Reflecting off a vertical surface */ + vec4 I2 = {1.0f, 0.0f, 0.0f, 0.0f}; /* Incoming vector */ + vec4 N2 = {-1.0f, 0.0f, 0.0f, 0.0f}; /* Normal vector */ + GLM(vec4_reflect)(I2, N2, dest); + ASSERT(fabsf(dest[0] + 1.0f) < 0.00001f && + fabsf(dest[1]) < 0.00001f && + fabsf(dest[2]) < 0.00001f && + fabsf(dest[3] - 0.0f) < 0.00001f); /* Expect reflection to the left */ + + /* Scenario 3: Reflecting at an angle */ + vec4 I3 = {sqrtf(2)/2, -sqrtf(2)/2, 0.0f, 0.0f}; /* Incoming vector at 45 degrees */ + vec4 N3 = {0.0f, 1.0f, 0.0f, 0.0f}; /* Upwards normal vector */ + GLM(vec4_reflect)(I3, N3, dest); + ASSERT(fabsf(dest[0] - sqrtf(2)/2) < 0.00001f && + fabsf(dest[1] - sqrtf(2)/2) < 0.00001f && + fabsf(dest[2]) < 0.00001f && + fabsf(dest[3] - 0.0f) < 0.00001f); /* Expect reflection upwards */ + + TEST_SUCCESS +} + +TEST_IMPL(GLM_PREFIX, vec4_refract) { + vec4 v = {sqrtf(0.5f), -sqrtf(0.5f), 0.0f, 0.0f}; /* Incoming vector */ + vec4 N = {0.0f, 1.0f, 0.0f, 0.0f}; /* Surface normal */ + vec4 dest; + float eta; + float r; + + /* Water to Air (eta = 1.33/1.0) */ + eta = 1.33f / 1.0f; + r = GLM(vec4_refract)(v, N, eta, dest); + if (!(dest[0] == 0.0f && dest[1] == 0.0f && dest[2] == 0.0f && dest[3] == 0.0f)) { + ASSERT(dest[1] < -0.3f); + ASSERT(r == true); + } else { + ASSERT(dest[0] == 0.0f && dest[1] == 0.0f && dest[2] == 0.0f && dest[3] == 0.0f); + ASSERT(r == false); + } + + /* Air to Glass (eta = 1.0 / 1.5) */ + eta = 1.0f / 1.5f; + r = GLM(vec4_refract)(v, N, eta, dest); + ASSERT(r == true); + ASSERT(dest[1] < -sqrtf(0.5f)); // Expect bending towards the normal + + /* Glass to Water (eta = 1.5 / 1.33) */ + eta = 1.5f / 1.33f; + r = GLM(vec4_refract)(v, N, eta, dest); + ASSERT(r == true); + ASSERT(dest[1] < -0.6f); // Expect bending towards the normal, less bending than air to glass + + /* Diamond to Air (eta = 2.42 / 1.0) */ + eta = 2.42f / 1.0f; + r = GLM(vec4_refract)(v, N, eta, dest); + if (!(dest[0] == 0.0f && dest[1] == 0.0f && dest[2] == 0.0f && dest[3] == 0.0f)) { + /* High potential for total internal reflection, but if it occurs, expect significant bending */ + ASSERT(dest[1] < -sqrtf(0.5f)); + ASSERT(r == true); + } else { + ASSERT(dest[0] == 0.0f && dest[1] == 0.0f && dest[2] == 0.0f && dest[3] == 0.0f); + ASSERT(r == false); + } + + TEST_SUCCESS +} diff --git a/cglm/test/src/tests.c b/cglm/test/src/tests.c new file mode 100644 index 0000000..bc24668 --- /dev/null +++ b/cglm/test/src/tests.c @@ -0,0 +1,91 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +/* test inline */ +/*---------------------------------------------------------------------------*/ + + +#define GLM_PREFIX glm_ +#define GLM(X) (glm_ ## X) + +#include "test_vec2.h" +#include "test_vec3.h" +#include "test_vec4.h" +#include "test_ivec2.h" +#include "test_ivec3.h" +#include "test_ivec4.h" +#include "test_mat2.h" +#include "test_mat2x3.h" +#include "test_mat2x4.h" +#include "test_mat3.h" +#include "test_mat3x2.h" +#include "test_mat3x4.h" +#include "test_mat4.h" +#include "test_mat4x2.h" +#include "test_mat4x3.h" +#include "test_quat.h" +#include "test_project.h" +#include "test_plane.h" +#include "test_noise.h" +#include "test_affine.h" +#include "test_affine2d.h" +#include "test_affine_mat.h" +#include "test_aabb2d.h" +#include "test_ray.h" +#include "test_cam.h" +#include "test_cam_lh_no.h" +#include "test_cam_lh_zo.h" +#include "test_cam_rh_no.h" +#include "test_cam_rh_zo.h" +#include "test_euler_to_quat_rh.h" +#include "test_euler_to_quat_lh.h" + +#undef GLM +#undef GLM_PREFIX + +/* test pre-compiled */ +/*---------------------------------------------------------------------------*/ + +#define GLM_PREFIX glmc_ +#define GLM(X) (glmc_ ## X) + +#include "test_vec2.h" +#include "test_vec3.h" +#include "test_vec4.h" +#include "test_ivec2.h" +#include "test_ivec3.h" +#include "test_ivec4.h" +#include "test_mat2.h" +#include "test_mat2x3.h" +#include "test_mat2x4.h" +#include "test_mat3.h" +#include "test_mat3x2.h" +#include "test_mat3x4.h" +#include "test_mat4.h" +#include "test_mat4x2.h" +#include "test_mat4x3.h" +#include "test_quat.h" +#include "test_project.h" +#include "test_plane.h" +#include "test_noise.h" +#include "test_affine.h" +#include "test_affine2d.h" +#include "test_affine_mat.h" +#include "test_aabb2d.h" +#include "test_ray.h" +#include "test_cam.h" +#include "test_cam_lh_no.h" +#include "test_cam_lh_zo.h" +#include "test_cam_rh_no.h" +#include "test_cam_rh_zo.h" +#include "test_euler_to_quat_rh.h" +#include "test_euler_to_quat_lh.h" + +#undef GLM +#undef GLM_PREFIX + +/*---------------------------------------------------------------------------*/ diff --git a/cglm/test/tests.h b/cglm/test/tests.h new file mode 100644 index 0000000..b959ad6 --- /dev/null +++ b/cglm/test/tests.h @@ -0,0 +1,2437 @@ +/* + * Copyright (c), Recep Aslantas. + * + * MIT License (MIT), http://opensource.org/licenses/MIT + * Full license can be found in the LICENSE file + */ + +#ifndef tests_h +#define tests_h + +#include "include/common.h" + +/* + * To register a test: + * 1. use TEST_DECLARE() to forward declare test + * 2. use TEST_ENTRY() to add test to list + */ + +/* affine mat */ +TEST_DECLARE(glm_mul) +TEST_DECLARE(glm_mul) +TEST_DECLARE(glm_inv_tr) + +TEST_DECLARE(glmc_mul) +TEST_DECLARE(glmc_mul_rot) +TEST_DECLARE(glmc_inv_tr) + +/* affine */ +TEST_DECLARE(glm_translate) +TEST_DECLARE(glm_translate_to) +TEST_DECLARE(glm_translate_x) +TEST_DECLARE(glm_translate_y) +TEST_DECLARE(glm_translate_z) +TEST_DECLARE(glm_translate_make) +TEST_DECLARE(glm_scale_to) +TEST_DECLARE(glm_scale_make) +TEST_DECLARE(glm_scale) +TEST_DECLARE(glm_scale_uni) +TEST_DECLARE(glm_rotate_x) +TEST_DECLARE(glm_rotate_y) +TEST_DECLARE(glm_rotate_z) +TEST_DECLARE(glm_rotate_make) +TEST_DECLARE(glm_rotate) +TEST_DECLARE(glm_rotate_at) +TEST_DECLARE(glm_rotate_atm) +TEST_DECLARE(glm_decompose_scalev) +TEST_DECLARE(glm_uniscaled) +TEST_DECLARE(glm_decompose_rs) +TEST_DECLARE(glm_decompose) + +TEST_DECLARE(glmc_translate) +TEST_DECLARE(glmc_translate_to) +TEST_DECLARE(glmc_translate_x) +TEST_DECLARE(glmc_translate_y) +TEST_DECLARE(glmc_translate_z) +TEST_DECLARE(glmc_translate_make) +TEST_DECLARE(glmc_scale_to) +TEST_DECLARE(glmc_scale_make) +TEST_DECLARE(glmc_scale) +TEST_DECLARE(glmc_scale_uni) +TEST_DECLARE(glmc_rotate_x) +TEST_DECLARE(glmc_rotate_y) +TEST_DECLARE(glmc_rotate_z) +TEST_DECLARE(glmc_rotate_make) +TEST_DECLARE(glmc_rotate) +TEST_DECLARE(glmc_rotate_at) +TEST_DECLARE(glmc_rotate_atm) +TEST_DECLARE(glmc_decompose_scalev) +TEST_DECLARE(glmc_uniscaled) +TEST_DECLARE(glmc_decompose_rs) +TEST_DECLARE(glmc_decompose) + +/* affine 2d */ +TEST_DECLARE(glm_translate2d) +TEST_DECLARE(glm_translate2d_to) +TEST_DECLARE(glm_translate2d_x) +TEST_DECLARE(glm_translate2d_y) +TEST_DECLARE(glm_translate2d_make) +TEST_DECLARE(glm_scale2d_to) +TEST_DECLARE(glm_scale2d_make) +TEST_DECLARE(glm_scale2d) +TEST_DECLARE(glm_scale2d_uni) +TEST_DECLARE(glm_rotate2d_make) +TEST_DECLARE(glm_rotate2d) +TEST_DECLARE(glm_rotate2d_to) + +TEST_DECLARE(glmc_translate2d) +TEST_DECLARE(glmc_translate2d_to) +TEST_DECLARE(glmc_translate2d_x) +TEST_DECLARE(glmc_translate2d_y) +TEST_DECLARE(glmc_translate2d_make) +TEST_DECLARE(glmc_scale2d_to) +TEST_DECLARE(glmc_scale2d_make) +TEST_DECLARE(glmc_scale2d) +TEST_DECLARE(glmc_scale2d_uni) +TEST_DECLARE(glmc_rotate2d_make) +TEST_DECLARE(glmc_rotate2d) +TEST_DECLARE(glmc_rotate2d_to) + +/* aabb2d */ +TEST_DECLARE(glm_aabb2d_sizev) + +/* mat4 */ +TEST_DECLARE(glm_mat4_ucopy) +TEST_DECLARE(glm_mat4_copy) +TEST_DECLARE(glm_mat4_identity) +TEST_DECLARE(glm_mat4_identity_array) +TEST_DECLARE(glm_mat4_zero) +TEST_DECLARE(glm_mat4_pick3) +TEST_DECLARE(glm_mat4_pick3t) +TEST_DECLARE(glm_mat4_ins3) +TEST_DECLARE(glm_mat4_mul) +TEST_DECLARE(glm_mat4_mulN) +TEST_DECLARE(glm_mat4_mulv) +TEST_DECLARE(glm_mat4_mulv3) +TEST_DECLARE(glm_mat4_trace) +TEST_DECLARE(glm_mat4_trace3) +TEST_DECLARE(glm_mat4_quat) +TEST_DECLARE(glm_mat4_transpose_to) +TEST_DECLARE(glm_mat4_transpose) +TEST_DECLARE(glm_mat4_scale_p) +TEST_DECLARE(glm_mat4_scale) +TEST_DECLARE(glm_mat4_det) +TEST_DECLARE(glm_mat4_inv) +TEST_DECLARE(glm_mat4_inv_fast) +TEST_DECLARE(glm_mat4_inv_precise) +TEST_DECLARE(glm_mat4_swap_col) +TEST_DECLARE(glm_mat4_swap_row) +TEST_DECLARE(glm_mat4_rmc) +TEST_DECLARE(glm_mat4_make) + +TEST_DECLARE(glmc_mat4_ucopy) +TEST_DECLARE(glmc_mat4_copy) +TEST_DECLARE(glmc_mat4_identity) +TEST_DECLARE(glmc_mat4_identity_array) +TEST_DECLARE(glmc_mat4_zero) +TEST_DECLARE(glmc_mat4_pick3) +TEST_DECLARE(glmc_mat4_pick3t) +TEST_DECLARE(glmc_mat4_ins3) +TEST_DECLARE(glmc_mat4_mul) +TEST_DECLARE(glmc_mat4_mulN) +TEST_DECLARE(glmc_mat4_mulv) +TEST_DECLARE(glmc_mat4_mulv3) +TEST_DECLARE(glmc_mat4_trace) +TEST_DECLARE(glmc_mat4_trace3) +TEST_DECLARE(glmc_mat4_quat) +TEST_DECLARE(glmc_mat4_transpose_to) +TEST_DECLARE(glmc_mat4_transpose) +TEST_DECLARE(glmc_mat4_scale_p) +TEST_DECLARE(glmc_mat4_scale) +TEST_DECLARE(glmc_mat4_det) +TEST_DECLARE(glmc_mat4_inv) +TEST_DECLARE(glmc_mat4_inv_fast) +TEST_DECLARE(glmc_mat4_swap_col) +TEST_DECLARE(glmc_mat4_swap_row) +TEST_DECLARE(glmc_mat4_rmc) +TEST_DECLARE(glmc_mat4_make) + +TEST_DECLARE(MACRO_GLM_MAT4X2_ZERO_INIT) +TEST_DECLARE(MACRO_GLM_MAT4X2_ZERO) +TEST_DECLARE(glm_mat4x2_copy) +TEST_DECLARE(glm_mat4x2_zero) +TEST_DECLARE(glm_mat4x2_make) +TEST_DECLARE(glm_mat4x2_mul) +TEST_DECLARE(glm_mat4x2_mulv) +TEST_DECLARE(glm_mat4x2_transpose) +TEST_DECLARE(glm_mat4x2_scale) + +TEST_DECLARE(glmc_mat4x2_copy) +TEST_DECLARE(glmc_mat4x2_zero) +TEST_DECLARE(glmc_mat4x2_make) +TEST_DECLARE(glmc_mat4x2_mul) +TEST_DECLARE(glmc_mat4x2_mulv) +TEST_DECLARE(glmc_mat4x2_transpose) +TEST_DECLARE(glmc_mat4x2_scale) + +TEST_DECLARE(MACRO_GLM_MAT4X3_ZERO_INIT) +TEST_DECLARE(MACRO_GLM_MAT4X3_ZERO) +TEST_DECLARE(glm_mat4x3_copy) +TEST_DECLARE(glm_mat4x3_zero) +TEST_DECLARE(glm_mat4x3_make) +TEST_DECLARE(glm_mat4x3_mul) +TEST_DECLARE(glm_mat4x3_mulv) +TEST_DECLARE(glm_mat4x3_transpose) +TEST_DECLARE(glm_mat4x3_scale) + +TEST_DECLARE(glmc_mat4x3_copy) +TEST_DECLARE(glmc_mat4x3_zero) +TEST_DECLARE(glmc_mat4x3_make) +TEST_DECLARE(glmc_mat4x3_mul) +TEST_DECLARE(glmc_mat4x3_mulv) +TEST_DECLARE(glmc_mat4x3_transpose) +TEST_DECLARE(glmc_mat4x3_scale) + +/* mat3 */ +TEST_DECLARE(glm_mat3_copy) +TEST_DECLARE(glm_mat3_identity) +TEST_DECLARE(glm_mat3_identity_array) +TEST_DECLARE(glm_mat3_zero) +TEST_DECLARE(glm_mat3_mul) +TEST_DECLARE(glm_mat3_mulv) +TEST_DECLARE(glm_mat3_trace) +TEST_DECLARE(glm_mat3_quat) +TEST_DECLARE(glm_mat3_transpose_to) +TEST_DECLARE(glm_mat3_transpose) +TEST_DECLARE(glm_mat3_scale) +TEST_DECLARE(glm_mat3_det) +TEST_DECLARE(glm_mat3_inv) +TEST_DECLARE(glm_mat3_swap_col) +TEST_DECLARE(glm_mat3_swap_row) +TEST_DECLARE(glm_mat3_rmc) +TEST_DECLARE(glm_mat3_make) + +TEST_DECLARE(glmc_mat3_copy) +TEST_DECLARE(glmc_mat3_identity) +TEST_DECLARE(glmc_mat3_identity_array) +TEST_DECLARE(glmc_mat3_zero) +TEST_DECLARE(glmc_mat3_mul) +TEST_DECLARE(glmc_mat3_mulv) +TEST_DECLARE(glmc_mat3_trace) +TEST_DECLARE(glmc_mat3_quat) +TEST_DECLARE(glmc_mat3_transpose_to) +TEST_DECLARE(glmc_mat3_transpose) +TEST_DECLARE(glmc_mat3_scale) +TEST_DECLARE(glmc_mat3_det) +TEST_DECLARE(glmc_mat3_inv) +TEST_DECLARE(glmc_mat3_swap_col) +TEST_DECLARE(glmc_mat3_swap_row) +TEST_DECLARE(glmc_mat3_rmc) +TEST_DECLARE(glmc_mat3_make) + +TEST_DECLARE(MACRO_GLM_MAT3X2_ZERO_INIT) +TEST_DECLARE(MACRO_GLM_MAT3X2_ZERO) +TEST_DECLARE(glm_mat3x2_copy) +TEST_DECLARE(glm_mat3x2_zero) +TEST_DECLARE(glm_mat3x2_make) +TEST_DECLARE(glm_mat3x2_mul) +TEST_DECLARE(glm_mat3x2_mulv) +TEST_DECLARE(glm_mat3x2_transpose) +TEST_DECLARE(glm_mat3x2_scale) + +TEST_DECLARE(glmc_mat3x2_copy) +TEST_DECLARE(glmc_mat3x2_zero) +TEST_DECLARE(glmc_mat3x2_make) +TEST_DECLARE(glmc_mat3x2_mul) +TEST_DECLARE(glmc_mat3x2_mulv) +TEST_DECLARE(glmc_mat3x2_transpose) +TEST_DECLARE(glmc_mat3x2_scale) + +TEST_DECLARE(MACRO_GLM_MAT3X4_ZERO_INIT) +TEST_DECLARE(MACRO_GLM_MAT3X4_ZERO) +TEST_DECLARE(glm_mat3x4_make) + +TEST_DECLARE(glmc_mat3x4_make) + +TEST_DECLARE(MACRO_GLM_MAT2_IDENTITY_INIT) +TEST_DECLARE(MACRO_GLM_MAT2_ZERO_INIT) +TEST_DECLARE(MACRO_GLM_MAT2_IDENTITY) +TEST_DECLARE(MACRO_GLM_MAT2_ZERO) +TEST_DECLARE(glm_mat2_copy) +TEST_DECLARE(glm_mat2_identity) +TEST_DECLARE(glm_mat2_identity_array) +TEST_DECLARE(glm_mat2_zero) +TEST_DECLARE(glm_mat2_mul) +TEST_DECLARE(glm_mat2_transpose_to) +TEST_DECLARE(glm_mat2_transpose) +TEST_DECLARE(glm_mat2_mulv) +TEST_DECLARE(glm_mat2_trace) +TEST_DECLARE(glm_mat2_scale) +TEST_DECLARE(glm_mat2_det) +TEST_DECLARE(glm_mat2_inv) +TEST_DECLARE(glm_mat2_swap_col) +TEST_DECLARE(glm_mat2_swap_row) +TEST_DECLARE(glm_mat2_rmc) +TEST_DECLARE(glm_mat2_make) + +TEST_DECLARE(glmc_mat2_copy) +TEST_DECLARE(glmc_mat2_identity) +TEST_DECLARE(glmc_mat2_identity_array) +TEST_DECLARE(glmc_mat2_zero) +TEST_DECLARE(glmc_mat2_mul) +TEST_DECLARE(glmc_mat2_transpose_to) +TEST_DECLARE(glmc_mat2_transpose) +TEST_DECLARE(glmc_mat2_mulv) +TEST_DECLARE(glmc_mat2_trace) +TEST_DECLARE(glmc_mat2_scale) +TEST_DECLARE(glmc_mat2_det) +TEST_DECLARE(glmc_mat2_inv) +TEST_DECLARE(glmc_mat2_swap_col) +TEST_DECLARE(glmc_mat2_swap_row) +TEST_DECLARE(glmc_mat2_rmc) +TEST_DECLARE(glmc_mat2_make) + +TEST_DECLARE(MACRO_GLM_MAT2X3_ZERO_INIT) +TEST_DECLARE(MACRO_GLM_MAT2X3_ZERO) +TEST_DECLARE(glm_mat2x3_copy) +TEST_DECLARE(glm_mat2x3_zero) +TEST_DECLARE(glm_mat2x3_make) +TEST_DECLARE(glm_mat2x3_mul) +TEST_DECLARE(glm_mat2x3_mulv) +TEST_DECLARE(glm_mat2x3_transpose) +TEST_DECLARE(glm_mat2x3_scale) + +TEST_DECLARE(glmc_mat2x3_copy) +TEST_DECLARE(glmc_mat2x3_zero) +TEST_DECLARE(glmc_mat2x3_make) +TEST_DECLARE(glmc_mat2x3_mul) +TEST_DECLARE(glmc_mat2x3_mulv) +TEST_DECLARE(glmc_mat2x3_transpose) +TEST_DECLARE(glmc_mat2x3_scale) + +TEST_DECLARE(MACRO_GLM_MAT2X4_ZERO_INIT) +TEST_DECLARE(MACRO_GLM_MAT2X4_ZERO) +TEST_DECLARE(glm_mat2x4_copy) +TEST_DECLARE(glm_mat2x4_zero) +TEST_DECLARE(glm_mat2x4_make) +TEST_DECLARE(glm_mat2x4_mul) +TEST_DECLARE(glm_mat2x4_mulv) +TEST_DECLARE(glm_mat2x4_transpose) +TEST_DECLARE(glm_mat2x4_scale) + +TEST_DECLARE(glmc_mat2x4_copy) +TEST_DECLARE(glmc_mat2x4_zero) +TEST_DECLARE(glmc_mat2x4_make) +TEST_DECLARE(glmc_mat2x4_mul) +TEST_DECLARE(glmc_mat2x4_mulv) +TEST_DECLARE(glmc_mat2x4_transpose) +TEST_DECLARE(glmc_mat2x4_scale) + +/* camera (incl [LR]H cross [NZ]O) */ +TEST_DECLARE(glm_perspective_lh_zo) +TEST_DECLARE(glm_perspective_rh_zo) +TEST_DECLARE(glm_perspective_lh_no) +TEST_DECLARE(glm_perspective_rh_no) +TEST_DECLARE(glm_camera_lookat) +TEST_DECLARE(glm_camera_decomp) + +TEST_DECLARE(glmc_perspective_lh_zo) +TEST_DECLARE(glmc_perspective_rh_zo) +TEST_DECLARE(glmc_perspective_lh_no) +TEST_DECLARE(glmc_perspective_rh_no) +TEST_DECLARE(glmc_camera_lookat) +TEST_DECLARE(glmc_camera_decomp) + +TEST_DECLARE(glm_frustum) + +TEST_DECLARE(glmc_frustum) + +/* project */ +TEST_DECLARE(glm_unprojecti) +TEST_DECLARE(glm_unproject) +TEST_DECLARE(glm_project) + +TEST_DECLARE(glmc_unprojecti) +TEST_DECLARE(glmc_unproject) +TEST_DECLARE(glmc_project) + +/* plane */ +TEST_DECLARE(glm_plane_normalize) +TEST_DECLARE(glmc_plane_normalize) + +/* noise */ +TEST_DECLARE(glm_perlin_vec4) +TEST_DECLARE(glmc_perlin_vec4) +TEST_DECLARE(glm_perlin_vec3) +TEST_DECLARE(glmc_perlin_vec3) +TEST_DECLARE(glm_perlin_vec2) +TEST_DECLARE(glmc_perlin_vec2) + +/* utils */ +TEST_DECLARE(clamp) + +/* euler */ +TEST_DECLARE(glm_euler_xyz_quat_rh) +TEST_DECLARE(glm_euler_xzy_quat_rh) +TEST_DECLARE(glm_euler_yxz_quat_rh) +TEST_DECLARE(glm_euler_yzx_quat_rh) +TEST_DECLARE(glm_euler_zxy_quat_rh) +TEST_DECLARE(glm_euler_zyx_quat_rh) + +TEST_DECLARE(glm_euler_xyz_quat_lh) +TEST_DECLARE(glm_euler_xzy_quat_lh) +TEST_DECLARE(glm_euler_yxz_quat_lh) +TEST_DECLARE(glm_euler_yzx_quat_lh) +TEST_DECLARE(glm_euler_zxy_quat_lh) +TEST_DECLARE(glm_euler_zyx_quat_lh) + +TEST_DECLARE(glmc_euler_xyz_quat_rh) +TEST_DECLARE(glmc_euler_xzy_quat_rh) +TEST_DECLARE(glmc_euler_yxz_quat_rh) +TEST_DECLARE(glmc_euler_yzx_quat_rh) +TEST_DECLARE(glmc_euler_zxy_quat_rh) +TEST_DECLARE(glmc_euler_zyx_quat_rh) + +TEST_DECLARE(glmc_euler_xyz_quat_lh) +TEST_DECLARE(glmc_euler_xzy_quat_lh) +TEST_DECLARE(glmc_euler_yxz_quat_lh) +TEST_DECLARE(glmc_euler_yzx_quat_lh) +TEST_DECLARE(glmc_euler_zxy_quat_lh) +TEST_DECLARE(glmc_euler_zyx_quat_lh) + +TEST_DECLARE(euler) + +/* ray */ +TEST_DECLARE(glm_ray_triangle) +TEST_DECLARE(glm_ray_sphere) +TEST_DECLARE(glm_ray_at) + +TEST_DECLARE(glmc_ray_triangle) +TEST_DECLARE(glmc_ray_sphere) +TEST_DECLARE(glmc_ray_at) + +/* quat */ +TEST_DECLARE(MACRO_GLM_QUAT_IDENTITY_INIT) +TEST_DECLARE(MACRO_GLM_QUAT_IDENTITY) + +TEST_DECLARE(glm_quat_identity) +TEST_DECLARE(glm_quat_identity_array) +TEST_DECLARE(glm_quat_init) +TEST_DECLARE(glm_quatv) +TEST_DECLARE(glm_quat) +TEST_DECLARE(glm_quat_copy) +TEST_DECLARE(glm_quat_norm) +TEST_DECLARE(glm_quat_normalize_to) +TEST_DECLARE(glm_quat_normalize) +TEST_DECLARE(glm_quat_dot) +TEST_DECLARE(glm_quat_conjugate) +TEST_DECLARE(glm_quat_inv) +TEST_DECLARE(glm_quat_add) +TEST_DECLARE(glm_quat_sub) +TEST_DECLARE(glm_quat_real) +TEST_DECLARE(glm_quat_imag) +TEST_DECLARE(glm_quat_imagn) +TEST_DECLARE(glm_quat_imaglen) +TEST_DECLARE(glm_quat_angle) +TEST_DECLARE(glm_quat_axis) +TEST_DECLARE(glm_quat_mul) +TEST_DECLARE(glm_quat_mat4) +TEST_DECLARE(glm_quat_mat4t) +TEST_DECLARE(glm_quat_mat3) +TEST_DECLARE(glm_quat_mat3t) +TEST_DECLARE(glm_quat_lerp) +TEST_DECLARE(glm_quat_lerpc) +TEST_DECLARE(glm_quat_nlerp) +TEST_DECLARE(glm_quat_slerp) +TEST_DECLARE(glm_quat_look) +TEST_DECLARE(glm_quat_for) +TEST_DECLARE(glm_quat_forp) +TEST_DECLARE(glm_quat_rotatev) +TEST_DECLARE(glm_quat_rotate) +TEST_DECLARE(glm_quat_rotate_at) +TEST_DECLARE(glm_quat_rotate_atm) +TEST_DECLARE(glm_quat_from_vecs) +TEST_DECLARE(glm_quat_make) + +TEST_DECLARE(glmc_quat_identity) +TEST_DECLARE(glmc_quat_identity_array) +TEST_DECLARE(glmc_quat_init) +TEST_DECLARE(glmc_quatv) +TEST_DECLARE(glmc_quat) +TEST_DECLARE(glmc_quat_copy) +TEST_DECLARE(glmc_quat_norm) +TEST_DECLARE(glmc_quat_normalize_to) +TEST_DECLARE(glmc_quat_normalize) +TEST_DECLARE(glmc_quat_dot) +TEST_DECLARE(glmc_quat_conjugate) +TEST_DECLARE(glmc_quat_inv) +TEST_DECLARE(glmc_quat_add) +TEST_DECLARE(glmc_quat_sub) +TEST_DECLARE(glmc_quat_real) +TEST_DECLARE(glmc_quat_imag) +TEST_DECLARE(glmc_quat_imagn) +TEST_DECLARE(glmc_quat_imaglen) +TEST_DECLARE(glmc_quat_angle) +TEST_DECLARE(glmc_quat_axis) +TEST_DECLARE(glmc_quat_mul) +TEST_DECLARE(glmc_quat_mat4) +TEST_DECLARE(glmc_quat_mat4t) +TEST_DECLARE(glmc_quat_mat3) +TEST_DECLARE(glmc_quat_mat3t) +TEST_DECLARE(glmc_quat_lerp) +TEST_DECLARE(glmc_quat_lerpc) +TEST_DECLARE(glmc_quat_nlerp) +TEST_DECLARE(glmc_quat_slerp) +TEST_DECLARE(glmc_quat_look) +TEST_DECLARE(glmc_quat_for) +TEST_DECLARE(glmc_quat_forp) +TEST_DECLARE(glmc_quat_rotatev) +TEST_DECLARE(glmc_quat_rotate) +TEST_DECLARE(glmc_quat_rotate_at) +TEST_DECLARE(glmc_quat_rotate_atm) +TEST_DECLARE(glmc_quat_from_vecs) +TEST_DECLARE(glmc_quat_make) + +/* bezier */ +TEST_DECLARE(bezier) + +/* vec2 */ +TEST_DECLARE(MACRO_GLM_VEC2_ONE_INIT) +TEST_DECLARE(MACRO_GLM_VEC2_ZERO_INIT) +TEST_DECLARE(MACRO_GLM_VEC2_ONE) +TEST_DECLARE(MACRO_GLM_VEC2_ZERO) + +TEST_DECLARE(glm_vec2) +TEST_DECLARE(glm_vec2_copy) +TEST_DECLARE(glm_vec2_zero) +TEST_DECLARE(glm_vec2_one) +TEST_DECLARE(glm_vec2_dot) +TEST_DECLARE(glm_vec2_cross) +TEST_DECLARE(glm_vec2_norm2) +TEST_DECLARE(glm_vec2_norm) +TEST_DECLARE(glm_vec2_add) +TEST_DECLARE(glm_vec2_adds) +TEST_DECLARE(glm_vec2_sub) +TEST_DECLARE(glm_vec2_subs) +TEST_DECLARE(glm_vec2_mul) +TEST_DECLARE(glm_vec2_scale) +TEST_DECLARE(glm_vec2_scale_as) +TEST_DECLARE(glm_vec2_div) +TEST_DECLARE(glm_vec2_divs) +TEST_DECLARE(glm_vec2_addadd) +TEST_DECLARE(glm_vec2_subadd) +TEST_DECLARE(glm_vec2_muladd) +TEST_DECLARE(glm_vec2_muladds) +TEST_DECLARE(glm_vec2_maxadd) +TEST_DECLARE(glm_vec2_minadd) +TEST_DECLARE(glm_vec2_subsub) +TEST_DECLARE(glm_vec2_addsub) +TEST_DECLARE(glm_vec2_mulsub) +TEST_DECLARE(glm_vec2_mulsubs) +TEST_DECLARE(glm_vec2_maxsub) +TEST_DECLARE(glm_vec2_minsub) +TEST_DECLARE(glm_vec2_negate_to) +TEST_DECLARE(glm_vec2_negate) +TEST_DECLARE(glm_vec2_normalize) +TEST_DECLARE(glm_vec2_normalize_to) +TEST_DECLARE(glm_vec2_rotate) +TEST_DECLARE(glm_vec2_center) +TEST_DECLARE(glm_vec2_distance2) +TEST_DECLARE(glm_vec2_distance) +TEST_DECLARE(glm_vec2_maxv) +TEST_DECLARE(glm_vec2_minv) +TEST_DECLARE(glm_vec2_clamp) +TEST_DECLARE(glm_vec2_abs) +TEST_DECLARE(glm_vec2_fract) +TEST_DECLARE(glm_vec2_floor) +TEST_DECLARE(glm_vec2_mods) +TEST_DECLARE(glm_vec2_steps) +TEST_DECLARE(glm_vec2_stepr) +TEST_DECLARE(glm_vec2_lerp) +TEST_DECLARE(glm_vec2_complex_mul) +TEST_DECLARE(glm_vec2_complex_div) +TEST_DECLARE(glm_vec2_make) +TEST_DECLARE(glm_vec2_reflect) +TEST_DECLARE(glm_vec2_refract) + +TEST_DECLARE(glmc_vec2) +TEST_DECLARE(glmc_vec2_copy) +TEST_DECLARE(glmc_vec2_zero) +TEST_DECLARE(glmc_vec2_one) +TEST_DECLARE(glmc_vec2_dot) +TEST_DECLARE(glmc_vec2_cross) +TEST_DECLARE(glmc_vec2_norm2) +TEST_DECLARE(glmc_vec2_norm) +TEST_DECLARE(glmc_vec2_add) +TEST_DECLARE(glmc_vec2_adds) +TEST_DECLARE(glmc_vec2_sub) +TEST_DECLARE(glmc_vec2_subs) +TEST_DECLARE(glmc_vec2_mul) +TEST_DECLARE(glmc_vec2_scale) +TEST_DECLARE(glmc_vec2_scale_as) +TEST_DECLARE(glmc_vec2_div) +TEST_DECLARE(glmc_vec2_divs) +TEST_DECLARE(glmc_vec2_addadd) +TEST_DECLARE(glmc_vec2_subadd) +TEST_DECLARE(glmc_vec2_muladd) +TEST_DECLARE(glmc_vec2_muladds) +TEST_DECLARE(glmc_vec2_maxadd) +TEST_DECLARE(glmc_vec2_minadd) +TEST_DECLARE(glmc_vec2_subsub) +TEST_DECLARE(glmc_vec2_addsub) +TEST_DECLARE(glmc_vec2_mulsub) +TEST_DECLARE(glmc_vec2_mulsubs) +TEST_DECLARE(glmc_vec2_maxsub) +TEST_DECLARE(glmc_vec2_minsub) +TEST_DECLARE(glmc_vec2_negate_to) +TEST_DECLARE(glmc_vec2_negate) +TEST_DECLARE(glmc_vec2_normalize) +TEST_DECLARE(glmc_vec2_normalize_to) +TEST_DECLARE(glmc_vec2_rotate) +TEST_DECLARE(glmc_vec2_center) +TEST_DECLARE(glmc_vec2_distance2) +TEST_DECLARE(glmc_vec2_distance) +TEST_DECLARE(glmc_vec2_maxv) +TEST_DECLARE(glmc_vec2_minv) +TEST_DECLARE(glmc_vec2_clamp) +TEST_DECLARE(glmc_vec2_abs) +TEST_DECLARE(glmc_vec2_fract) +TEST_DECLARE(glmc_vec2_floor) +TEST_DECLARE(glmc_vec2_mods) +TEST_DECLARE(glmc_vec2_steps) +TEST_DECLARE(glmc_vec2_stepr) +TEST_DECLARE(glmc_vec2_lerp) +TEST_DECLARE(glmc_vec2_complex_mul) +TEST_DECLARE(glmc_vec2_complex_div) +TEST_DECLARE(glmc_vec2_make) +TEST_DECLARE(glmc_vec2_reflect) +TEST_DECLARE(glmc_vec2_refract) + +/* vec3 */ +TEST_DECLARE(MACRO_GLM_VEC3_ONE_INIT) +TEST_DECLARE(MACRO_GLM_VEC3_ZERO_INIT) +TEST_DECLARE(MACRO_GLM_VEC3_ONE) +TEST_DECLARE(MACRO_GLM_VEC3_ZERO) +TEST_DECLARE(MACRO_GLM_YUP) +TEST_DECLARE(MACRO_GLM_ZUP) +TEST_DECLARE(MACRO_GLM_XUP) +TEST_DECLARE(MACRO_GLM_FORWARD_RH) +TEST_DECLARE(MACRO_GLM_SHUFFLE3) +TEST_DECLARE(MACRO_GLM_XXX) +TEST_DECLARE(MACRO_GLM_YYY) +TEST_DECLARE(MACRO_GLM_ZZZ) +TEST_DECLARE(MACRO_GLM_ZYX) + +TEST_DECLARE(MACRO_glm_vec3_dup) +TEST_DECLARE(MACRO_glm_vec3_flipsign) +TEST_DECLARE(MACRO_glm_vec3_flipsign_to) +TEST_DECLARE(MACRO_glm_vec3_inv) +TEST_DECLARE(MACRO_glm_vec3_inv_to) +TEST_DECLARE(MACRO_glm_vec3_mulv) + +TEST_DECLARE(glm_vec3) +TEST_DECLARE(glm_vec3_copy) +TEST_DECLARE(glm_vec3_zero) +TEST_DECLARE(glm_vec3_one) +TEST_DECLARE(glm_vec3_dot) +TEST_DECLARE(glm_dot) +TEST_DECLARE(glm_vec3_norm2) +TEST_DECLARE(glm_vec3_norm) +TEST_DECLARE(glm_vec3_norm_one) +TEST_DECLARE(glm_vec3_norm_inf) +TEST_DECLARE(glm_vec3_add) +TEST_DECLARE(glm_vec3_adds) +TEST_DECLARE(glm_vec3_sub) +TEST_DECLARE(glm_vec3_subs) +TEST_DECLARE(glm_vec3_mul) +TEST_DECLARE(glm_vec3_scale) +TEST_DECLARE(glm_vec3_scale_as) +TEST_DECLARE(glm_vec3_div) +TEST_DECLARE(glm_vec3_divs) +TEST_DECLARE(glm_vec3_addadd) +TEST_DECLARE(glm_vec3_subadd) +TEST_DECLARE(glm_vec3_muladd) +TEST_DECLARE(glm_vec3_muladds) +TEST_DECLARE(glm_vec3_maxadd) +TEST_DECLARE(glm_vec3_minadd) +TEST_DECLARE(glm_vec3_subsub) +TEST_DECLARE(glm_vec3_addsub) +TEST_DECLARE(glm_vec3_mulsub) +TEST_DECLARE(glm_vec3_mulsubs) +TEST_DECLARE(glm_vec3_maxsub) +TEST_DECLARE(glm_vec3_minsub) +TEST_DECLARE(glm_vec3_negate_to) +TEST_DECLARE(glm_vec3_negate) +TEST_DECLARE(glm_vec3_normalize) +TEST_DECLARE(glm_vec3_normalize_to) +TEST_DECLARE(glm_normalize) +TEST_DECLARE(glm_normalize_to) +TEST_DECLARE(glm_vec3_cross) +TEST_DECLARE(glm_vec3_crossn) +TEST_DECLARE(glm_cross) +TEST_DECLARE(glm_vec3_angle) +TEST_DECLARE(glm_vec3_rotate) +TEST_DECLARE(glm_vec3_rotate_m4) +TEST_DECLARE(glm_vec3_rotate_m3) +TEST_DECLARE(glm_vec3_proj) +TEST_DECLARE(glm_vec3_center) +TEST_DECLARE(glm_vec3_distance2) +TEST_DECLARE(glm_vec3_distance) +TEST_DECLARE(glm_vec3_maxv) +TEST_DECLARE(glm_vec3_minv) +TEST_DECLARE(glm_vec3_ortho) +TEST_DECLARE(glm_vec3_clamp) +TEST_DECLARE(glm_vec3_mix) +TEST_DECLARE(glm_vec3_mixc) +TEST_DECLARE(glm_vec3_step) +TEST_DECLARE(glm_vec3_smoothstep_uni) +TEST_DECLARE(glm_vec3_smoothstep) +TEST_DECLARE(glm_vec3_smoothinterp) +TEST_DECLARE(glm_vec3_smoothinterpc) +TEST_DECLARE(glm_vec3_swizzle) +TEST_DECLARE(glm_vec3_broadcast) +TEST_DECLARE(glm_vec3_fill) +TEST_DECLARE(glm_vec3_eq) +TEST_DECLARE(glm_vec3_eq_eps) +TEST_DECLARE(glm_vec3_eq_all) +TEST_DECLARE(glm_vec3_eqv) +TEST_DECLARE(glm_vec3_eqv_eps) +TEST_DECLARE(glm_vec3_max) +TEST_DECLARE(glm_vec3_min) +TEST_DECLARE(glm_vec3_isnan) +TEST_DECLARE(glm_vec3_isinf) +TEST_DECLARE(glm_vec3_isvalid) +TEST_DECLARE(glm_vec3_sign) +TEST_DECLARE(glm_vec3_abs) +TEST_DECLARE(glm_vec3_fract) +TEST_DECLARE(glm_vec3_floor) +TEST_DECLARE(glm_vec3_mods) +TEST_DECLARE(glm_vec3_steps) +TEST_DECLARE(glm_vec3_stepr) +TEST_DECLARE(glm_vec3_hadd) +TEST_DECLARE(glm_vec3_sqrt) +TEST_DECLARE(glm_vec3_make) +TEST_DECLARE(glm_vec3_faceforward) +TEST_DECLARE(glm_vec3_reflect) +TEST_DECLARE(glm_vec3_refract) + +TEST_DECLARE(glmc_vec3) +TEST_DECLARE(glmc_vec3_copy) +TEST_DECLARE(glmc_vec3_zero) +TEST_DECLARE(glmc_vec3_one) +TEST_DECLARE(glmc_vec3_dot) +TEST_DECLARE(glmc_vec3_norm2) +TEST_DECLARE(glmc_vec3_norm) +TEST_DECLARE(glmc_vec3_norm_one) +TEST_DECLARE(glmc_vec3_norm_inf) +TEST_DECLARE(glmc_vec3_add) +TEST_DECLARE(glmc_vec3_adds) +TEST_DECLARE(glmc_vec3_sub) +TEST_DECLARE(glmc_vec3_subs) +TEST_DECLARE(glmc_vec3_mul) +TEST_DECLARE(glmc_vec3_scale) +TEST_DECLARE(glmc_vec3_scale_as) +TEST_DECLARE(glmc_vec3_div) +TEST_DECLARE(glmc_vec3_divs) +TEST_DECLARE(glmc_vec3_addadd) +TEST_DECLARE(glmc_vec3_subadd) +TEST_DECLARE(glmc_vec3_muladd) +TEST_DECLARE(glmc_vec3_muladds) +TEST_DECLARE(glmc_vec3_maxadd) +TEST_DECLARE(glmc_vec3_minadd) +TEST_DECLARE(glmc_vec3_subsub) +TEST_DECLARE(glmc_vec3_addsub) +TEST_DECLARE(glmc_vec3_mulsub) +TEST_DECLARE(glmc_vec3_mulsubs) +TEST_DECLARE(glmc_vec3_maxsub) +TEST_DECLARE(glmc_vec3_minsub) +TEST_DECLARE(glmc_vec3_negate_to) +TEST_DECLARE(glmc_vec3_negate) +TEST_DECLARE(glmc_vec3_normalize) +TEST_DECLARE(glmc_vec3_normalize_to) +TEST_DECLARE(glmc_vec3_cross) +TEST_DECLARE(glmc_vec3_crossn) +TEST_DECLARE(glmc_vec3_angle) +TEST_DECLARE(glmc_vec3_rotate) +TEST_DECLARE(glmc_vec3_rotate_m4) +TEST_DECLARE(glmc_vec3_rotate_m3) +TEST_DECLARE(glmc_vec3_proj) +TEST_DECLARE(glmc_vec3_center) +TEST_DECLARE(glmc_vec3_distance2) +TEST_DECLARE(glmc_vec3_distance) +TEST_DECLARE(glmc_vec3_maxv) +TEST_DECLARE(glmc_vec3_minv) +TEST_DECLARE(glmc_vec3_ortho) +TEST_DECLARE(glmc_vec3_clamp) +TEST_DECLARE(glmc_vec3_mix) +TEST_DECLARE(glmc_vec3_mixc) +TEST_DECLARE(glmc_vec3_step) +TEST_DECLARE(glmc_vec3_smoothstep_uni) +TEST_DECLARE(glmc_vec3_smoothstep) +TEST_DECLARE(glmc_vec3_smoothinterp) +TEST_DECLARE(glmc_vec3_smoothinterpc) +TEST_DECLARE(glmc_vec3_swizzle) +TEST_DECLARE(glmc_vec3_broadcast) +TEST_DECLARE(glmc_vec3_fill) +TEST_DECLARE(glmc_vec3_eq) +TEST_DECLARE(glmc_vec3_eq_eps) +TEST_DECLARE(glmc_vec3_eq_all) +TEST_DECLARE(glmc_vec3_eqv) +TEST_DECLARE(glmc_vec3_eqv_eps) +TEST_DECLARE(glmc_vec3_max) +TEST_DECLARE(glmc_vec3_min) +TEST_DECLARE(glmc_vec3_isnan) +TEST_DECLARE(glmc_vec3_isinf) +TEST_DECLARE(glmc_vec3_isvalid) +TEST_DECLARE(glmc_vec3_sign) +TEST_DECLARE(glmc_vec3_abs) +TEST_DECLARE(glmc_vec3_fract) +TEST_DECLARE(glmc_vec3_floor) +TEST_DECLARE(glmc_vec3_mods) +TEST_DECLARE(glmc_vec3_steps) +TEST_DECLARE(glmc_vec3_stepr) +TEST_DECLARE(glmc_vec3_hadd) +TEST_DECLARE(glmc_vec3_sqrt) +TEST_DECLARE(glmc_vec3_make) +TEST_DECLARE(glmc_vec3_faceforward) +TEST_DECLARE(glmc_vec3_reflect) +TEST_DECLARE(glmc_vec3_refract) + +/* vec4 */ +TEST_DECLARE(MACRO_GLM_VEC4_ONE_INIT) +TEST_DECLARE(MACRO_GLM_VEC4_ZERO_INIT) +TEST_DECLARE(MACRO_GLM_VEC4_ONE) +TEST_DECLARE(MACRO_GLM_VEC4_ZERO) +TEST_DECLARE(MACRO_GLM_XXXX) +TEST_DECLARE(MACRO_GLM_YYYY) +TEST_DECLARE(MACRO_GLM_ZZZZ) +TEST_DECLARE(MACRO_GLM_WZYX) +TEST_DECLARE(MACRO_glm_vec4_dup) +TEST_DECLARE(MACRO_glm_vec4_flipsign) +TEST_DECLARE(MACRO_glm_vec4_flipsign_to) +TEST_DECLARE(MACRO_glm_vec4_inv) +TEST_DECLARE(MACRO_glm_vec4_inv_to) +TEST_DECLARE(MACRO_glm_vec4_mulv) + +TEST_DECLARE(glm_vec4) +TEST_DECLARE(glm_vec4_copy3) +TEST_DECLARE(glm_vec4_copy) +TEST_DECLARE(glm_vec4_ucopy) +TEST_DECLARE(glm_vec4_zero) +TEST_DECLARE(glm_vec4_one) +TEST_DECLARE(glm_vec4_dot) +TEST_DECLARE(glm_vec4_norm2) +TEST_DECLARE(glm_vec4_norm) +TEST_DECLARE(glm_vec4_norm_one) +TEST_DECLARE(glm_vec4_norm_inf) +TEST_DECLARE(glm_vec4_add) +TEST_DECLARE(glm_vec4_adds) +TEST_DECLARE(glm_vec4_sub) +TEST_DECLARE(glm_vec4_subs) +TEST_DECLARE(glm_vec4_mul) +TEST_DECLARE(glm_vec4_scale) +TEST_DECLARE(glm_vec4_scale_as) +TEST_DECLARE(glm_vec4_div) +TEST_DECLARE(glm_vec4_divs) +TEST_DECLARE(glm_vec4_addadd) +TEST_DECLARE(glm_vec4_subadd) +TEST_DECLARE(glm_vec4_muladd) +TEST_DECLARE(glm_vec4_muladds) +TEST_DECLARE(glm_vec4_maxadd) +TEST_DECLARE(glm_vec4_minadd) +TEST_DECLARE(glm_vec4_subsub) +TEST_DECLARE(glm_vec4_addsub) +TEST_DECLARE(glm_vec4_mulsub) +TEST_DECLARE(glm_vec4_mulsubs) +TEST_DECLARE(glm_vec4_maxsub) +TEST_DECLARE(glm_vec4_minsub) +TEST_DECLARE(glm_vec4_negate_to) +TEST_DECLARE(glm_vec4_negate) +TEST_DECLARE(glm_vec4_normalize) +TEST_DECLARE(glm_vec4_normalize_to) +TEST_DECLARE(glm_vec4_distance2) +TEST_DECLARE(glm_vec4_distance) +TEST_DECLARE(glm_vec4_maxv) +TEST_DECLARE(glm_vec4_minv) +TEST_DECLARE(glm_vec4_clamp) +TEST_DECLARE(glm_vec4_lerp) +TEST_DECLARE(glm_vec4_lerpc) +TEST_DECLARE(glm_vec4_mix) +TEST_DECLARE(glm_vec4_mixc) +TEST_DECLARE(glm_vec4_step) +TEST_DECLARE(glm_vec4_smoothstep_uni) +TEST_DECLARE(glm_vec4_smoothstep) +TEST_DECLARE(glm_vec4_smoothinterp) +TEST_DECLARE(glm_vec4_smoothinterpc) +TEST_DECLARE(glm_vec4_cubic) +TEST_DECLARE(glm_vec4_swizzle) +TEST_DECLARE(glm_vec4_broadcast) +TEST_DECLARE(glm_vec4_fill) +TEST_DECLARE(glm_vec4_eq) +TEST_DECLARE(glm_vec4_eq_eps) +TEST_DECLARE(glm_vec4_eq_all) +TEST_DECLARE(glm_vec4_eqv) +TEST_DECLARE(glm_vec4_eqv_eps) +TEST_DECLARE(glm_vec4_max) +TEST_DECLARE(glm_vec4_min) +TEST_DECLARE(glm_vec4_isnan) +TEST_DECLARE(glm_vec4_isinf) +TEST_DECLARE(glm_vec4_isvalid) +TEST_DECLARE(glm_vec4_sign) +TEST_DECLARE(glm_vec4_abs) +TEST_DECLARE(glm_vec4_fract) +TEST_DECLARE(glm_vec4_floor) +TEST_DECLARE(glm_vec4_mods) +TEST_DECLARE(glm_vec4_steps) +TEST_DECLARE(glm_vec4_stepr) +TEST_DECLARE(glm_vec4_hadd) +TEST_DECLARE(glm_vec4_sqrt) +TEST_DECLARE(glm_vec4_make) +TEST_DECLARE(glm_vec4_reflect) +TEST_DECLARE(glm_vec4_refract) + +TEST_DECLARE(glmc_vec4) +TEST_DECLARE(glmc_vec4_copy3) +TEST_DECLARE(glmc_vec4_copy) +TEST_DECLARE(glmc_vec4_ucopy) +TEST_DECLARE(glmc_vec4_zero) +TEST_DECLARE(glmc_vec4_one) +TEST_DECLARE(glmc_vec4_dot) +TEST_DECLARE(glmc_vec4_norm2) +TEST_DECLARE(glmc_vec4_norm) +TEST_DECLARE(glmc_vec4_norm_one) +TEST_DECLARE(glmc_vec4_norm_inf) +TEST_DECLARE(glmc_vec4_add) +TEST_DECLARE(glmc_vec4_adds) +TEST_DECLARE(glmc_vec4_sub) +TEST_DECLARE(glmc_vec4_subs) +TEST_DECLARE(glmc_vec4_mul) +TEST_DECLARE(glmc_vec4_scale) +TEST_DECLARE(glmc_vec4_scale_as) +TEST_DECLARE(glmc_vec4_div) +TEST_DECLARE(glmc_vec4_divs) +TEST_DECLARE(glmc_vec4_addadd) +TEST_DECLARE(glmc_vec4_subadd) +TEST_DECLARE(glmc_vec4_muladd) +TEST_DECLARE(glmc_vec4_muladds) +TEST_DECLARE(glmc_vec4_maxadd) +TEST_DECLARE(glmc_vec4_minadd) +TEST_DECLARE(glmc_vec4_subsub) +TEST_DECLARE(glmc_vec4_addsub) +TEST_DECLARE(glmc_vec4_mulsub) +TEST_DECLARE(glmc_vec4_mulsubs) +TEST_DECLARE(glmc_vec4_maxsub) +TEST_DECLARE(glmc_vec4_minsub) +TEST_DECLARE(glmc_vec4_negate_to) +TEST_DECLARE(glmc_vec4_negate) +TEST_DECLARE(glmc_vec4_normalize) +TEST_DECLARE(glmc_vec4_normalize_to) +TEST_DECLARE(glmc_vec4_distance2) +TEST_DECLARE(glmc_vec4_distance) +TEST_DECLARE(glmc_vec4_maxv) +TEST_DECLARE(glmc_vec4_minv) +TEST_DECLARE(glmc_vec4_clamp) +TEST_DECLARE(glmc_vec4_lerp) +TEST_DECLARE(glmc_vec4_lerpc) +TEST_DECLARE(glmc_vec4_mix) +TEST_DECLARE(glmc_vec4_mixc) +TEST_DECLARE(glmc_vec4_step) +TEST_DECLARE(glmc_vec4_smoothstep_uni) +TEST_DECLARE(glmc_vec4_smoothstep) +TEST_DECLARE(glmc_vec4_smoothinterp) +TEST_DECLARE(glmc_vec4_smoothinterpc) +TEST_DECLARE(glmc_vec4_cubic) +TEST_DECLARE(glmc_vec4_swizzle) +TEST_DECLARE(glmc_vec4_broadcast) +TEST_DECLARE(glmc_vec4_fill) +TEST_DECLARE(glmc_vec4_eq) +TEST_DECLARE(glmc_vec4_eq_eps) +TEST_DECLARE(glmc_vec4_eq_all) +TEST_DECLARE(glmc_vec4_eqv) +TEST_DECLARE(glmc_vec4_eqv_eps) +TEST_DECLARE(glmc_vec4_max) +TEST_DECLARE(glmc_vec4_min) +TEST_DECLARE(glmc_vec4_isnan) +TEST_DECLARE(glmc_vec4_isinf) +TEST_DECLARE(glmc_vec4_isvalid) +TEST_DECLARE(glmc_vec4_sign) +TEST_DECLARE(glmc_vec4_abs) +TEST_DECLARE(glmc_vec4_fract) +TEST_DECLARE(glmc_vec4_floor) +TEST_DECLARE(glmc_vec4_mods) +TEST_DECLARE(glmc_vec4_steps) +TEST_DECLARE(glmc_vec4_stepr) +TEST_DECLARE(glmc_vec4_hadd) +TEST_DECLARE(glmc_vec4_sqrt) +TEST_DECLARE(glmc_vec4_make) +TEST_DECLARE(glmc_vec4_reflect) +TEST_DECLARE(glmc_vec4_refract) + +/* ivec2 */ +TEST_DECLARE(glm_ivec2) +TEST_DECLARE(glm_ivec2_copy) +TEST_DECLARE(glm_ivec2_zero) +TEST_DECLARE(glm_ivec2_one) +TEST_DECLARE(glm_ivec2_dot) +TEST_DECLARE(glm_ivec2_cross) +TEST_DECLARE(glm_ivec2_add) +TEST_DECLARE(glm_ivec2_adds) +TEST_DECLARE(glm_ivec2_sub) +TEST_DECLARE(glm_ivec2_subs) +TEST_DECLARE(glm_ivec2_mul) +TEST_DECLARE(glm_ivec2_scale) +TEST_DECLARE(glm_ivec2_div) +TEST_DECLARE(glm_ivec2_divs) +TEST_DECLARE(glm_ivec2_mod) +TEST_DECLARE(glm_ivec2_addadd) +TEST_DECLARE(glm_ivec2_addadds) +TEST_DECLARE(glm_ivec2_subadd) +TEST_DECLARE(glm_ivec2_subadds) +TEST_DECLARE(glm_ivec2_muladd) +TEST_DECLARE(glm_ivec2_muladds) +TEST_DECLARE(glm_ivec2_maxadd) +TEST_DECLARE(glm_ivec2_minadd) +TEST_DECLARE(glm_ivec2_subsub) +TEST_DECLARE(glm_ivec2_subsubs) +TEST_DECLARE(glm_ivec2_addsub) +TEST_DECLARE(glm_ivec2_addsubs) +TEST_DECLARE(glm_ivec2_mulsub) +TEST_DECLARE(glm_ivec2_mulsubs) +TEST_DECLARE(glm_ivec2_maxsub) +TEST_DECLARE(glm_ivec2_minsub) +TEST_DECLARE(glm_ivec2_distance2) +TEST_DECLARE(glm_ivec2_distance) +TEST_DECLARE(glm_ivec2_fill) +TEST_DECLARE(glm_ivec2_eq) +TEST_DECLARE(glm_ivec2_eqv) +TEST_DECLARE(glm_ivec2_maxv) +TEST_DECLARE(glm_ivec2_minv) +TEST_DECLARE(glm_ivec2_clamp) +TEST_DECLARE(glm_ivec2_abs) + +TEST_DECLARE(glmc_ivec2) +TEST_DECLARE(glmc_ivec2_copy) +TEST_DECLARE(glmc_ivec2_zero) +TEST_DECLARE(glmc_ivec2_one) +TEST_DECLARE(glmc_ivec2_dot) +TEST_DECLARE(glmc_ivec2_cross) +TEST_DECLARE(glmc_ivec2_add) +TEST_DECLARE(glmc_ivec2_adds) +TEST_DECLARE(glmc_ivec2_sub) +TEST_DECLARE(glmc_ivec2_subs) +TEST_DECLARE(glmc_ivec2_mul) +TEST_DECLARE(glmc_ivec2_scale) +TEST_DECLARE(glmc_ivec2_div) +TEST_DECLARE(glmc_ivec2_divs) +TEST_DECLARE(glmc_ivec2_mod) +TEST_DECLARE(glmc_ivec2_addadd) +TEST_DECLARE(glmc_ivec2_addadds) +TEST_DECLARE(glmc_ivec2_subadd) +TEST_DECLARE(glmc_ivec2_subadds) +TEST_DECLARE(glmc_ivec2_muladd) +TEST_DECLARE(glmc_ivec2_muladds) +TEST_DECLARE(glmc_ivec2_maxadd) +TEST_DECLARE(glmc_ivec2_minadd) +TEST_DECLARE(glmc_ivec2_subsub) +TEST_DECLARE(glmc_ivec2_subsubs) +TEST_DECLARE(glmc_ivec2_addsub) +TEST_DECLARE(glmc_ivec2_addsubs) +TEST_DECLARE(glmc_ivec2_mulsub) +TEST_DECLARE(glmc_ivec2_mulsubs) +TEST_DECLARE(glmc_ivec2_maxsub) +TEST_DECLARE(glmc_ivec2_minsub) +TEST_DECLARE(glmc_ivec2_distance2) +TEST_DECLARE(glmc_ivec2_distance) +TEST_DECLARE(glmc_ivec2_fill) +TEST_DECLARE(glmc_ivec2_eq) +TEST_DECLARE(glmc_ivec2_eqv) +TEST_DECLARE(glmc_ivec2_maxv) +TEST_DECLARE(glmc_ivec2_minv) +TEST_DECLARE(glmc_ivec2_clamp) +TEST_DECLARE(glmc_ivec2_abs) + +/* ivec3 */ +TEST_DECLARE(glm_ivec3) +TEST_DECLARE(glm_ivec3_copy) +TEST_DECLARE(glm_ivec3_zero) +TEST_DECLARE(glm_ivec3_one) +TEST_DECLARE(glm_ivec3_dot) +TEST_DECLARE(glm_ivec3_norm2) +TEST_DECLARE(glm_ivec3_norm) +TEST_DECLARE(glm_ivec3_add) +TEST_DECLARE(glm_ivec3_adds) +TEST_DECLARE(glm_ivec3_sub) +TEST_DECLARE(glm_ivec3_subs) +TEST_DECLARE(glm_ivec3_mul) +TEST_DECLARE(glm_ivec3_scale) +TEST_DECLARE(glm_ivec3_div) +TEST_DECLARE(glm_ivec3_divs) +TEST_DECLARE(glm_ivec3_mod) +TEST_DECLARE(glm_ivec3_addadd) +TEST_DECLARE(glm_ivec3_addadds) +TEST_DECLARE(glm_ivec3_subadd) +TEST_DECLARE(glm_ivec3_subadds) +TEST_DECLARE(glm_ivec3_muladd) +TEST_DECLARE(glm_ivec3_muladds) +TEST_DECLARE(glm_ivec3_maxadd) +TEST_DECLARE(glm_ivec3_minadd) +TEST_DECLARE(glm_ivec3_subsub) +TEST_DECLARE(glm_ivec3_subsubs) +TEST_DECLARE(glm_ivec3_addsub) +TEST_DECLARE(glm_ivec3_addsubs) +TEST_DECLARE(glm_ivec3_mulsub) +TEST_DECLARE(glm_ivec3_mulsubs) +TEST_DECLARE(glm_ivec3_maxsub) +TEST_DECLARE(glm_ivec3_minsub) +TEST_DECLARE(glm_ivec3_distance2) +TEST_DECLARE(glm_ivec3_distance) +TEST_DECLARE(glm_ivec3_fill) +TEST_DECLARE(glm_ivec3_eq) +TEST_DECLARE(glm_ivec3_eqv) +TEST_DECLARE(glm_ivec3_maxv) +TEST_DECLARE(glm_ivec3_minv) +TEST_DECLARE(glm_ivec3_clamp) + +TEST_DECLARE(glmc_ivec3) +TEST_DECLARE(glmc_ivec3_copy) +TEST_DECLARE(glmc_ivec3_zero) +TEST_DECLARE(glmc_ivec3_one) +TEST_DECLARE(glmc_ivec3_dot) +TEST_DECLARE(glmc_ivec3_norm2) +TEST_DECLARE(glmc_ivec3_norm) +TEST_DECLARE(glmc_ivec3_add) +TEST_DECLARE(glmc_ivec3_adds) +TEST_DECLARE(glmc_ivec3_sub) +TEST_DECLARE(glmc_ivec3_subs) +TEST_DECLARE(glmc_ivec3_mul) +TEST_DECLARE(glmc_ivec3_scale) +TEST_DECLARE(glmc_ivec3_div) +TEST_DECLARE(glmc_ivec3_divs) +TEST_DECLARE(glmc_ivec3_mod) +TEST_DECLARE(glmc_ivec3_addadd) +TEST_DECLARE(glmc_ivec3_addadds) +TEST_DECLARE(glmc_ivec3_subadd) +TEST_DECLARE(glmc_ivec3_subadds) +TEST_DECLARE(glmc_ivec3_muladd) +TEST_DECLARE(glmc_ivec3_muladds) +TEST_DECLARE(glmc_ivec3_maxadd) +TEST_DECLARE(glmc_ivec3_minadd) +TEST_DECLARE(glmc_ivec3_subsub) +TEST_DECLARE(glmc_ivec3_subsubs) +TEST_DECLARE(glmc_ivec3_addsub) +TEST_DECLARE(glmc_ivec3_addsubs) +TEST_DECLARE(glmc_ivec3_mulsub) +TEST_DECLARE(glmc_ivec3_mulsubs) +TEST_DECLARE(glmc_ivec3_maxsub) +TEST_DECLARE(glmc_ivec3_minsub) +TEST_DECLARE(glmc_ivec3_distance2) +TEST_DECLARE(glmc_ivec3_distance) +TEST_DECLARE(glmc_ivec3_fill) +TEST_DECLARE(glmc_ivec3_eq) +TEST_DECLARE(glmc_ivec3_eqv) +TEST_DECLARE(glmc_ivec3_maxv) +TEST_DECLARE(glmc_ivec3_minv) +TEST_DECLARE(glmc_ivec3_clamp) + +/* ivec4 */ +TEST_DECLARE(glm_ivec4) +TEST_DECLARE(glm_ivec4_copy) +TEST_DECLARE(glm_ivec4_zero) +TEST_DECLARE(glm_ivec4_one) +TEST_DECLARE(glm_ivec4_add) +TEST_DECLARE(glm_ivec4_adds) +TEST_DECLARE(glm_ivec4_sub) +TEST_DECLARE(glm_ivec4_subs) +TEST_DECLARE(glm_ivec4_mul) +TEST_DECLARE(glm_ivec4_scale) +TEST_DECLARE(glm_ivec4_addadd) +TEST_DECLARE(glm_ivec4_addadds) +TEST_DECLARE(glm_ivec4_subadd) +TEST_DECLARE(glm_ivec4_subadds) +TEST_DECLARE(glm_ivec4_muladd) +TEST_DECLARE(glm_ivec4_muladds) +TEST_DECLARE(glm_ivec4_maxadd) +TEST_DECLARE(glm_ivec4_minadd) +TEST_DECLARE(glm_ivec4_subsub) +TEST_DECLARE(glm_ivec4_subsubs) +TEST_DECLARE(glm_ivec4_addsub) +TEST_DECLARE(glm_ivec4_addsubs) +TEST_DECLARE(glm_ivec4_mulsub) +TEST_DECLARE(glm_ivec4_mulsubs) +TEST_DECLARE(glm_ivec4_maxsub) +TEST_DECLARE(glm_ivec4_minsub) +TEST_DECLARE(glm_ivec4_distance2) +TEST_DECLARE(glm_ivec4_distance) +TEST_DECLARE(glm_ivec4_maxv) +TEST_DECLARE(glm_ivec4_minv) +TEST_DECLARE(glm_ivec4_clamp) +TEST_DECLARE(glm_ivec4_abs) + +TEST_DECLARE(glmc_ivec4) +TEST_DECLARE(glmc_ivec4_copy) +TEST_DECLARE(glmc_ivec4_zero) +TEST_DECLARE(glmc_ivec4_one) +TEST_DECLARE(glmc_ivec4_add) +TEST_DECLARE(glmc_ivec4_adds) +TEST_DECLARE(glmc_ivec4_sub) +TEST_DECLARE(glmc_ivec4_subs) +TEST_DECLARE(glmc_ivec4_mul) +TEST_DECLARE(glmc_ivec4_scale) +TEST_DECLARE(glmc_ivec4_addadd) +TEST_DECLARE(glmc_ivec4_addadds) +TEST_DECLARE(glmc_ivec4_subadd) +TEST_DECLARE(glmc_ivec4_subadds) +TEST_DECLARE(glmc_ivec4_muladd) +TEST_DECLARE(glmc_ivec4_muladds) +TEST_DECLARE(glmc_ivec4_maxadd) +TEST_DECLARE(glmc_ivec4_minadd) +TEST_DECLARE(glmc_ivec4_subsub) +TEST_DECLARE(glmc_ivec4_subsubs) +TEST_DECLARE(glmc_ivec4_addsub) +TEST_DECLARE(glmc_ivec4_addsubs) +TEST_DECLARE(glmc_ivec4_mulsub) +TEST_DECLARE(glmc_ivec4_mulsubs) +TEST_DECLARE(glmc_ivec4_maxsub) +TEST_DECLARE(glmc_ivec4_minsub) +TEST_DECLARE(glmc_ivec4_distance2) +TEST_DECLARE(glmc_ivec4_distance) +TEST_DECLARE(glmc_ivec4_maxv) +TEST_DECLARE(glmc_ivec4_minv) +TEST_DECLARE(glmc_ivec4_clamp) +TEST_DECLARE(glmc_ivec4_abs) + +/* structs */ +TEST_DECLARE(mat2x3s_zero_init) +TEST_DECLARE(mat2x3s_zero) +TEST_DECLARE(mat2x4s_zero_init) +TEST_DECLARE(mat2x4s_zero) +TEST_DECLARE(mat3s_identity_init) +TEST_DECLARE(mat3s_zero_init) +TEST_DECLARE(mat3x2s_zero_init) +TEST_DECLARE(mat3x2s_zero) +TEST_DECLARE(mat3x4s_zero_init) +TEST_DECLARE(mat3x4s_zero) +TEST_DECLARE(mat4s_identity_init) +TEST_DECLARE(mat4s_zero_init) +TEST_DECLARE(mat4x2s_zero_init) +TEST_DECLARE(mat4x2s_zero) +TEST_DECLARE(mat4x3s_zero_init) +TEST_DECLARE(mat4x3s_zero) +TEST_DECLARE(quats_zero_init) +TEST_DECLARE(vec3s_one_init) +TEST_DECLARE(vec3s_zero_init) +TEST_DECLARE(vec4s_black_init) +TEST_DECLARE(vec4s_one_init) +TEST_DECLARE(vec4s_zero_init) + +/*****************************************************************************/ + +TEST_LIST { + /* affine mat */ + TEST_ENTRY(glm_mul) + TEST_ENTRY(glm_mul) + TEST_ENTRY(glm_inv_tr) + + TEST_ENTRY(glmc_mul) + TEST_ENTRY(glmc_mul_rot) + TEST_ENTRY(glmc_inv_tr) + + /* affine */ + TEST_ENTRY(glm_translate) + TEST_ENTRY(glm_translate_to) + TEST_ENTRY(glm_translate_x) + TEST_ENTRY(glm_translate_y) + TEST_ENTRY(glm_translate_z) + TEST_ENTRY(glm_translate_make) + TEST_ENTRY(glm_scale_to) + TEST_ENTRY(glm_scale_make) + TEST_ENTRY(glm_scale) + TEST_ENTRY(glm_scale_uni) + TEST_ENTRY(glm_rotate_x) + TEST_ENTRY(glm_rotate_y) + TEST_ENTRY(glm_rotate_z) + TEST_ENTRY(glm_rotate_make) + TEST_ENTRY(glm_rotate) + TEST_ENTRY(glm_rotate_at) + TEST_ENTRY(glm_rotate_atm) + TEST_ENTRY(glm_decompose_scalev) + TEST_ENTRY(glm_uniscaled) + TEST_ENTRY(glm_decompose_rs) + TEST_ENTRY(glm_decompose) + + TEST_ENTRY(glmc_translate) + TEST_ENTRY(glmc_translate_to) + TEST_ENTRY(glmc_translate_x) + TEST_ENTRY(glmc_translate_y) + TEST_ENTRY(glmc_translate_z) + TEST_ENTRY(glmc_translate_make) + TEST_ENTRY(glmc_scale_to) + TEST_ENTRY(glmc_scale_make) + TEST_ENTRY(glmc_scale) + TEST_ENTRY(glmc_scale_uni) + TEST_ENTRY(glmc_rotate_x) + TEST_ENTRY(glmc_rotate_y) + TEST_ENTRY(glmc_rotate_z) + TEST_ENTRY(glmc_rotate_make) + TEST_ENTRY(glmc_rotate) + TEST_ENTRY(glmc_rotate_at) + TEST_ENTRY(glmc_rotate_atm) + TEST_ENTRY(glmc_decompose_scalev) + TEST_ENTRY(glmc_uniscaled) + TEST_ENTRY(glmc_decompose_rs) + TEST_ENTRY(glmc_decompose) + + /* affine 2d */ + TEST_ENTRY(glm_translate2d) + TEST_ENTRY(glm_translate2d_to) + TEST_ENTRY(glm_translate2d_x) + TEST_ENTRY(glm_translate2d_y) + TEST_ENTRY(glm_translate2d_make) + TEST_ENTRY(glm_scale2d_to) + TEST_ENTRY(glm_scale2d_make) + TEST_ENTRY(glm_scale2d) + TEST_ENTRY(glm_scale2d_uni) + TEST_ENTRY(glm_rotate2d_make) + TEST_ENTRY(glm_rotate2d) + TEST_ENTRY(glm_rotate2d_to) + + TEST_ENTRY(glmc_translate2d) + TEST_ENTRY(glmc_translate2d_to) + TEST_ENTRY(glmc_translate2d_x) + TEST_ENTRY(glmc_translate2d_y) + TEST_ENTRY(glmc_translate2d_make) + TEST_ENTRY(glmc_scale2d_to) + TEST_ENTRY(glmc_scale2d_make) + TEST_ENTRY(glmc_scale2d) + TEST_ENTRY(glmc_scale2d_uni) + TEST_ENTRY(glmc_rotate2d_make) + TEST_ENTRY(glmc_rotate2d) + TEST_ENTRY(glmc_rotate2d_to) + + /* aabb2d */ + TEST_ENTRY(glm_aabb2d_sizev) + + /* mat4 */ + TEST_ENTRY(glm_mat4_ucopy) + TEST_ENTRY(glm_mat4_copy) + TEST_ENTRY(glm_mat4_identity) + TEST_ENTRY(glm_mat4_identity_array) + TEST_ENTRY(glm_mat4_zero) + TEST_ENTRY(glm_mat4_pick3) + TEST_ENTRY(glm_mat4_pick3t) + TEST_ENTRY(glm_mat4_ins3) + TEST_ENTRY(glm_mat4_mul) + TEST_ENTRY(glm_mat4_mulN) + TEST_ENTRY(glm_mat4_mulv) + TEST_ENTRY(glm_mat4_mulv3) + TEST_ENTRY(glm_mat4_trace) + TEST_ENTRY(glm_mat4_trace3) + TEST_ENTRY(glm_mat4_quat) + TEST_ENTRY(glm_mat4_transpose_to) + TEST_ENTRY(glm_mat4_transpose) + TEST_ENTRY(glm_mat4_scale_p) + TEST_ENTRY(glm_mat4_scale) + TEST_ENTRY(glm_mat4_det) + TEST_ENTRY(glm_mat4_inv) + TEST_ENTRY(glm_mat4_inv_fast) + TEST_ENTRY(glm_mat4_inv_precise) + TEST_ENTRY(glm_mat4_swap_col) + TEST_ENTRY(glm_mat4_swap_row) + TEST_ENTRY(glm_mat4_rmc) + TEST_ENTRY(glm_mat4_make) + + TEST_ENTRY(glmc_mat4_ucopy) + TEST_ENTRY(glmc_mat4_copy) + TEST_ENTRY(glmc_mat4_identity) + TEST_ENTRY(glmc_mat4_identity_array) + TEST_ENTRY(glmc_mat4_zero) + TEST_ENTRY(glmc_mat4_pick3) + TEST_ENTRY(glmc_mat4_pick3t) + TEST_ENTRY(glmc_mat4_ins3) + TEST_ENTRY(glmc_mat4_mul) + TEST_ENTRY(glmc_mat4_mulN) + TEST_ENTRY(glmc_mat4_mulv) + TEST_ENTRY(glmc_mat4_mulv3) + TEST_ENTRY(glmc_mat4_trace) + TEST_ENTRY(glmc_mat4_trace3) + TEST_ENTRY(glmc_mat4_quat) + TEST_ENTRY(glmc_mat4_transpose_to) + TEST_ENTRY(glmc_mat4_transpose) + TEST_ENTRY(glmc_mat4_scale_p) + TEST_ENTRY(glmc_mat4_scale) + TEST_ENTRY(glmc_mat4_det) + TEST_ENTRY(glmc_mat4_inv) + TEST_ENTRY(glmc_mat4_inv_fast) + TEST_ENTRY(glmc_mat4_swap_col) + TEST_ENTRY(glmc_mat4_swap_row) + TEST_ENTRY(glmc_mat4_rmc) + TEST_ENTRY(glmc_mat4_make) + + TEST_ENTRY(MACRO_GLM_MAT4X2_ZERO_INIT) + TEST_ENTRY(MACRO_GLM_MAT4X2_ZERO) + TEST_ENTRY(glm_mat4x2_copy) + TEST_ENTRY(glm_mat4x2_zero) + TEST_ENTRY(glm_mat4x2_make) + TEST_ENTRY(glm_mat4x2_mul) + TEST_ENTRY(glm_mat4x2_mulv) + TEST_ENTRY(glm_mat4x2_transpose) + TEST_ENTRY(glm_mat4x2_scale) + + TEST_ENTRY(glmc_mat4x2_copy) + TEST_ENTRY(glmc_mat4x2_zero) + TEST_ENTRY(glmc_mat4x2_make) + TEST_ENTRY(glmc_mat4x2_mul) + TEST_ENTRY(glmc_mat4x2_mulv) + TEST_ENTRY(glmc_mat4x2_transpose) + TEST_ENTRY(glmc_mat4x2_scale) + + TEST_ENTRY(MACRO_GLM_MAT4X3_ZERO_INIT) + TEST_ENTRY(MACRO_GLM_MAT4X3_ZERO) + TEST_ENTRY(glm_mat4x3_copy) + TEST_ENTRY(glm_mat4x3_zero) + TEST_ENTRY(glm_mat4x3_make) + TEST_ENTRY(glm_mat4x3_mul) + TEST_ENTRY(glm_mat4x3_mulv) + TEST_ENTRY(glm_mat4x3_transpose) + TEST_ENTRY(glm_mat4x3_scale) + + TEST_ENTRY(glmc_mat4x3_copy) + TEST_ENTRY(glmc_mat4x3_zero) + TEST_ENTRY(glmc_mat4x3_make) + TEST_ENTRY(glmc_mat4x3_mul) + TEST_ENTRY(glmc_mat4x3_mulv) + TEST_ENTRY(glmc_mat4x3_transpose) + TEST_ENTRY(glmc_mat4x3_scale) + + /* mat3 */ + TEST_ENTRY(glm_mat3_copy) + TEST_ENTRY(glm_mat3_identity) + TEST_ENTRY(glm_mat3_identity_array) + TEST_ENTRY(glm_mat3_zero) + TEST_ENTRY(glm_mat3_mul) + TEST_ENTRY(glm_mat3_mulv) + TEST_ENTRY(glm_mat3_trace) + TEST_ENTRY(glm_mat3_quat) + TEST_ENTRY(glm_mat3_transpose_to) + TEST_ENTRY(glm_mat3_transpose) + TEST_ENTRY(glm_mat3_scale) + TEST_ENTRY(glm_mat3_det) + TEST_ENTRY(glm_mat3_inv) + TEST_ENTRY(glm_mat3_swap_col) + TEST_ENTRY(glm_mat3_swap_row) + TEST_ENTRY(glm_mat3_rmc) + TEST_ENTRY(glm_mat3_make) + + TEST_ENTRY(glmc_mat3_copy) + TEST_ENTRY(glmc_mat3_identity) + TEST_ENTRY(glmc_mat3_identity_array) + TEST_ENTRY(glmc_mat3_zero) + TEST_ENTRY(glmc_mat3_mul) + TEST_ENTRY(glmc_mat3_mulv) + TEST_ENTRY(glmc_mat3_trace) + TEST_ENTRY(glmc_mat3_quat) + TEST_ENTRY(glmc_mat3_transpose_to) + TEST_ENTRY(glmc_mat3_transpose) + TEST_ENTRY(glmc_mat3_scale) + TEST_ENTRY(glmc_mat3_det) + TEST_ENTRY(glmc_mat3_inv) + TEST_ENTRY(glmc_mat3_swap_col) + TEST_ENTRY(glmc_mat3_swap_row) + TEST_ENTRY(glmc_mat3_rmc) + TEST_ENTRY(glmc_mat3_make) + + TEST_ENTRY(MACRO_GLM_MAT3X2_ZERO_INIT) + TEST_ENTRY(MACRO_GLM_MAT3X2_ZERO) + TEST_ENTRY(glm_mat3x2_copy) + TEST_ENTRY(glm_mat3x2_zero) + TEST_ENTRY(glm_mat3x2_make) + TEST_ENTRY(glm_mat3x2_mul) + TEST_ENTRY(glm_mat3x2_mulv) + TEST_ENTRY(glm_mat3x2_transpose) + TEST_ENTRY(glm_mat3x2_scale) + + TEST_ENTRY(glmc_mat3x2_copy) + TEST_ENTRY(glmc_mat3x2_zero) + TEST_ENTRY(glmc_mat3x2_make) + TEST_ENTRY(glmc_mat3x2_mul) + TEST_ENTRY(glmc_mat3x2_mulv) + TEST_ENTRY(glmc_mat3x2_transpose) + TEST_ENTRY(glmc_mat3x2_scale) + + TEST_ENTRY(MACRO_GLM_MAT3X4_ZERO_INIT) + TEST_ENTRY(MACRO_GLM_MAT3X4_ZERO) + TEST_ENTRY(glm_mat3x4_make) + + TEST_ENTRY(glmc_mat3x4_make) + + TEST_ENTRY(MACRO_GLM_MAT2_IDENTITY_INIT) + TEST_ENTRY(MACRO_GLM_MAT2_ZERO_INIT) + TEST_ENTRY(MACRO_GLM_MAT2_IDENTITY) + TEST_ENTRY(MACRO_GLM_MAT2_ZERO) + TEST_ENTRY(glm_mat2_copy) + TEST_ENTRY(glm_mat2_identity) + TEST_ENTRY(glm_mat2_identity_array) + TEST_ENTRY(glm_mat2_zero) + TEST_ENTRY(glm_mat2_mul) + TEST_ENTRY(glm_mat2_transpose_to) + TEST_ENTRY(glm_mat2_transpose) + TEST_ENTRY(glm_mat2_mulv) + TEST_ENTRY(glm_mat2_trace) + TEST_ENTRY(glm_mat2_scale) + TEST_ENTRY(glm_mat2_det) + TEST_ENTRY(glm_mat2_inv) + TEST_ENTRY(glm_mat2_swap_col) + TEST_ENTRY(glm_mat2_swap_row) + TEST_ENTRY(glm_mat2_rmc) + TEST_ENTRY(glm_mat2_make) + + TEST_ENTRY(glmc_mat2_copy) + TEST_ENTRY(glmc_mat2_identity) + TEST_ENTRY(glmc_mat2_identity_array) + TEST_ENTRY(glmc_mat2_zero) + TEST_ENTRY(glmc_mat2_mul) + TEST_ENTRY(glmc_mat2_transpose_to) + TEST_ENTRY(glmc_mat2_transpose) + TEST_ENTRY(glmc_mat2_mulv) + TEST_ENTRY(glmc_mat2_trace) + TEST_ENTRY(glmc_mat2_scale) + TEST_ENTRY(glmc_mat2_det) + TEST_ENTRY(glmc_mat2_inv) + TEST_ENTRY(glmc_mat2_swap_col) + TEST_ENTRY(glmc_mat2_swap_row) + TEST_ENTRY(glmc_mat2_rmc) + TEST_ENTRY(glmc_mat2_make) + + TEST_ENTRY(MACRO_GLM_MAT2X3_ZERO_INIT) + TEST_ENTRY(MACRO_GLM_MAT2X3_ZERO) + TEST_ENTRY(glm_mat2x3_copy) + TEST_ENTRY(glm_mat2x3_zero) + TEST_ENTRY(glm_mat2x3_make) + TEST_ENTRY(glm_mat2x3_mul) + TEST_ENTRY(glm_mat2x3_mulv) + TEST_ENTRY(glm_mat2x3_transpose) + TEST_ENTRY(glm_mat2x3_scale) + + TEST_ENTRY(glmc_mat2x3_copy) + TEST_ENTRY(glmc_mat2x3_zero) + TEST_ENTRY(glmc_mat2x3_make) + TEST_ENTRY(glmc_mat2x3_mul) + TEST_ENTRY(glmc_mat2x3_mulv) + TEST_ENTRY(glmc_mat2x3_transpose) + TEST_ENTRY(glmc_mat2x3_scale) + + TEST_ENTRY(MACRO_GLM_MAT2X4_ZERO_INIT) + TEST_ENTRY(MACRO_GLM_MAT2X4_ZERO) + TEST_ENTRY(glm_mat2x4_copy) + TEST_ENTRY(glm_mat2x4_zero) + TEST_ENTRY(glm_mat2x4_make) + TEST_ENTRY(glm_mat2x4_mul) + TEST_ENTRY(glm_mat2x4_mulv) + TEST_ENTRY(glm_mat2x4_transpose) + TEST_ENTRY(glm_mat2x4_scale) + + TEST_ENTRY(glmc_mat2x4_copy) + TEST_ENTRY(glmc_mat2x4_zero) + TEST_ENTRY(glmc_mat2x4_make) + TEST_ENTRY(glmc_mat2x4_mul) + TEST_ENTRY(glmc_mat2x4_mulv) + TEST_ENTRY(glmc_mat2x4_transpose) + TEST_ENTRY(glmc_mat2x4_scale) + + /* camera (incl [LR]H cross [NZ]O) */ + TEST_ENTRY(glm_perspective_lh_zo) + TEST_ENTRY(glm_perspective_rh_zo) + TEST_ENTRY(glm_perspective_lh_no) + TEST_ENTRY(glm_perspective_rh_no) + TEST_ENTRY(glm_camera_lookat) + TEST_ENTRY(glm_camera_decomp) + + TEST_ENTRY(glmc_perspective_lh_zo) + TEST_ENTRY(glmc_perspective_rh_zo) + TEST_ENTRY(glmc_perspective_lh_no) + TEST_ENTRY(glmc_perspective_rh_no) + TEST_ENTRY(glmc_camera_lookat) + TEST_ENTRY(glmc_camera_decomp) + + TEST_ENTRY(glm_frustum) + + TEST_ENTRY(glmc_frustum) + + /* project */ + TEST_ENTRY(glm_unprojecti) + TEST_ENTRY(glm_unproject) + TEST_ENTRY(glm_project) + + TEST_ENTRY(glmc_unprojecti) + TEST_ENTRY(glmc_unproject) + TEST_ENTRY(glmc_project) + + /* plane */ + TEST_ENTRY(glm_plane_normalize) + TEST_ENTRY(glmc_plane_normalize) + + /* noise */ + TEST_ENTRY(glm_perlin_vec4) + TEST_ENTRY(glmc_perlin_vec4) + TEST_ENTRY(glm_perlin_vec3) + TEST_ENTRY(glmc_perlin_vec3) + TEST_ENTRY(glm_perlin_vec2) + TEST_ENTRY(glmc_perlin_vec2) + + /* utils */ + TEST_ENTRY(clamp) + + /* euler */ + TEST_ENTRY(glm_euler_xyz_quat_rh) + TEST_ENTRY(glm_euler_xzy_quat_rh) + TEST_ENTRY(glm_euler_yxz_quat_rh) + TEST_ENTRY(glm_euler_yzx_quat_rh) + TEST_ENTRY(glm_euler_zxy_quat_rh) + TEST_ENTRY(glm_euler_zyx_quat_rh) + + TEST_ENTRY(glm_euler_xyz_quat_lh) + TEST_ENTRY(glm_euler_xzy_quat_lh) + TEST_ENTRY(glm_euler_yxz_quat_lh) + TEST_ENTRY(glm_euler_yzx_quat_lh) + TEST_ENTRY(glm_euler_zxy_quat_lh) + TEST_ENTRY(glm_euler_zyx_quat_lh) + + TEST_ENTRY(glmc_euler_xyz_quat_rh) + TEST_ENTRY(glmc_euler_xzy_quat_rh) + TEST_ENTRY(glmc_euler_yxz_quat_rh) + TEST_ENTRY(glmc_euler_yzx_quat_rh) + TEST_ENTRY(glmc_euler_zxy_quat_rh) + TEST_ENTRY(glmc_euler_zyx_quat_rh) + + TEST_ENTRY(glmc_euler_xyz_quat_lh) + TEST_ENTRY(glmc_euler_xzy_quat_lh) + TEST_ENTRY(glmc_euler_yxz_quat_lh) + TEST_ENTRY(glmc_euler_yzx_quat_lh) + TEST_ENTRY(glmc_euler_zxy_quat_lh) + TEST_ENTRY(glmc_euler_zyx_quat_lh) + + TEST_ENTRY(euler) + + /* ray */ + TEST_ENTRY(glm_ray_triangle) + TEST_ENTRY(glm_ray_sphere) + TEST_ENTRY(glm_ray_at) + + TEST_ENTRY(glmc_ray_triangle) + TEST_ENTRY(glmc_ray_sphere) + TEST_ENTRY(glmc_ray_at) + + /* quat */ + TEST_ENTRY(MACRO_GLM_QUAT_IDENTITY_INIT) + TEST_ENTRY(MACRO_GLM_QUAT_IDENTITY) + + TEST_ENTRY(glm_quat_identity) + TEST_ENTRY(glm_quat_identity_array) + TEST_ENTRY(glm_quat_init) + TEST_ENTRY(glm_quatv) + TEST_ENTRY(glm_quat) + TEST_ENTRY(glm_quat_copy) + TEST_ENTRY(glm_quat_norm) + TEST_ENTRY(glm_quat_normalize_to) + TEST_ENTRY(glm_quat_normalize) + TEST_ENTRY(glm_quat_dot) + TEST_ENTRY(glm_quat_conjugate) + TEST_ENTRY(glm_quat_inv) + TEST_ENTRY(glm_quat_add) + TEST_ENTRY(glm_quat_sub) + TEST_ENTRY(glm_quat_real) + TEST_ENTRY(glm_quat_imag) + TEST_ENTRY(glm_quat_imagn) + TEST_ENTRY(glm_quat_imaglen) + TEST_ENTRY(glm_quat_angle) + TEST_ENTRY(glm_quat_axis) + TEST_ENTRY(glm_quat_mul) + TEST_ENTRY(glm_quat_mat4) + TEST_ENTRY(glm_quat_mat4t) + TEST_ENTRY(glm_quat_mat3) + TEST_ENTRY(glm_quat_mat3t) + TEST_ENTRY(glm_quat_lerp) + TEST_ENTRY(glm_quat_lerpc) + TEST_ENTRY(glm_quat_nlerp) + TEST_ENTRY(glm_quat_slerp) + TEST_ENTRY(glm_quat_look) + TEST_ENTRY(glm_quat_for) + TEST_ENTRY(glm_quat_forp) + TEST_ENTRY(glm_quat_rotatev) + TEST_ENTRY(glm_quat_rotate) + TEST_ENTRY(glm_quat_rotate_at) + TEST_ENTRY(glm_quat_rotate_atm) + TEST_ENTRY(glm_quat_from_vecs) + TEST_ENTRY(glm_quat_make) + + TEST_ENTRY(glmc_quat_identity) + TEST_ENTRY(glmc_quat_identity_array) + TEST_ENTRY(glmc_quat_init) + TEST_ENTRY(glmc_quatv) + TEST_ENTRY(glmc_quat) + TEST_ENTRY(glmc_quat_copy) + TEST_ENTRY(glmc_quat_norm) + TEST_ENTRY(glmc_quat_normalize_to) + TEST_ENTRY(glmc_quat_normalize) + TEST_ENTRY(glmc_quat_dot) + TEST_ENTRY(glmc_quat_conjugate) + TEST_ENTRY(glmc_quat_inv) + TEST_ENTRY(glmc_quat_add) + TEST_ENTRY(glmc_quat_sub) + TEST_ENTRY(glmc_quat_real) + TEST_ENTRY(glmc_quat_imag) + TEST_ENTRY(glmc_quat_imagn) + TEST_ENTRY(glmc_quat_imaglen) + TEST_ENTRY(glmc_quat_angle) + TEST_ENTRY(glmc_quat_axis) + TEST_ENTRY(glmc_quat_mul) + TEST_ENTRY(glmc_quat_mat4) + TEST_ENTRY(glmc_quat_mat4t) + TEST_ENTRY(glmc_quat_mat3) + TEST_ENTRY(glmc_quat_mat3t) + TEST_ENTRY(glmc_quat_lerp) + TEST_ENTRY(glmc_quat_lerpc) + TEST_ENTRY(glmc_quat_nlerp) + TEST_ENTRY(glmc_quat_slerp) + TEST_ENTRY(glmc_quat_look) + TEST_ENTRY(glmc_quat_for) + TEST_ENTRY(glmc_quat_forp) + TEST_ENTRY(glmc_quat_rotatev) + TEST_ENTRY(glmc_quat_rotate) + TEST_ENTRY(glmc_quat_rotate_at) + TEST_ENTRY(glmc_quat_rotate_atm) + TEST_ENTRY(glmc_quat_from_vecs) + TEST_ENTRY(glmc_quat_make) + + /* bezier */ + TEST_ENTRY(bezier) + + /* vec2 */ + TEST_ENTRY(MACRO_GLM_VEC2_ONE_INIT) + TEST_ENTRY(MACRO_GLM_VEC2_ZERO_INIT) + TEST_ENTRY(MACRO_GLM_VEC2_ONE) + TEST_ENTRY(MACRO_GLM_VEC2_ZERO) + + TEST_ENTRY(glm_vec2) + TEST_ENTRY(glm_vec2_copy) + TEST_ENTRY(glm_vec2_zero) + TEST_ENTRY(glm_vec2_one) + TEST_ENTRY(glm_vec2_dot) + TEST_ENTRY(glm_vec2_cross) + TEST_ENTRY(glm_vec2_norm2) + TEST_ENTRY(glm_vec2_norm) + TEST_ENTRY(glm_vec2_add) + TEST_ENTRY(glm_vec2_adds) + TEST_ENTRY(glm_vec2_sub) + TEST_ENTRY(glm_vec2_subs) + TEST_ENTRY(glm_vec2_mul) + TEST_ENTRY(glm_vec2_scale) + TEST_ENTRY(glm_vec2_scale_as) + TEST_ENTRY(glm_vec2_div) + TEST_ENTRY(glm_vec2_divs) + TEST_ENTRY(glm_vec2_addadd) + TEST_ENTRY(glm_vec2_subadd) + TEST_ENTRY(glm_vec2_muladd) + TEST_ENTRY(glm_vec2_muladds) + TEST_ENTRY(glm_vec2_maxadd) + TEST_ENTRY(glm_vec2_minadd) + TEST_ENTRY(glm_vec2_subsub) + TEST_ENTRY(glm_vec2_addsub) + TEST_ENTRY(glm_vec2_mulsub) + TEST_ENTRY(glm_vec2_mulsubs) + TEST_ENTRY(glm_vec2_maxsub) + TEST_ENTRY(glm_vec2_minsub) + TEST_ENTRY(glm_vec2_negate_to) + TEST_ENTRY(glm_vec2_negate) + TEST_ENTRY(glm_vec2_normalize) + TEST_ENTRY(glm_vec2_normalize_to) + TEST_ENTRY(glm_vec2_rotate) + TEST_ENTRY(glm_vec2_center) + TEST_ENTRY(glm_vec2_distance2) + TEST_ENTRY(glm_vec2_distance) + TEST_ENTRY(glm_vec2_maxv) + TEST_ENTRY(glm_vec2_minv) + TEST_ENTRY(glm_vec2_clamp) + TEST_ENTRY(glm_vec2_abs) + TEST_ENTRY(glm_vec2_fract) + TEST_ENTRY(glm_vec2_floor) + TEST_ENTRY(glm_vec2_mods) + TEST_ENTRY(glm_vec2_steps) + TEST_ENTRY(glm_vec2_stepr) + TEST_ENTRY(glm_vec2_lerp) + TEST_ENTRY(glm_vec2_complex_mul) + TEST_ENTRY(glm_vec2_complex_div) + TEST_ENTRY(glm_vec2_make) + TEST_ENTRY(glm_vec2_reflect) + TEST_ENTRY(glm_vec2_refract) + + TEST_ENTRY(glmc_vec2) + TEST_ENTRY(glmc_vec2_copy) + TEST_ENTRY(glmc_vec2_zero) + TEST_ENTRY(glmc_vec2_one) + TEST_ENTRY(glmc_vec2_dot) + TEST_ENTRY(glmc_vec2_cross) + TEST_ENTRY(glmc_vec2_norm2) + TEST_ENTRY(glmc_vec2_norm) + TEST_ENTRY(glmc_vec2_add) + TEST_ENTRY(glmc_vec2_adds) + TEST_ENTRY(glmc_vec2_sub) + TEST_ENTRY(glmc_vec2_subs) + TEST_ENTRY(glmc_vec2_mul) + TEST_ENTRY(glmc_vec2_scale) + TEST_ENTRY(glmc_vec2_scale_as) + TEST_ENTRY(glmc_vec2_div) + TEST_ENTRY(glmc_vec2_divs) + TEST_ENTRY(glmc_vec2_addadd) + TEST_ENTRY(glmc_vec2_subadd) + TEST_ENTRY(glmc_vec2_muladd) + TEST_ENTRY(glmc_vec2_muladds) + TEST_ENTRY(glmc_vec2_maxadd) + TEST_ENTRY(glmc_vec2_minadd) + TEST_ENTRY(glmc_vec2_subsub) + TEST_ENTRY(glmc_vec2_addsub) + TEST_ENTRY(glmc_vec2_mulsub) + TEST_ENTRY(glmc_vec2_mulsubs) + TEST_ENTRY(glmc_vec2_maxsub) + TEST_ENTRY(glmc_vec2_minsub) + TEST_ENTRY(glmc_vec2_negate_to) + TEST_ENTRY(glmc_vec2_negate) + TEST_ENTRY(glmc_vec2_normalize) + TEST_ENTRY(glmc_vec2_normalize_to) + TEST_ENTRY(glmc_vec2_rotate) + TEST_ENTRY(glmc_vec2_center) + TEST_ENTRY(glmc_vec2_distance2) + TEST_ENTRY(glmc_vec2_distance) + TEST_ENTRY(glmc_vec2_maxv) + TEST_ENTRY(glmc_vec2_minv) + TEST_ENTRY(glmc_vec2_clamp) + TEST_ENTRY(glmc_vec2_abs) + TEST_ENTRY(glmc_vec2_fract) + TEST_ENTRY(glmc_vec2_floor) + TEST_ENTRY(glmc_vec2_mods) + TEST_ENTRY(glmc_vec2_steps) + TEST_ENTRY(glmc_vec2_stepr) + TEST_ENTRY(glmc_vec2_lerp) + TEST_ENTRY(glmc_vec2_complex_mul) + TEST_ENTRY(glmc_vec2_complex_div) + TEST_ENTRY(glmc_vec2_make) + TEST_ENTRY(glmc_vec2_reflect) + TEST_ENTRY(glmc_vec2_refract) + + /* vec3 */ + TEST_ENTRY(MACRO_GLM_VEC3_ONE_INIT) + TEST_ENTRY(MACRO_GLM_VEC3_ZERO_INIT) + TEST_ENTRY(MACRO_GLM_VEC3_ONE) + TEST_ENTRY(MACRO_GLM_VEC3_ZERO) + TEST_ENTRY(MACRO_GLM_YUP) + TEST_ENTRY(MACRO_GLM_ZUP) + TEST_ENTRY(MACRO_GLM_XUP) + TEST_ENTRY(MACRO_GLM_FORWARD_RH) + TEST_ENTRY(MACRO_GLM_SHUFFLE3) + TEST_ENTRY(MACRO_GLM_XXX) + TEST_ENTRY(MACRO_GLM_YYY) + TEST_ENTRY(MACRO_GLM_ZZZ) + TEST_ENTRY(MACRO_GLM_ZYX) + TEST_ENTRY(MACRO_glm_vec3_dup) + TEST_ENTRY(MACRO_glm_vec3_flipsign) + TEST_ENTRY(MACRO_glm_vec3_flipsign_to) + TEST_ENTRY(MACRO_glm_vec3_inv) + TEST_ENTRY(MACRO_glm_vec3_inv_to) + TEST_ENTRY(MACRO_glm_vec3_mulv) + + TEST_ENTRY(glm_vec3) + TEST_ENTRY(glm_vec3_copy) + TEST_ENTRY(glm_vec3_zero) + TEST_ENTRY(glm_vec3_one) + TEST_ENTRY(glm_vec3_dot) + TEST_ENTRY(glm_dot) + TEST_ENTRY(glm_vec3_norm2) + TEST_ENTRY(glm_vec3_norm) + TEST_ENTRY(glm_vec3_norm_one) + TEST_ENTRY(glm_vec3_norm_inf) + TEST_ENTRY(glm_vec3_add) + TEST_ENTRY(glm_vec3_adds) + TEST_ENTRY(glm_vec3_sub) + TEST_ENTRY(glm_vec3_subs) + TEST_ENTRY(glm_vec3_mul) + TEST_ENTRY(glm_vec3_scale) + TEST_ENTRY(glm_vec3_scale_as) + TEST_ENTRY(glm_vec3_div) + TEST_ENTRY(glm_vec3_divs) + TEST_ENTRY(glm_vec3_addadd) + TEST_ENTRY(glm_vec3_subadd) + TEST_ENTRY(glm_vec3_muladd) + TEST_ENTRY(glm_vec3_muladds) + TEST_ENTRY(glm_vec3_maxadd) + TEST_ENTRY(glm_vec3_minadd) + TEST_ENTRY(glm_vec3_subsub) + TEST_ENTRY(glm_vec3_addsub) + TEST_ENTRY(glm_vec3_mulsub) + TEST_ENTRY(glm_vec3_mulsubs) + TEST_ENTRY(glm_vec3_maxsub) + TEST_ENTRY(glm_vec3_minsub) + TEST_ENTRY(glm_vec3_negate_to) + TEST_ENTRY(glm_vec3_negate) + TEST_ENTRY(glm_vec3_normalize) + TEST_ENTRY(glm_vec3_normalize_to) + TEST_ENTRY(glm_normalize) + TEST_ENTRY(glm_normalize_to) + TEST_ENTRY(glm_vec3_cross) + TEST_ENTRY(glm_vec3_crossn) + TEST_ENTRY(glm_cross) + TEST_ENTRY(glm_vec3_angle) + TEST_ENTRY(glm_vec3_rotate) + TEST_ENTRY(glm_vec3_rotate_m4) + TEST_ENTRY(glm_vec3_rotate_m3) + TEST_ENTRY(glm_vec3_proj) + TEST_ENTRY(glm_vec3_center) + TEST_ENTRY(glmc_vec3_distance2) + TEST_ENTRY(glmc_vec3_distance) + TEST_ENTRY(glm_vec3_maxv) + TEST_ENTRY(glm_vec3_minv) + TEST_ENTRY(glm_vec3_ortho) + TEST_ENTRY(glm_vec3_clamp) + TEST_ENTRY(glm_vec3_mix) + TEST_ENTRY(glm_vec3_mixc) + TEST_ENTRY(glm_vec3_step) + TEST_ENTRY(glm_vec3_smoothstep_uni) + TEST_ENTRY(glm_vec3_smoothstep) + TEST_ENTRY(glm_vec3_smoothinterp) + TEST_ENTRY(glm_vec3_smoothinterpc) + TEST_ENTRY(glm_vec3_swizzle) + TEST_ENTRY(glm_vec3_broadcast) + TEST_ENTRY(glm_vec3_fill) + TEST_ENTRY(glm_vec3_eq) + TEST_ENTRY(glm_vec3_eq_eps) + TEST_ENTRY(glm_vec3_eq_all) + TEST_ENTRY(glm_vec3_eqv) + TEST_ENTRY(glm_vec3_eqv_eps) + TEST_ENTRY(glm_vec3_max) + TEST_ENTRY(glm_vec3_min) + TEST_ENTRY(glm_vec3_isnan) + TEST_ENTRY(glm_vec3_isinf) + TEST_ENTRY(glm_vec3_isvalid) + TEST_ENTRY(glm_vec3_sign) + TEST_ENTRY(glm_vec3_abs) + TEST_ENTRY(glm_vec3_fract) + TEST_ENTRY(glm_vec3_floor) + TEST_ENTRY(glm_vec3_mods) + TEST_ENTRY(glm_vec3_steps) + TEST_ENTRY(glm_vec3_stepr) + TEST_ENTRY(glm_vec3_hadd) + TEST_ENTRY(glm_vec3_sqrt) + TEST_ENTRY(glm_vec3_make) + TEST_ENTRY(glm_vec3_faceforward) + TEST_ENTRY(glm_vec3_reflect) + TEST_ENTRY(glm_vec3_refract) + + TEST_ENTRY(glmc_vec3) + TEST_ENTRY(glmc_vec3_copy) + TEST_ENTRY(glmc_vec3_zero) + TEST_ENTRY(glmc_vec3_one) + TEST_ENTRY(glmc_vec3_dot) + TEST_ENTRY(glmc_vec3_norm2) + TEST_ENTRY(glmc_vec3_norm) + TEST_ENTRY(glmc_vec3_norm_one) + TEST_ENTRY(glmc_vec3_norm_inf) + TEST_ENTRY(glmc_vec3_add) + TEST_ENTRY(glmc_vec3_adds) + TEST_ENTRY(glmc_vec3_sub) + TEST_ENTRY(glmc_vec3_subs) + TEST_ENTRY(glmc_vec3_mul) + TEST_ENTRY(glmc_vec3_scale) + TEST_ENTRY(glmc_vec3_scale_as) + TEST_ENTRY(glmc_vec3_div) + TEST_ENTRY(glmc_vec3_divs) + TEST_ENTRY(glmc_vec3_addadd) + TEST_ENTRY(glmc_vec3_subadd) + TEST_ENTRY(glmc_vec3_muladd) + TEST_ENTRY(glmc_vec3_muladds) + TEST_ENTRY(glmc_vec3_maxadd) + TEST_ENTRY(glmc_vec3_minadd) + TEST_ENTRY(glmc_vec3_subsub) + TEST_ENTRY(glmc_vec3_addsub) + TEST_ENTRY(glmc_vec3_mulsub) + TEST_ENTRY(glmc_vec3_mulsubs) + TEST_ENTRY(glmc_vec3_maxsub) + TEST_ENTRY(glmc_vec3_minsub) + TEST_ENTRY(glmc_vec3_negate_to) + TEST_ENTRY(glmc_vec3_negate) + TEST_ENTRY(glmc_vec3_normalize) + TEST_ENTRY(glmc_vec3_normalize_to) + TEST_ENTRY(glmc_vec3_cross) + TEST_ENTRY(glmc_vec3_crossn) + TEST_ENTRY(glmc_vec3_angle) + TEST_ENTRY(glmc_vec3_rotate) + TEST_ENTRY(glmc_vec3_rotate_m4) + TEST_ENTRY(glmc_vec3_rotate_m3) + TEST_ENTRY(glmc_vec3_proj) + TEST_ENTRY(glmc_vec3_center) + TEST_ENTRY(glmc_vec3_distance2) + TEST_ENTRY(glmc_vec3_distance) + TEST_ENTRY(glmc_vec3_maxv) + TEST_ENTRY(glmc_vec3_minv) + TEST_ENTRY(glmc_vec3_ortho) + TEST_ENTRY(glmc_vec3_clamp) + TEST_ENTRY(glmc_vec3_mix) + TEST_ENTRY(glmc_vec3_mixc) + TEST_ENTRY(glmc_vec3_step) + TEST_ENTRY(glmc_vec3_smoothstep_uni) + TEST_ENTRY(glmc_vec3_smoothstep) + TEST_ENTRY(glmc_vec3_smoothinterp) + TEST_ENTRY(glmc_vec3_smoothinterpc) + TEST_ENTRY(glmc_vec3_swizzle) + TEST_ENTRY(glmc_vec3_broadcast) + TEST_ENTRY(glmc_vec3_fill) + TEST_ENTRY(glmc_vec3_eq) + TEST_ENTRY(glmc_vec3_eq_eps) + TEST_ENTRY(glmc_vec3_eq_all) + TEST_ENTRY(glmc_vec3_eqv) + TEST_ENTRY(glmc_vec3_eqv_eps) + TEST_ENTRY(glmc_vec3_max) + TEST_ENTRY(glmc_vec3_min) + TEST_ENTRY(glmc_vec3_isnan) + TEST_ENTRY(glmc_vec3_isinf) + TEST_ENTRY(glmc_vec3_isvalid) + TEST_ENTRY(glmc_vec3_sign) + TEST_ENTRY(glmc_vec3_abs) + TEST_ENTRY(glmc_vec3_fract) + TEST_ENTRY(glmc_vec3_floor) + TEST_ENTRY(glmc_vec3_mods) + TEST_ENTRY(glmc_vec3_steps) + TEST_ENTRY(glmc_vec3_stepr) + TEST_ENTRY(glmc_vec3_hadd) + TEST_ENTRY(glmc_vec3_sqrt) + TEST_ENTRY(glmc_vec3_make) + TEST_ENTRY(glmc_vec3_faceforward) + TEST_ENTRY(glmc_vec3_reflect) + TEST_ENTRY(glmc_vec3_refract) + + /* vec4 */ + TEST_ENTRY(MACRO_GLM_VEC4_ONE_INIT) + TEST_ENTRY(MACRO_GLM_VEC4_ZERO_INIT) + TEST_ENTRY(MACRO_GLM_VEC4_ONE) + TEST_ENTRY(MACRO_GLM_VEC4_ZERO) + TEST_ENTRY(MACRO_GLM_XXXX) + TEST_ENTRY(MACRO_GLM_YYYY) + TEST_ENTRY(MACRO_GLM_ZZZZ) + TEST_ENTRY(MACRO_GLM_WZYX) + TEST_ENTRY(MACRO_glm_vec4_dup) + TEST_ENTRY(MACRO_glm_vec4_flipsign) + TEST_ENTRY(MACRO_glm_vec4_flipsign_to) + TEST_ENTRY(MACRO_glm_vec4_inv) + TEST_ENTRY(MACRO_glm_vec4_inv_to) + TEST_ENTRY(MACRO_glm_vec4_mulv) + + TEST_ENTRY(glm_vec4) + TEST_ENTRY(glm_vec4_copy3) + TEST_ENTRY(glm_vec4_copy) + TEST_ENTRY(glm_vec4_ucopy) + TEST_ENTRY(glm_vec4_zero) + TEST_ENTRY(glm_vec4_one) + TEST_ENTRY(glm_vec4_dot) + TEST_ENTRY(glm_vec4_norm2) + TEST_ENTRY(glm_vec4_norm) + TEST_ENTRY(glm_vec4_norm_one) + TEST_ENTRY(glm_vec4_norm_inf) + TEST_ENTRY(glm_vec4_add) + TEST_ENTRY(glm_vec4_adds) + TEST_ENTRY(glm_vec4_sub) + TEST_ENTRY(glm_vec4_subs) + TEST_ENTRY(glm_vec4_mul) + TEST_ENTRY(glm_vec4_scale) + TEST_ENTRY(glm_vec4_scale_as) + TEST_ENTRY(glm_vec4_div) + TEST_ENTRY(glm_vec4_divs) + TEST_ENTRY(glm_vec4_addadd) + TEST_ENTRY(glm_vec4_subadd) + TEST_ENTRY(glm_vec4_muladd) + TEST_ENTRY(glm_vec4_muladds) + TEST_ENTRY(glm_vec4_maxadd) + TEST_ENTRY(glm_vec4_minadd) + TEST_ENTRY(glm_vec4_subsub) + TEST_ENTRY(glm_vec4_addsub) + TEST_ENTRY(glm_vec4_mulsub) + TEST_ENTRY(glm_vec4_mulsubs) + TEST_ENTRY(glm_vec4_maxsub) + TEST_ENTRY(glm_vec4_minsub) + TEST_ENTRY(glm_vec4_negate_to) + TEST_ENTRY(glm_vec4_negate) + TEST_ENTRY(glm_vec4_normalize) + TEST_ENTRY(glm_vec4_normalize_to) + TEST_ENTRY(glm_vec4_distance2) + TEST_ENTRY(glm_vec4_distance) + TEST_ENTRY(glm_vec4_maxv) + TEST_ENTRY(glm_vec4_minv) + TEST_ENTRY(glm_vec4_clamp) + TEST_ENTRY(glm_vec4_lerp) + TEST_ENTRY(glm_vec4_lerpc) + TEST_ENTRY(glm_vec4_mix) + TEST_ENTRY(glm_vec4_mixc) + TEST_ENTRY(glm_vec4_step) + TEST_ENTRY(glm_vec4_smoothstep_uni) + TEST_ENTRY(glm_vec4_smoothstep) + TEST_ENTRY(glm_vec4_smoothinterp) + TEST_ENTRY(glm_vec4_smoothinterpc) + TEST_ENTRY(glm_vec4_cubic) + TEST_ENTRY(glm_vec4_swizzle) + TEST_ENTRY(glm_vec4_broadcast) + TEST_ENTRY(glm_vec4_fill) + TEST_ENTRY(glm_vec4_eq) + TEST_ENTRY(glm_vec4_eq_eps) + TEST_ENTRY(glm_vec4_eq_all) + TEST_ENTRY(glm_vec4_eqv) + TEST_ENTRY(glm_vec4_eqv_eps) + TEST_ENTRY(glm_vec4_max) + TEST_ENTRY(glm_vec4_min) + TEST_ENTRY(glm_vec4_isnan) + TEST_ENTRY(glm_vec4_isinf) + TEST_ENTRY(glm_vec4_isvalid) + TEST_ENTRY(glm_vec4_sign) + TEST_ENTRY(glm_vec4_abs) + TEST_ENTRY(glm_vec4_fract) + TEST_ENTRY(glm_vec4_floor) + TEST_ENTRY(glm_vec4_mods) + TEST_ENTRY(glm_vec4_steps) + TEST_ENTRY(glm_vec4_stepr) + TEST_ENTRY(glm_vec4_hadd) + TEST_ENTRY(glm_vec4_sqrt) + TEST_ENTRY(glm_vec4_make) + TEST_ENTRY(glm_vec4_reflect) + TEST_ENTRY(glm_vec4_refract) + + TEST_ENTRY(glmc_vec4) + TEST_ENTRY(glmc_vec4_copy3) + TEST_ENTRY(glmc_vec4_copy) + TEST_ENTRY(glmc_vec4_ucopy) + TEST_ENTRY(glmc_vec4_zero) + TEST_ENTRY(glmc_vec4_one) + TEST_ENTRY(glmc_vec4_dot) + TEST_ENTRY(glmc_vec4_norm2) + TEST_ENTRY(glmc_vec4_norm) + TEST_ENTRY(glmc_vec4_norm_one) + TEST_ENTRY(glmc_vec4_norm_inf) + TEST_ENTRY(glmc_vec4_add) + TEST_ENTRY(glmc_vec4_adds) + TEST_ENTRY(glmc_vec4_sub) + TEST_ENTRY(glmc_vec4_subs) + TEST_ENTRY(glmc_vec4_mul) + TEST_ENTRY(glmc_vec4_scale) + TEST_ENTRY(glmc_vec4_scale_as) + TEST_ENTRY(glmc_vec4_div) + TEST_ENTRY(glmc_vec4_divs) + TEST_ENTRY(glmc_vec4_addadd) + TEST_ENTRY(glmc_vec4_subadd) + TEST_ENTRY(glmc_vec4_muladd) + TEST_ENTRY(glmc_vec4_muladds) + TEST_ENTRY(glmc_vec4_maxadd) + TEST_ENTRY(glmc_vec4_minadd) + TEST_ENTRY(glmc_vec4_subsub) + TEST_ENTRY(glmc_vec4_addsub) + TEST_ENTRY(glmc_vec4_mulsub) + TEST_ENTRY(glmc_vec4_mulsubs) + TEST_ENTRY(glmc_vec4_maxsub) + TEST_ENTRY(glmc_vec4_minsub) + TEST_ENTRY(glmc_vec4_negate_to) + TEST_ENTRY(glmc_vec4_negate) + TEST_ENTRY(glmc_vec4_normalize) + TEST_ENTRY(glmc_vec4_normalize_to) + TEST_ENTRY(glmc_vec4_distance2) + TEST_ENTRY(glmc_vec4_distance) + TEST_ENTRY(glmc_vec4_maxv) + TEST_ENTRY(glmc_vec4_minv) + TEST_ENTRY(glmc_vec4_clamp) + TEST_ENTRY(glmc_vec4_lerp) + TEST_ENTRY(glmc_vec4_lerpc) + TEST_ENTRY(glmc_vec4_mix) + TEST_ENTRY(glmc_vec4_mixc) + TEST_ENTRY(glmc_vec4_step) + TEST_ENTRY(glmc_vec4_smoothstep_uni) + TEST_ENTRY(glmc_vec4_smoothstep) + TEST_ENTRY(glmc_vec4_smoothinterp) + TEST_ENTRY(glmc_vec4_smoothinterpc) + TEST_ENTRY(glmc_vec4_cubic) + TEST_ENTRY(glmc_vec4_swizzle) + TEST_ENTRY(glmc_vec4_broadcast) + TEST_ENTRY(glmc_vec4_fill) + TEST_ENTRY(glmc_vec4_eq) + TEST_ENTRY(glmc_vec4_eq_eps) + TEST_ENTRY(glmc_vec4_eq_all) + TEST_ENTRY(glmc_vec4_eqv) + TEST_ENTRY(glmc_vec4_eqv_eps) + TEST_ENTRY(glmc_vec4_max) + TEST_ENTRY(glmc_vec4_min) + TEST_ENTRY(glmc_vec4_isnan) + TEST_ENTRY(glmc_vec4_isinf) + TEST_ENTRY(glmc_vec4_isvalid) + TEST_ENTRY(glmc_vec4_sign) + TEST_ENTRY(glmc_vec4_abs) + TEST_ENTRY(glmc_vec4_fract) + TEST_ENTRY(glmc_vec4_floor) + TEST_ENTRY(glmc_vec4_mods) + TEST_ENTRY(glmc_vec4_steps) + TEST_ENTRY(glmc_vec4_stepr) + TEST_ENTRY(glmc_vec4_hadd) + TEST_ENTRY(glmc_vec4_sqrt) + TEST_ENTRY(glmc_vec4_make) + TEST_ENTRY(glmc_vec4_reflect) + TEST_ENTRY(glmc_vec4_refract) + + /* ivec2 */ + TEST_ENTRY(glm_ivec2) + TEST_ENTRY(glm_ivec2_copy) + TEST_ENTRY(glm_ivec2_zero) + TEST_ENTRY(glm_ivec2_one) + TEST_ENTRY(glm_ivec2_dot) + TEST_ENTRY(glm_ivec2_cross) + TEST_ENTRY(glm_ivec2_add) + TEST_ENTRY(glm_ivec2_adds) + TEST_ENTRY(glm_ivec2_sub) + TEST_ENTRY(glm_ivec2_subs) + TEST_ENTRY(glm_ivec2_mul) + TEST_ENTRY(glm_ivec2_scale) + TEST_ENTRY(glm_ivec2_div) + TEST_ENTRY(glm_ivec2_divs) + TEST_ENTRY(glm_ivec2_mod) + TEST_ENTRY(glm_ivec2_addadd) + TEST_ENTRY(glm_ivec2_addadds) + TEST_ENTRY(glm_ivec2_subadd) + TEST_ENTRY(glm_ivec2_subadds) + TEST_ENTRY(glm_ivec2_muladd) + TEST_ENTRY(glm_ivec2_muladds) + TEST_ENTRY(glm_ivec2_maxadd) + TEST_ENTRY(glm_ivec2_minadd) + TEST_ENTRY(glm_ivec2_subsub) + TEST_ENTRY(glm_ivec2_subsubs) + TEST_ENTRY(glm_ivec2_addsub) + TEST_ENTRY(glm_ivec2_addsubs) + TEST_ENTRY(glm_ivec2_mulsub) + TEST_ENTRY(glm_ivec2_mulsubs) + TEST_ENTRY(glm_ivec2_maxsub) + TEST_ENTRY(glm_ivec2_minsub) + TEST_ENTRY(glm_ivec2_distance2) + TEST_ENTRY(glm_ivec2_distance) + TEST_ENTRY(glm_ivec2_fill) + TEST_ENTRY(glm_ivec2_eq) + TEST_ENTRY(glm_ivec2_eqv) + TEST_ENTRY(glm_ivec2_maxv) + TEST_ENTRY(glm_ivec2_minv) + TEST_ENTRY(glm_ivec2_clamp) + TEST_ENTRY(glm_ivec2_abs) + + TEST_ENTRY(glmc_ivec2) + TEST_ENTRY(glmc_ivec2_copy) + TEST_ENTRY(glmc_ivec2_zero) + TEST_ENTRY(glmc_ivec2_one) + TEST_ENTRY(glmc_ivec2_dot) + TEST_ENTRY(glmc_ivec2_cross) + TEST_ENTRY(glmc_ivec2_add) + TEST_ENTRY(glmc_ivec2_adds) + TEST_ENTRY(glmc_ivec2_sub) + TEST_ENTRY(glmc_ivec2_subs) + TEST_ENTRY(glmc_ivec2_mul) + TEST_ENTRY(glmc_ivec2_scale) + TEST_ENTRY(glmc_ivec2_div) + TEST_ENTRY(glmc_ivec2_divs) + TEST_ENTRY(glmc_ivec2_mod) + TEST_ENTRY(glmc_ivec2_addadd) + TEST_ENTRY(glmc_ivec2_addadds) + TEST_ENTRY(glmc_ivec2_subadd) + TEST_ENTRY(glmc_ivec2_subadds) + TEST_ENTRY(glmc_ivec2_muladd) + TEST_ENTRY(glmc_ivec2_muladds) + TEST_ENTRY(glmc_ivec2_maxadd) + TEST_ENTRY(glmc_ivec2_minadd) + TEST_ENTRY(glmc_ivec2_subsub) + TEST_ENTRY(glmc_ivec2_subsubs) + TEST_ENTRY(glmc_ivec2_addsub) + TEST_ENTRY(glmc_ivec2_addsubs) + TEST_ENTRY(glmc_ivec2_mulsub) + TEST_ENTRY(glmc_ivec2_mulsubs) + TEST_ENTRY(glmc_ivec2_maxsub) + TEST_ENTRY(glmc_ivec2_minsub) + TEST_ENTRY(glmc_ivec2_distance2) + TEST_ENTRY(glmc_ivec2_distance) + TEST_ENTRY(glmc_ivec2_fill) + TEST_ENTRY(glmc_ivec2_eq) + TEST_ENTRY(glmc_ivec2_eqv) + TEST_ENTRY(glmc_ivec2_maxv) + TEST_ENTRY(glmc_ivec2_minv) + TEST_ENTRY(glmc_ivec2_clamp) + TEST_ENTRY(glmc_ivec2_abs) + + /* ivec3 */ + TEST_ENTRY(glm_ivec3) + TEST_ENTRY(glm_ivec3_copy) + TEST_ENTRY(glm_ivec3_zero) + TEST_ENTRY(glm_ivec3_one) + TEST_ENTRY(glm_ivec3_dot) + TEST_ENTRY(glm_ivec3_norm2) + TEST_ENTRY(glm_ivec3_norm) + TEST_ENTRY(glm_ivec3_add) + TEST_ENTRY(glm_ivec3_adds) + TEST_ENTRY(glm_ivec3_sub) + TEST_ENTRY(glm_ivec3_subs) + TEST_ENTRY(glm_ivec3_mul) + TEST_ENTRY(glm_ivec3_scale) + TEST_ENTRY(glm_ivec3_div) + TEST_ENTRY(glm_ivec3_divs) + TEST_ENTRY(glm_ivec3_mod) + TEST_ENTRY(glm_ivec3_addadd) + TEST_ENTRY(glm_ivec3_addadds) + TEST_ENTRY(glm_ivec3_subadd) + TEST_ENTRY(glm_ivec3_subadds) + TEST_ENTRY(glm_ivec3_muladd) + TEST_ENTRY(glm_ivec3_muladds) + TEST_ENTRY(glm_ivec3_maxadd) + TEST_ENTRY(glm_ivec3_minadd) + TEST_ENTRY(glm_ivec3_subsub) + TEST_ENTRY(glm_ivec3_subsubs) + TEST_ENTRY(glm_ivec3_addsub) + TEST_ENTRY(glm_ivec3_addsubs) + TEST_ENTRY(glm_ivec3_mulsub) + TEST_ENTRY(glm_ivec3_mulsubs) + TEST_ENTRY(glm_ivec3_maxsub) + TEST_ENTRY(glm_ivec3_minsub) + TEST_ENTRY(glm_ivec3_distance2) + TEST_ENTRY(glm_ivec3_distance) + TEST_ENTRY(glm_ivec3_fill) + TEST_ENTRY(glm_ivec3_eq) + TEST_ENTRY(glm_ivec3_eqv) + TEST_ENTRY(glm_ivec3_maxv) + TEST_ENTRY(glm_ivec3_minv) + TEST_ENTRY(glm_ivec3_clamp) + + TEST_ENTRY(glmc_ivec3) + TEST_ENTRY(glmc_ivec3_copy) + TEST_ENTRY(glmc_ivec3_zero) + TEST_ENTRY(glmc_ivec3_one) + TEST_ENTRY(glmc_ivec3_dot) + TEST_ENTRY(glmc_ivec3_norm2) + TEST_ENTRY(glmc_ivec3_norm) + TEST_ENTRY(glmc_ivec3_add) + TEST_ENTRY(glmc_ivec3_adds) + TEST_ENTRY(glmc_ivec3_sub) + TEST_ENTRY(glmc_ivec3_subs) + TEST_ENTRY(glmc_ivec3_mul) + TEST_ENTRY(glmc_ivec3_scale) + TEST_ENTRY(glmc_ivec3_div) + TEST_ENTRY(glmc_ivec3_divs) + TEST_ENTRY(glmc_ivec3_mod) + TEST_ENTRY(glmc_ivec3_addadd) + TEST_ENTRY(glmc_ivec3_addadds) + TEST_ENTRY(glmc_ivec3_subadd) + TEST_ENTRY(glmc_ivec3_subadds) + TEST_ENTRY(glmc_ivec3_muladd) + TEST_ENTRY(glmc_ivec3_muladds) + TEST_ENTRY(glmc_ivec3_maxadd) + TEST_ENTRY(glmc_ivec3_minadd) + TEST_ENTRY(glmc_ivec3_subsub) + TEST_ENTRY(glmc_ivec3_subsubs) + TEST_ENTRY(glmc_ivec3_addsub) + TEST_ENTRY(glmc_ivec3_addsubs) + TEST_ENTRY(glmc_ivec3_mulsub) + TEST_ENTRY(glmc_ivec3_mulsubs) + TEST_ENTRY(glmc_ivec3_maxsub) + TEST_ENTRY(glmc_ivec3_minsub) + TEST_ENTRY(glmc_ivec3_distance2) + TEST_ENTRY(glmc_ivec3_distance) + TEST_ENTRY(glmc_ivec3_fill) + TEST_ENTRY(glmc_ivec3_eq) + TEST_ENTRY(glmc_ivec3_eqv) + TEST_ENTRY(glmc_ivec3_maxv) + TEST_ENTRY(glmc_ivec3_minv) + TEST_ENTRY(glmc_ivec3_clamp) + + /* ivec4 */ + TEST_ENTRY(glm_ivec4) + TEST_ENTRY(glm_ivec4_copy) + TEST_ENTRY(glm_ivec4_zero) + TEST_ENTRY(glm_ivec4_one) + TEST_ENTRY(glm_ivec4_add) + TEST_ENTRY(glm_ivec4_adds) + TEST_ENTRY(glm_ivec4_sub) + TEST_ENTRY(glm_ivec4_subs) + TEST_ENTRY(glm_ivec4_mul) + TEST_ENTRY(glm_ivec4_scale) + TEST_ENTRY(glm_ivec4_addadd) + TEST_ENTRY(glm_ivec4_addadds) + TEST_ENTRY(glm_ivec4_subadd) + TEST_ENTRY(glm_ivec4_subadds) + TEST_ENTRY(glm_ivec4_muladd) + TEST_ENTRY(glm_ivec4_muladds) + TEST_ENTRY(glm_ivec4_maxadd) + TEST_ENTRY(glm_ivec4_minadd) + TEST_ENTRY(glm_ivec4_subsub) + TEST_ENTRY(glm_ivec4_subsubs) + TEST_ENTRY(glm_ivec4_addsub) + TEST_ENTRY(glm_ivec4_addsubs) + TEST_ENTRY(glm_ivec4_mulsub) + TEST_ENTRY(glm_ivec4_mulsubs) + TEST_ENTRY(glm_ivec4_maxsub) + TEST_ENTRY(glm_ivec4_minsub) + TEST_ENTRY(glm_ivec4_distance2) + TEST_ENTRY(glm_ivec4_distance) + TEST_ENTRY(glm_ivec4_maxv) + TEST_ENTRY(glm_ivec4_minv) + TEST_ENTRY(glm_ivec4_clamp) + + TEST_ENTRY(glmc_ivec4) + TEST_ENTRY(glmc_ivec4_copy) + TEST_ENTRY(glmc_ivec4_zero) + TEST_ENTRY(glmc_ivec4_one) + TEST_ENTRY(glmc_ivec4_add) + TEST_ENTRY(glmc_ivec4_adds) + TEST_ENTRY(glmc_ivec4_sub) + TEST_ENTRY(glmc_ivec4_subs) + TEST_ENTRY(glmc_ivec4_mul) + TEST_ENTRY(glmc_ivec4_scale) + TEST_ENTRY(glmc_ivec4_addadd) + TEST_ENTRY(glmc_ivec4_addadds) + TEST_ENTRY(glmc_ivec4_subadd) + TEST_ENTRY(glmc_ivec4_subadds) + TEST_ENTRY(glmc_ivec4_muladd) + TEST_ENTRY(glmc_ivec4_muladds) + TEST_ENTRY(glmc_ivec4_maxadd) + TEST_ENTRY(glmc_ivec4_minadd) + TEST_ENTRY(glmc_ivec4_subsub) + TEST_ENTRY(glmc_ivec4_subsubs) + TEST_ENTRY(glmc_ivec4_addsub) + TEST_ENTRY(glmc_ivec4_addsubs) + TEST_ENTRY(glmc_ivec4_mulsub) + TEST_ENTRY(glmc_ivec4_mulsubs) + TEST_ENTRY(glmc_ivec4_maxsub) + TEST_ENTRY(glmc_ivec4_minsub) + TEST_ENTRY(glmc_ivec4_distance2) + TEST_ENTRY(glmc_ivec4_distance) + TEST_ENTRY(glmc_ivec4_maxv) + TEST_ENTRY(glmc_ivec4_minv) + TEST_ENTRY(glmc_ivec4_clamp) + TEST_ENTRY(glmc_ivec4_abs) + + /* structs */ + TEST_ENTRY(mat2x3s_zero_init) + TEST_ENTRY(mat2x3s_zero) + TEST_ENTRY(mat2x4s_zero_init) + TEST_ENTRY(mat2x4s_zero) + TEST_ENTRY(mat3s_identity_init) + TEST_ENTRY(mat3s_zero_init) + TEST_ENTRY(mat3x2s_zero_init) + TEST_ENTRY(mat3x2s_zero) + TEST_ENTRY(mat3x4s_zero_init) + TEST_ENTRY(mat3x4s_zero) + TEST_ENTRY(mat4s_identity_init) + TEST_ENTRY(mat4s_zero_init) + TEST_ENTRY(mat4x2s_zero_init) + TEST_ENTRY(mat4x2s_zero) + TEST_ENTRY(mat4x3s_zero_init) + TEST_ENTRY(mat4x3s_zero) + TEST_ENTRY(quats_zero_init) + TEST_ENTRY(vec3s_one_init) + TEST_ENTRY(vec3s_zero_init) + TEST_ENTRY(vec4s_black_init) + TEST_ENTRY(vec4s_one_init) + TEST_ENTRY(vec4s_zero_init) +}; + +#endif /* tests_h */ diff --git a/cglm/win/.gitignore b/cglm/win/.gitignore new file mode 100644 index 0000000..96c0a51 --- /dev/null +++ b/cglm/win/.gitignore @@ -0,0 +1,9 @@ +!cglm.sln + +!cglm.vcxproj +!cglm.vcxproj.filters + +!cglm-test.vcxproj +!cglm-test.vcxproj.filters + +!packages.config diff --git a/cglm/win/build.bat b/cglm/win/build.bat new file mode 100755 index 0000000..a2f0bf0 --- /dev/null +++ b/cglm/win/build.bat @@ -0,0 +1 @@ +msbuild cglm.vcxproj /p:Configuration=Release diff --git a/cglm/win/cglm-test.vcxproj b/cglm/win/cglm-test.vcxproj new file mode 100644 index 0000000..b9e1cf3 --- /dev/null +++ b/cglm/win/cglm-test.vcxproj @@ -0,0 +1,432 @@ + + + + + Debug + ARM + + + Debug + ARM64 + + + Debug + ARM64EC + + + Debug + Win32 + + + Release + ARM + + + Release + ARM64 + + + Release + ARM64EC + + + Release + Win32 + + + Debug + x64 + + + Release + x64 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + {ca8bcaf9-cd25-4133-8f62-3d1449b5d2fc} + + + + 16.0 + {200E0DF1-7532-44E6-8273-84FB92C5557E} + Win32Proj + cglmtest + 10.0 + + + + Application + true + v142 + Unicode + + + Application + false + v142 + true + Unicode + + + Application + true + v142 + Unicode + + + Application + true + v142 + Unicode + + + Application + true + v142 + Unicode + + + Application + true + v142 + Unicode + + + Application + false + v142 + true + Unicode + + + Application + false + v142 + true + Unicode + + + Application + false + v142 + true + Unicode + + + Application + false + v142 + true + Unicode + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + false + + + false + + + false + + + false + + + true + + + true + + + true + + + true + + + true + + + false + + + + + + Level3 + MaxSpeed + true + true + true + NDEBUG;_CONSOLE;%(PreprocessorDefinitions) + true + ../include;%(AdditionalIncludeDirectories) + + + Console + true + true + true + %(AdditionalDependencies) + + + + + + + Level3 + MaxSpeed + true + true + true + NDEBUG;_CONSOLE;%(PreprocessorDefinitions) + true + ../include;%(AdditionalIncludeDirectories) + + + Console + true + true + true + %(AdditionalDependencies) + + + + + + + Level3 + MaxSpeed + true + true + true + NDEBUG;_CONSOLE;%(PreprocessorDefinitions) + true + ../include;%(AdditionalIncludeDirectories) + + + Console + true + true + true + %(AdditionalDependencies) + + + + + + + Level3 + MaxSpeed + true + true + true + NDEBUG;_CONSOLE;%(PreprocessorDefinitions) + true + ../include;%(AdditionalIncludeDirectories) + + + Console + true + true + true + %(AdditionalDependencies) + + + + + + + Level3 + Disabled + true + WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions) + true + ../include;%(AdditionalIncludeDirectories) + Default + + + Console + true + %(AdditionalDependencies) + + + + + + + Level3 + Disabled + true + _DEBUG;_CONSOLE;%(PreprocessorDefinitions) + true + ../include;%(AdditionalIncludeDirectories) + Default + + + Console + true + %(AdditionalDependencies) + + + + + + + Level3 + Disabled + true + _DEBUG;_CONSOLE;%(PreprocessorDefinitions) + true + ../include;%(AdditionalIncludeDirectories) + Default + + + Console + true + %(AdditionalDependencies) + + + + + + + Level3 + Disabled + true + _DEBUG;_CONSOLE;%(PreprocessorDefinitions) + true + ../include;%(AdditionalIncludeDirectories) + Default + + + Console + true + %(AdditionalDependencies) + + + + + + + Level3 + Disabled + true + _DEBUG;_CONSOLE;%(PreprocessorDefinitions) + true + ../include;%(AdditionalIncludeDirectories) + Default + + + Console + true + %(AdditionalDependencies) + + + + + + + Level3 + MaxSpeed + true + true + true + WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions) + true + ../include;%(AdditionalIncludeDirectories) + + + Console + true + true + true + %(AdditionalDependencies) + + + + + + diff --git a/cglm/win/cglm-test.vcxproj.filters b/cglm/win/cglm-test.vcxproj.filters new file mode 100644 index 0000000..a3c4634 --- /dev/null +++ b/cglm/win/cglm-test.vcxproj.filters @@ -0,0 +1,140 @@ + + + + + {4FC737F1-C7A5-4376-A066-2A32D752A2FF} + cpp;c;cc;cxx;def;odl;idl;hpj;bat;asm;asmx + + + {93995380-89BD-4b04-88EB-625FBE52EBFB} + h;hh;hpp;hxx;hm;inl;inc;ipp;xsd + + + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + + + include + + + include + + + include + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + \ No newline at end of file diff --git a/cglm/win/cglm.sln b/cglm/win/cglm.sln new file mode 100644 index 0000000..22f929b --- /dev/null +++ b/cglm/win/cglm.sln @@ -0,0 +1,71 @@ + +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio Version 16 +VisualStudioVersion = 16.0.29123.88 +MinimumVisualStudioVersion = 10.0.40219.1 +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "cglm", "cglm.vcxproj", "{CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}" +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "cglm-test", "cglm-test.vcxproj", "{200E0DF1-7532-44E6-8273-84FB92C5557E}" +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|ARM = Debug|ARM + Debug|ARM64 = Debug|ARM64 + Debug|ARM64EC = Debug|ARM64EC + Debug|x64 = Debug|x64 + Debug|x86 = Debug|x86 + Release|ARM = Release|ARM + Release|ARM64 = Release|ARM64 + Release|ARM64EC = Release|ARM64EC + Release|x64 = Release|x64 + Release|x86 = Release|x86 + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Debug|ARM.ActiveCfg = Debug|ARM + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Debug|ARM.Build.0 = Debug|ARM + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Debug|ARM64.ActiveCfg = Debug|ARM64 + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Debug|ARM64.Build.0 = Debug|ARM64 + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Debug|ARM64EC.ActiveCfg = Debug|ARM64EC + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Debug|ARM64EC.Build.0 = Debug|ARM64EC + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Debug|x64.ActiveCfg = Debug|x64 + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Debug|x64.Build.0 = Debug|x64 + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Debug|x86.ActiveCfg = Debug|Win32 + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Debug|x86.Build.0 = Debug|Win32 + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Release|ARM.ActiveCfg = Release|ARM + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Release|ARM.Build.0 = Release|ARM + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Release|ARM64.ActiveCfg = Release|ARM64 + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Release|ARM64.Build.0 = Release|ARM64 + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Release|ARM64EC.ActiveCfg = Release|ARM64EC + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Release|ARM64EC.Build.0 = Release|ARM64EC + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Release|x64.ActiveCfg = Release|x64 + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Release|x64.Build.0 = Release|x64 + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Release|x86.ActiveCfg = Release|Win32 + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC}.Release|x86.Build.0 = Release|Win32 + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Debug|ARM.ActiveCfg = Debug|ARM + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Debug|ARM.Build.0 = Debug|ARM + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Debug|ARM64.ActiveCfg = Debug|ARM64 + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Debug|ARM64.Build.0 = Debug|ARM64 + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Debug|ARM64EC.ActiveCfg = Debug|ARM64EC + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Debug|ARM64EC.Build.0 = Debug|ARM64EC + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Debug|x64.ActiveCfg = Debug|x64 + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Debug|x64.Build.0 = Debug|x64 + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Debug|x86.ActiveCfg = Debug|Win32 + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Debug|x86.Build.0 = Debug|Win32 + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Release|ARM.ActiveCfg = Release|ARM + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Release|ARM.Build.0 = Release|ARM + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Release|ARM64.ActiveCfg = Release|ARM64 + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Release|ARM64.Build.0 = Release|ARM64 + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Release|ARM64EC.ActiveCfg = Release|ARM64EC + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Release|ARM64EC.Build.0 = Release|ARM64EC + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Release|x64.ActiveCfg = Release|x64 + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Release|x64.Build.0 = Release|x64 + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Release|x86.ActiveCfg = Release|Win32 + {200E0DF1-7532-44E6-8273-84FB92C5557E}.Release|x86.Build.0 = Release|Win32 + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection + GlobalSection(ExtensibilityGlobals) = postSolution + SolutionGuid = {2AEF23C9-433B-428B-BEBC-068BF3AC9A65} + EndGlobalSection +EndGlobal diff --git a/cglm/win/cglm.vcxproj b/cglm/win/cglm.vcxproj new file mode 100644 index 0000000..88574ca --- /dev/null +++ b/cglm/win/cglm.vcxproj @@ -0,0 +1,633 @@ + + + + + Debug + ARM + + + Debug + ARM64 + + + Debug + ARM64EC + + + Debug + Win32 + + + Release + ARM + + + Release + ARM64 + + + Release + ARM64EC + + + Release + Win32 + + + Debug + x64 + + + Release + x64 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 15.0 + {CA8BCAF9-CD25-4133-8F62-3D1449B5D2FC} + Win32Proj + cglm + 10.0 + + + + DynamicLibrary + true + v141 + Unicode + + + DynamicLibrary + false + v141 + true + Unicode + + + DynamicLibrary + true + v141 + Unicode + + + DynamicLibrary + true + v141 + Unicode + + + DynamicLibrary + true + v141 + Unicode + + + DynamicLibrary + true + v141 + Unicode + + + DynamicLibrary + false + v141 + true + Unicode + + + DynamicLibrary + false + v141 + true + Unicode + + + DynamicLibrary + false + v141 + true + Unicode + + + DynamicLibrary + false + v141 + true + Unicode + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + true + NativeRecommendedRules.ruleset + true + + + NativeRecommendedRules.ruleset + true + + + NativeRecommendedRules.ruleset + true + + + NativeRecommendedRules.ruleset + true + + + NativeRecommendedRules.ruleset + true + + + false + NativeRecommendedRules.ruleset + true + + + false + NativeRecommendedRules.ruleset + true + + + false + NativeRecommendedRules.ruleset + true + + + false + NativeRecommendedRules.ruleset + true + + + false + NativeRecommendedRules.ruleset + true + + + + NotUsing + Level3 + MaxSpeed + WIN32;_DEBUG;_WINDOWS;_USRDLL;CGLM_EXPORTS;%(PreprocessorDefinitions) + None + Default + + CompileAsC + true + true + + + Windows + + + + + Level3 + MaxSpeed + _DEBUG;_WINDOWS;_USRDLL;CGLM_EXPORTS;%(PreprocessorDefinitions) + AnySuitable + true + true + + CompileAsC + None + Default + true + + + Windows + + + + + Level3 + MaxSpeed + _DEBUG;_WINDOWS;_USRDLL;CGLM_EXPORTS;%(PreprocessorDefinitions) + AnySuitable + true + true + + + CompileAsC + None + Default + true + + + Windows + + + + + Level3 + MaxSpeed + _DEBUG;_WINDOWS;_USRDLL;CGLM_EXPORTS;%(PreprocessorDefinitions) + AnySuitable + true + true + + + CompileAsC + None + Default + true + + + Windows + + + + + Level3 + MaxSpeed + _DEBUG;_WINDOWS;_USRDLL;CGLM_EXPORTS;%(PreprocessorDefinitions) + AnySuitable + true + true + + + CompileAsC + None + Default + true + + + Windows + + + + + Level3 + NotUsing + MaxSpeed + true + true + WIN32;NDEBUG;_WINDOWS;_USRDLL;CGLM_EXPORTS;%(PreprocessorDefinitions) + None + + CompileAsC + true + Default + + + Windows + true + true + + + + + Level3 + NotUsing + MaxSpeed + true + true + NDEBUG;_WINDOWS;_USRDLL;CGLM_EXPORTS;%(PreprocessorDefinitions) + None + + CompileAsC + true + Default + + + Windows + true + true + + + + + Level3 + NotUsing + MaxSpeed + true + true + NDEBUG;_WINDOWS;_USRDLL;CGLM_EXPORTS;%(PreprocessorDefinitions) + None + + + CompileAsC + true + Default + + + Windows + true + true + + + + + Level3 + NotUsing + MaxSpeed + true + true + NDEBUG;_WINDOWS;_USRDLL;CGLM_EXPORTS;%(PreprocessorDefinitions) + None + + + CompileAsC + true + Default + + + Windows + true + true + + + + + Level3 + NotUsing + MaxSpeed + true + true + NDEBUG;_WINDOWS;_USRDLL;CGLM_EXPORTS;%(PreprocessorDefinitions) + None + + + CompileAsC + true + Default + + + Windows + true + true + + + + + + diff --git a/cglm/win/cglm.vcxproj.filters b/cglm/win/cglm.vcxproj.filters new file mode 100644 index 0000000..b14df86 --- /dev/null +++ b/cglm/win/cglm.vcxproj.filters @@ -0,0 +1,748 @@ + + + + + {4FC737F1-C7A5-4376-A066-2A32D752A2FF} + cpp;c;cc;cxx;def;odl;idl;hpj;bat;asm;asmx + + + {f46634e5-c098-41a0-afb3-45afb6a53593} + + + {b982cbd3-03d9-4557-bed8-8224e4d95945} + + + {6a2a4ebb-ac68-4ed6-a4a1-f6d2243814e9} + + + {4bcb64d3-e522-48f7-85ef-3ddfacf50874} + + + {ccf6244c-614c-4016-a5d9-8a78ffc7765a} + + + {8607b04c-6eb6-49af-8ac2-e534ba78bf76} + + + {fb97f276-fe14-47ba-8a9f-01032f065a19} + + + {0b5febe7-a88d-4330-94ae-305897a5e957} + + + {8044a657-123d-4a0e-8de1-f348fad7dadb} + + + {842a48fc-5c7e-4951-9623-64af96dfb95a} + + + {86101de1-0722-4c88-92a0-e6d71158354e} + + + {98a166bb-ba2d-4649-a20f-ba6bf4ce6383} + + + {24571788-2e78-4969-b66f-065aea5489be} + + + {e6a410e7-83db-41a5-b9fc-34bbfcf46ae5} + + + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src + + + src\clipspace + + + src\clipspace + + + src\clipspace + + + src\clipspace + + + src\clipspace + + + src\clipspace + + + src\clipspace + + + src\clipspace + + + src\clipspace + + + src\clipspace + + + src\clipspace + + + src\clipspace + + + src\clipspace + + + src\clipspace + + + src + + + src + + + src + + + src + + + src + + + + + src + + + include\cglm\call + + + include\cglm\call + + + include\cglm\call + + + include\cglm\call + + + include\cglm\call + + + include\cglm\call + + + include\cglm\call + + + include\cglm\call + + + include\cglm\call + + + include\cglm\call + + + include\cglm\call + + + include\cglm\call + + + include\cglm\call + + + include\cglm\call + + + include\cglm\simd\avx + + + include\cglm\simd\avx + + + include\cglm\simd\neon + + + include\cglm\simd\sse2 + + + include\cglm\simd\sse2 + + + include\cglm\simd\sse2 + + + include\cglm\simd\sse2 + + + include\cglm\simd + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm\call + + + include\cglm\call + + + include\cglm + + + include\cglm + + + include\cglm\call + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm\call + + + include\cglm\call + + + include\cglm + + + include\cglm\call + + + include\cglm + + + include\cglm + + + include\cglm\simd + + + include\cglm\simd + + + include\cglm\call + + + include\cglm + + + include\cglm + + + include\cglm\call + + + include\cglm + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm + + + include\cglm + + + include\cglm\call + + + include\cglm\call + + + include\cglm\call + + + include\cglm\call + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\simd\sse2 + + + include\cglm\call + + + include\cglm + + + include\cglm\struct + + + include\cglm\simd\neon + + + include\cglm\simd\neon + + + include\cglm\simd\neon + + + include\cglm\clipspace + + + include\cglm\clipspace + + + include\cglm\clipspace + + + include\cglm\clipspace + + + include\cglm\clipspace + + + include\cglm\clipspace + + + include\cglm\clipspace + + + include\cglm\clipspace + + + include\cglm\clipspace + + + include\cglm\clipspace + + + include\cglm\clipspace + + + include\cglm\clipspace + + + include\cglm\clipspace + + + include\cglm\clipspace + + + include\cglm\clipspace + + + include\cglm\call\clipspace + + + include\cglm\call\clipspace + + + include\cglm\call\clipspace + + + include\cglm\call\clipspace + + + include\cglm\call\clipspace + + + include\cglm\call\clipspace + + + include\cglm\call\clipspace + + + include\cglm\call\clipspace + + + include\cglm\call\clipspace + + + include\cglm\call\clipspace + + + include\cglm\call\clipspace + + + include\cglm\call\clipspace + + + include\cglm\struct\clipspace + + + include\cglm\struct\clipspace + + + include\cglm\struct\clipspace + + + include\cglm\struct\clipspace + + + include\cglm\struct\clipspace + + + include\cglm\struct\clipspace + + + include\cglm\struct\clipspace + + + include\cglm\struct\clipspace + + + include\cglm\struct\clipspace + + + include\cglm\struct\clipspace + + + include\cglm\struct\clipspace + + + include\cglm\struct\clipspace + + + include\cglm\clipspace + + + include\cglm\clipspace + + + include\cglm\call\clipspace + + + include\cglm\call\clipspace + + + include\cglm\call + + + include\cglm\call + + + include\cglm\call + + + include\cglm\struct\clipspace + + + include\cglm\struct\clipspace + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm + + + include\cglm\call + + + include\cglm\simd\wasm + + + include\cglm\simd\wasm + + + include\cglm\simd\wasm + + + include\cglm\simd\wasm + + + include\cglm\simd\wasm + + + include\cglm\simd + + + include\cglm\struct + + + include\cglm\struct\handed + + + include\cglm\struct\handed + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm\struct + + + include\cglm + + + include\cglm\call + + + include\cglm\struct + + + \ No newline at end of file diff --git a/main.c b/main.c new file mode 100644 index 0000000..619597e --- /dev/null +++ b/main.c @@ -0,0 +1,1310 @@ +// vim:fileencoding=utf-8:foldmethod=marker + +#include "volk/volk.h" +#include "vulkan/vulkan_core.h" +#include "wapp/wapp.h" +#include +#define CGLM_FORCE_DEPTH_ZERO_TO_ONE +#include "cglm/cglm.h" +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#define WINDOW_WIDTH 1600 +#define WINDOW_HEIGHT 1200 +#define FRAMES_IN_FLIGHT 3 +#define FRAME_TIME 1.0 / 60.0 * 1e+9 +#define SHADER_FILE "09_shader_base.spv" +#define SHADER_COUNT 2 +#define MIN_DEPTH 0.0f +#define MAX_DEPTH 1.0f + +typedef struct { + vec3 position; + vec3 color; +} Vertex; + +WAPP_DEF_ARRAY_TYPE(VkFence, VkFenceArray); +WAPP_DEF_ARRAY_TYPE(VkSemaphore, VkSemaphoreArray); +WAPP_DEF_ARRAY_TYPE(VkImage, VkImageArray); +WAPP_DEF_ARRAY_TYPE(VkImageView, VkImageViewArray); +WAPP_DEF_ARRAY_TYPE(Vertex, VertexArray); + +enum ReturnCodes { + RETURN_SUCCESS, + RETURN_FAILED_TO_GET_INSTANCE_EXTENSIONS, + RETURN_FAILED_TO_ALLOCATE_EXTENSION_NAMES, + RETURN_FAILED_TO_GET_VK_GET_INSTANCE_PROC_ADDR, + RETURN_FAILED_TO_INITIALISE_VOLK, + RETURN_FAILED_TO_QUERY_API_VERSION, + RETURN_FAILED_TO_CREATE_INSTANCE, + RETURN_FAILED_TO_CREATE_SURFACE, + RETURN_FAILED_TO_QUERY_PHYSICAL_DEVICE_COUNT, + RETURN_FAILED_TO_ALLOCATE_MEMORY_FOR_PHYSICAL_DEVICES, + RETURN_FAILED_TO_RETRIEVE_PHYSICAL_DEVICES, + RETURN_FAILED_TO_FIND_SUITABLE_PHYSICAL_DEVICE, + RETURN_NO_QUEUE_FAMILY_PROPERTIES_FOUND, + RETURN_FAILED_TO_ALLOCATE_MEMORY_FOR_QUEUE_FAMILY_PROPERTIES, + RETURN_SELECTED_DEVICE_DOES_NOT_SUPPORT_GRAPHICS, + RETURN_FAILED_TO_ENUMERATE_PHYSICAL_DEVICE_EXTENSIONS, + RETURN_FAILED_TO_QUERY_PHYSICAL_DEVICE_EXTENSIONS, + RETURN_FAILED_TO_CREATE_LOGICAL_DEVICE, + RETURN_FAILED_TO_CREATE_COMMAND_POOL, + RETURN_FAILED_TO_ALLOCATE_COMMAND_BUFFER, + RETURN_FAILED_TO_QUERY_SURFACE_CAPABILITIES, + RETURN_FAILED_TO_ENUMERATE_SUPPORTED_SURFACE_FORMATS, + RETURN_FAILED_TO_ALLOCATE_MEMORY_FOR_SURFACE_FORMATS, + RETURN_FAILED_TO_QUERY_SURPPORTED_SURFACE_FORMATS, + RETURN_FAILED_TO_ENUMERATE_SUPPORTED_SURFACE_PRESENT_MODES, + RETURN_FAILED_TO_ALLOCATE_MEMORY_FOR_SURFACE_PRESENT_MODES, + RETURN_FAILED_TO_QUERY_SURPPORTED_SURFACE_PRESENT_MODES, + RETURN_FAILED_TO_CREATE_SWAPCHAIN, + RETURN_FAILED_TO_GET_SWAPCHAIN_IMAGE_COUNT, + RETURN_FAILED_TO_ALLOCATE_SWAPCHAIN_IMAGES_MEMORY, + RETURN_FAILED_TO_GET_SWAPCHAIN_IMAGES, + RETURN_FAILED_TO_CREATE_IMAGE_VIEWS, + RETURN_FAILED_TO_CREATE_SYNC_OBJECTS, + RETURN_FAILED_TO_CREATE_SHADER_MODULE, + RETURN_FAILED_TO_CREATE_DEPTH_IMAGE, + RETURN_FAILED_TO_FIND_SUITABLE_MEMORY_TYPE_FOR_DEPTH_IMAGE, + RETURN_FAILED_TO_ALLOCATE_DEPTH_IMAGE_MEMORY, + RETURN_FAILED_TO_BIND_DEPTH_IMAGE_MEMORY, + RETURN_FAILED_TO_CREATE_DEPTH_IMAGE_VIEW, + RETURN_FAILED_TO_CREATE_COPY_COMMAND_BUFFER, + RETURN_FAILED_TO_CREATE_VERTEX_BUFFER, + RETURN_FAILED_TO_CREATE_INDEX_BUFFER, + RETURN_FAILED_TO_COPY_VERTEX_AND_INDEX_DATA, + RETURN_FAILED_TO_CREATE_PIPELINE_LAYOUT, + RETURN_FAILED_TO_CREATE_GRAPHICS_PIPELINE, +}; + +VkInstance instance = VK_NULL_HANDLE; +VkPhysicalDevice physical_device = VK_NULL_HANDLE; +u32 queue_family_index = VK_QUEUE_FAMILY_IGNORED; +VkDevice device = VK_NULL_HANDLE; +VkQueue queue = VK_NULL_HANDLE; +VkCommandPool command_pool = VK_NULL_HANDLE; +VkCommandBuffer command_buffers[FRAMES_IN_FLIGHT] = {0}; +VkSurfaceKHR surface = VK_NULL_HANDLE; +VkSwapchainKHR swapchain = VK_NULL_HANDLE; +VkSurfaceFormatKHR swapchain_format; +VkPresentModeKHR swapchain_present_mode; +VkImageArray *swapchain_images; +VkImageViewArray *swapchain_image_views; +VkExtent2D swapchain_extent; +u32 image_index; +VkImage depth_image = VK_NULL_HANDLE; +VkDeviceMemory depth_image_memory = VK_NULL_HANDLE; +VkImageView depth_image_view = VK_NULL_HANDLE; +VkFormat depth_image_format; +VkBuffer vertex_buffer = VK_NULL_HANDLE; +VkDeviceMemory vertex_buffer_memory = VK_NULL_HANDLE; +VkBuffer index_buffer = VK_NULL_HANDLE; +VkDeviceMemory index_buffer_memory = VK_NULL_HANDLE; +VkFenceArray fences = wapp_array_with_capacity(VkFence, + VkFenceArray, + FRAMES_IN_FLIGHT, + true); +VkSemaphoreArray present_complete_semaphores = wapp_array_with_capacity(VkSemaphore, + VkSemaphoreArray, + FRAMES_IN_FLIGHT, + true); +VkSemaphoreArray render_finished_semaphores = wapp_array_with_capacity(VkSemaphore, + VkSemaphoreArray, + FRAMES_IN_FLIGHT, + true); +VkShaderModule shader_module = VK_NULL_HANDLE; +VkPipelineLayout pipeline_layout = VK_NULL_HANDLE; +VkPipeline graphics_pipeline = VK_NULL_HANDLE; +u32 physical_device_count; +u64 frame_index = 0; +VertexArray vertices = wapp_array(Vertex, VertexArray, + {{-0.5f, 0.5f, 0.5f}, {0.42f, 0.05f, 0.14f}}, + {{-0.5f, -0.5f, 0.5f}, {0.42f, 0.05f, 0.14f}}, + {{-0.5f, 0.5f, -0.5f}, { 0.0f, 0.1f, 0.21f}}, + {{-0.5f, -0.5f, -0.5f}, { 0.0f, 0.1f, 0.21f}}, + {{ 0.5f, 0.5f, 0.5f}, {0.42f, 0.05f, 0.14f}}, + {{ 0.5f, -0.5f, 0.5f}, {0.42f, 0.05f, 0.14f}}, + {{ 0.5f, 0.5f, -0.5f}, { 0.0f, 0.1f, 0.21f}}, + {{ 0.5f, -0.5f, -0.5f}, { 0.0f, 0.1f, 0.21f}}); +U32Array indices = wapp_array(u32, U32Array, 0, 2, 3, + 3, 1, 0, + 6, 4, 5, + 5, 7, 6, + 4, 0, 1, + 1, 5, 4, + 2, 6, 7, + 7, 3, 2, + 0, 4, 6, + 6, 2, 0, + 3, 7, 5, + 5, 1, 3); + +wapp_intern inline void draw_frame(void); +wapp_intern inline void record_command_buffer(void); +wapp_intern inline b8 create_buffer(VkPhysicalDeviceMemoryProperties *properties, VkDevice device, + VkBufferCreateFlags flags, VkDeviceSize size, VkBufferUsageFlags usage, + VkMemoryPropertyFlags memory_properties, VkBuffer *buffer, + VkDeviceMemory *buffer_memory); +wapp_intern inline void transition_image_layout( + VkCommandBuffer command_buffer, VkImage image, VkImageLayout old_layout, VkImageLayout new_layout, + VkPipelineStageFlags2 src_stage, VkPipelineStageFlags2 dst_stage, + VkAccessFlags2 src_access, VkAccessFlags2 dst_access, VkImageAspectFlags aspect_mask +); +wapp_intern inline i32 get_memory_type(const VkPhysicalDeviceMemoryProperties *properties, + u32 memory_type_bits, VkMemoryPropertyFlags flags); +wapp_intern inline u32 set_error(u32 code, const char *msg); + +int main(void) { + // Initialisation {{{ + Allocator arena = wapp_mem_arena_allocator_init(MB(256)); + u32 status = RETURN_SUCCESS; + + SDL_Init(SDL_INIT_VIDEO); + + SDL_Window *window = SDL_CreateWindow("Vulkan", WINDOW_WIDTH, WINDOW_HEIGHT, SDL_WINDOW_VULKAN); + + PFN_vkGetInstanceProcAddr vkGetInstanceProcAddr = (PFN_vkGetInstanceProcAddr)SDL_Vulkan_GetVkGetInstanceProcAddr(); + if (!vkGetInstanceProcAddr) { + status = set_error(RETURN_FAILED_TO_GET_VK_GET_INSTANCE_PROC_ADDR, + "Failed to get address of vkGetInstanceProcAddr"); + goto SDL_CLEANUP; + } + + if (volkInitialize() != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_INITIALISE_VOLK, "Failed to initialise volk"); + goto SDL_CLEANUP; + } + // }}} + + // Instance creation {{{ + u32 sdl_extensions_count; + const char *const *sdl_extensions = SDL_Vulkan_GetInstanceExtensions(&sdl_extensions_count); + if (!sdl_extensions) { + status = set_error(RETURN_FAILED_TO_GET_INSTANCE_EXTENSIONS, + "Failed to get instance extensions"); + goto SDL_CLEANUP; + } + + const char *extra_extensions[] = { + VK_KHR_SURFACE_EXTENSION_NAME, + VK_KHR_GET_SURFACE_CAPABILITIES_2_EXTENSION_NAME + }; + u32 extra_extensions_count = sizeof(extra_extensions) / sizeof(const char *); + + u32 all_extensions_count = sdl_extensions_count + extra_extensions_count; + const char **all_instance_extensions = wapp_mem_allocator_alloc(&arena, all_extensions_count * sizeof(const char *)); + if (!all_instance_extensions) { + status = set_error(RETURN_FAILED_TO_ALLOCATE_EXTENSION_NAMES, + "Failed to allocate memory for instance extensions names"); + goto SDL_CLEANUP; + } + + memcpy(all_instance_extensions, sdl_extensions, sdl_extensions_count * sizeof(const char *)); + memcpy(&(all_instance_extensions[sdl_extensions_count]), extra_extensions, extra_extensions_count * sizeof(const char *)); + + u32 api_version; + if (vkEnumerateInstanceVersion(&api_version) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_QUERY_API_VERSION, + "Failed to query API version"); + goto SDL_CLEANUP; + } + + fprintf(stdout, "Vulkan API version: %u.%u.%u.%u\n", VK_API_VERSION_MAJOR(api_version), + VK_API_VERSION_MINOR(api_version), + VK_API_VERSION_PATCH(api_version), + VK_API_VERSION_VARIANT(api_version)); + + VkApplicationInfo app_info = { + .sType = VK_STRUCTURE_TYPE_APPLICATION_INFO, + .pNext = NULL, + .pApplicationName = "Vulkan spec test", + .applicationVersion = VK_MAKE_VERSION(1, 0, 0), + .pEngineName = "My vulkan engine", + .engineVersion = VK_MAKE_VERSION(1, 0, 0), + .apiVersion = api_version, + }; + + VkInstanceCreateInfo instance_create_info = { + .sType = VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO, + .pApplicationInfo = &app_info, + .enabledExtensionCount = all_extensions_count, + .ppEnabledExtensionNames = all_instance_extensions, + }; + + if (vkCreateInstance(&instance_create_info, NULL, &instance) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_CREATE_INSTANCE, + "Failed to create vulkan instance"); + goto SDL_CLEANUP; + } + + volkLoadInstanceOnly(instance); + // }}} + + // Surface creation {{{ + if (!SDL_Vulkan_CreateSurface(window, instance, NULL, &surface)) { + status = set_error(RETURN_FAILED_TO_CREATE_SURFACE, + "Failed to create surface"); + goto DESTROY_VULKAN_INSTANCE; + } + // }}} + + // Find suitable physical device {{{ + if (vkEnumeratePhysicalDevices(instance, &physical_device_count, NULL) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_QUERY_PHYSICAL_DEVICE_COUNT, + "Failed to query physical device count"); + goto DESTROY_VULKAN_SURFACE; + } + + VkPhysicalDevice *physical_devices = wapp_mem_allocator_alloc(&arena, physical_device_count * sizeof(VkPhysicalDevice)); + if (!physical_devices) { + status = set_error(RETURN_FAILED_TO_ALLOCATE_MEMORY_FOR_PHYSICAL_DEVICES, + "Failed to allocate memory for physical devices"); + goto DESTROY_VULKAN_SURFACE; + } + + if (vkEnumeratePhysicalDevices(instance, &physical_device_count, physical_devices) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_RETRIEVE_PHYSICAL_DEVICES, + "Failed to retrieve physical devices"); + goto DESTROY_VULKAN_SURFACE; + } + + int32_t index = -1; + u32 api = 0; + for (int32_t i = 0; i < physical_device_count; ++i) { + VkPhysicalDeviceProperties2 base_properties = { .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PROPERTIES_2 }; + vkGetPhysicalDeviceProperties2(physical_devices[i], &base_properties); + switch (base_properties.properties.deviceType) { + case VK_PHYSICAL_DEVICE_TYPE_DISCRETE_GPU: { + if (base_properties.properties.apiVersion > api) { + api = base_properties.properties.apiVersion; + index = i; + } + } break; + case VK_PHYSICAL_DEVICE_TYPE_INTEGRATED_GPU: { + if (base_properties.properties.apiVersion > api) { + api = base_properties.properties.apiVersion; + index = i; + } + } break; + default: continue; + } + } + + if (index == -1 || api == 0) { + status = set_error(RETURN_FAILED_TO_FIND_SUITABLE_PHYSICAL_DEVICE, + "Couldn't find suitable physical device"); + goto DESTROY_VULKAN_SURFACE; + } + + physical_device = physical_devices[index]; + + VkPhysicalDeviceProperties2 physical_device_properties = { .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PROPERTIES_2 }; + VkPhysicalDeviceVulkan11Properties vulkan_11_properties = { .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_VULKAN_1_1_PROPERTIES }; + VkPhysicalDeviceVulkan12Properties vulkan_12_properties = { .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_VULKAN_1_2_PROPERTIES }; + VkPhysicalDeviceVulkan13Properties vulkan_13_properties = { .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_VULKAN_1_3_PROPERTIES }; + VkPhysicalDeviceVulkan14Properties vulkan_14_properties = { .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_VULKAN_1_4_PROPERTIES }; + + void *last = &physical_device_properties; + if (api >= VK_API_VERSION_1_1) { + last = ((VkPhysicalDeviceProperties2 *)last)->pNext = &vulkan_11_properties; + } + if (api >= VK_API_VERSION_1_2) { + last = ((VkPhysicalDeviceVulkan11Properties *)last)->pNext = &vulkan_12_properties; + } + if (api >= VK_API_VERSION_1_3) { + last = ((VkPhysicalDeviceVulkan12Properties *)last)->pNext = &vulkan_13_properties; + } + if (api >= VK_API_VERSION_1_4) { + last = ((VkPhysicalDeviceVulkan13Properties *)last)->pNext = &vulkan_14_properties; + } + + vkGetPhysicalDeviceProperties2(physical_device, &physical_device_properties); + + VkPhysicalDeviceMemoryProperties2 memory_properties = { + .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_MEMORY_PROPERTIES_2, + }; + vkGetPhysicalDeviceMemoryProperties2(physical_device, &memory_properties); + // }}} + + // Find suitable queue family {{{ + u32 queue_family_property_count = 0; + vkGetPhysicalDeviceQueueFamilyProperties2(physical_device, &queue_family_property_count, NULL); + if (queue_family_property_count == 0) { + status = set_error(RETURN_NO_QUEUE_FAMILY_PROPERTIES_FOUND, "No queue family properties found"); + goto DESTROY_VULKAN_SURFACE; + } + + u32 alloc_size = queue_family_property_count * sizeof(VkQueueFamilyProperties2); + VkQueueFamilyProperties2 *queue_family_properties = wapp_mem_allocator_alloc(&arena, alloc_size); + if (!queue_family_properties) { + status = set_error(RETURN_FAILED_TO_ALLOCATE_MEMORY_FOR_QUEUE_FAMILY_PROPERTIES, + "Failed to allocate memory for queue family properties"); + goto DESTROY_VULKAN_SURFACE; + } + + memset(queue_family_properties, 0, alloc_size); + for (u32 i = 0; i < queue_family_property_count; ++i) { + queue_family_properties[i].sType = VK_STRUCTURE_TYPE_QUEUE_FAMILY_PROPERTIES_2; + } + + vkGetPhysicalDeviceQueueFamilyProperties2(physical_device, &queue_family_property_count, queue_family_properties); + b8 supports_graphics = false; + for (u32 i = 0; i < queue_family_property_count; ++i) { + supports_graphics = supports_graphics || + (queue_family_properties[i].queueFamilyProperties.queueFlags & VK_QUEUE_GRAPHICS_BIT) == VK_QUEUE_GRAPHICS_BIT; + if (supports_graphics) { + queue_family_index = i; + break; + } + } + + if (!supports_graphics) { + status = set_error(RETURN_SELECTED_DEVICE_DOES_NOT_SUPPORT_GRAPHICS, + "Selected physical device doesn't support graphics operations"); + goto DESTROY_VULKAN_SURFACE; + } + // }}} + + // Logical device creation {{{ + VkPhysicalDeviceShaderObjectFeaturesEXT shader_object_feature = { + .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_SHADER_OBJECT_FEATURES_EXT, + .shaderObject = VK_TRUE, + }; + VkPhysicalDevicePresentModeFifoLatestReadyFeaturesKHR fifo_latest_ready_feature = { + .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PRESENT_MODE_FIFO_LATEST_READY_FEATURES_KHR, + .pNext = &shader_object_feature, + .presentModeFifoLatestReady = VK_TRUE, + }; + VkPhysicalDeviceVulkan14Features vulkan_14_features = { + .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_VULKAN_1_4_FEATURES, + .pNext = &fifo_latest_ready_feature, + .bresenhamLines = VK_TRUE, + .smoothLines = VK_TRUE, + .pushDescriptor = VK_TRUE, + }; + VkPhysicalDeviceVulkan13Features vulkan_13_features = { + .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_VULKAN_1_3_FEATURES, + .pNext = &vulkan_14_features, + .dynamicRendering = VK_TRUE, + .synchronization2 = VK_TRUE, + }; + VkPhysicalDeviceVulkan12Features vulkan_12_features = { + .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_VULKAN_1_2_FEATURES, + .pNext = &vulkan_13_features, + .timelineSemaphore = VK_TRUE, + }; + VkPhysicalDeviceVulkan11Features vulkan_11_features = { + .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_VULKAN_1_1_FEATURES, + .pNext = &vulkan_12_features, + .shaderDrawParameters = VK_TRUE, + }; + VkPhysicalDeviceFeatures core_features = { + .logicOp = VK_TRUE, + .fillModeNonSolid = VK_TRUE, + .samplerAnisotropy = VK_TRUE, + }; + VkPhysicalDeviceFeatures2 device_features = { + .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_FEATURES_2, + .pNext = &vulkan_11_features, + .features = core_features, + }; + f32 priorities[] = {0.5}; + VkDeviceQueueCreateInfo queue_create_info = { + .sType = VK_STRUCTURE_TYPE_DEVICE_QUEUE_CREATE_INFO, + .queueFamilyIndex = queue_family_index, + .queueCount = 1, + .pQueuePriorities = priorities, + }; + const char *device_extensions[] = { + VK_KHR_SWAPCHAIN_EXTENSION_NAME, + VK_EXT_SHADER_OBJECT_EXTENSION_NAME, + VK_EXT_VERTEX_INPUT_DYNAMIC_STATE_EXTENSION_NAME, + VK_EXT_PRESENT_MODE_FIFO_LATEST_READY_EXTENSION_NAME, + }; + u32 device_extensions_count = sizeof(device_extensions) / sizeof(const char *); + VkDeviceCreateInfo device_create_info = { + .sType = VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO, + .pNext = &device_features, + .queueCreateInfoCount = 1, + .pQueueCreateInfos = &queue_create_info, + .enabledExtensionCount = device_extensions_count, + .ppEnabledExtensionNames = device_extensions, + }; + + u32 dev_ext_count; + if (vkEnumerateDeviceExtensionProperties(physical_device, NULL, + &dev_ext_count, NULL) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_ENUMERATE_PHYSICAL_DEVICE_EXTENSIONS, + "Failed to enumerate physical device extensions"); + goto DESTROY_VULKAN_SURFACE; + } + VkExtensionProperties *extension_properties = wapp_mem_allocator_alloc(&arena, dev_ext_count * sizeof(VkExtensionProperties)); + if (vkEnumerateDeviceExtensionProperties(physical_device, NULL, + &dev_ext_count, extension_properties) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_QUERY_PHYSICAL_DEVICE_EXTENSIONS, + "Failed to query physical device extensions"); + goto DESTROY_VULKAN_SURFACE; + } + for (u32 i = 0; i < dev_ext_count; ++i) { + if (memcmp(extension_properties[i].extensionName, VK_KHR_PRESENT_MODE_FIFO_LATEST_READY_EXTENSION_NAME, + sizeof(VK_KHR_PRESENT_MODE_FIFO_LATEST_READY_EXTENSION_NAME)) == 0) { + device_extensions[device_extensions_count - 1] = VK_KHR_PRESENT_MODE_FIFO_LATEST_READY_EXTENSION_NAME; + break; + } + } + + if (vkCreateDevice(physical_device, &device_create_info, NULL, &device) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_CREATE_LOGICAL_DEVICE, + "Failed to create device"); + goto DESTROY_VULKAN_SURFACE; + } + + volkLoadDevice(device); + /// }}} + + // Get device queue {{{ + vkGetDeviceQueue(device, queue_family_index, 0, &queue); + // }}} + + // Create command pool {{{ + VkCommandPoolCreateInfo pool_create_info = { + .sType = VK_STRUCTURE_TYPE_COMMAND_POOL_CREATE_INFO, + .flags = VK_COMMAND_POOL_CREATE_RESET_COMMAND_BUFFER_BIT, + .queueFamilyIndex = queue_family_index, + }; + if (vkCreateCommandPool(device, &pool_create_info, NULL, &command_pool) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_CREATE_COMMAND_POOL, "Failed to create command pool"); + goto DESTROY_VULKAN_DEVICE; + } + // }}} + + // Allocate command buffers {{{ + VkCommandBufferAllocateInfo cmd_buf_alloc_info = { + .sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_ALLOCATE_INFO, + .commandBufferCount = 1, + .commandPool = command_pool, + .level = VK_COMMAND_BUFFER_LEVEL_PRIMARY, + }; + for (u32 i = 0; i < FRAMES_IN_FLIGHT; ++i) { + if (vkAllocateCommandBuffers(device, &cmd_buf_alloc_info, &command_buffers[i]) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_ALLOCATE_COMMAND_BUFFER, "Failed to allocate command buffer"); + goto FREE_COMMAND_BUFFER; + } + } + // }}} + + // Create swapchain {{{ + VkSurfaceCapabilitiesKHR capabilities = {0}; + if (vkGetPhysicalDeviceSurfaceCapabilitiesKHR(physical_device, surface, &capabilities) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_QUERY_SURFACE_CAPABILITIES, "Failed to query surface capabilities"); + goto FREE_COMMAND_BUFFER; + } + + swapchain_extent = capabilities.currentExtent; + + u32 format_count; + if (vkGetPhysicalDeviceSurfaceFormatsKHR(physical_device, surface, &format_count, NULL) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_ENUMERATE_SUPPORTED_SURFACE_FORMATS, + "Failed to enumerate supported surface formats"); + goto FREE_COMMAND_BUFFER; + } + VkSurfaceFormatKHR *formats = wapp_mem_allocator_alloc(&arena, format_count * sizeof(VkSurfaceFormatKHR)); + if (!formats) { + status = set_error(RETURN_FAILED_TO_ALLOCATE_MEMORY_FOR_SURFACE_FORMATS, + "Failed to allocate memory for surface formats"); + goto FREE_COMMAND_BUFFER; + } + if (vkGetPhysicalDeviceSurfaceFormatsKHR(physical_device, surface, &format_count, formats) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_QUERY_SURPPORTED_SURFACE_FORMATS, + "Failed to query supported surface formats"); + goto FREE_COMMAND_BUFFER; + } + + VkSurfaceFormatKHR *format = NULL; + for (u32 i = 0; i < format_count; ++i) { + if (formats[i].format == VK_FORMAT_B8G8R8A8_SRGB && formats[i].colorSpace == VK_COLORSPACE_SRGB_NONLINEAR_KHR) { + format = &(formats[i]); + break; + } + } + if (!format) { format = &formats[0]; } + swapchain_format = *format; + + u32 present_mode_count; + if (vkGetPhysicalDeviceSurfacePresentModesKHR(physical_device, surface, &present_mode_count, NULL) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_ENUMERATE_SUPPORTED_SURFACE_PRESENT_MODES, + "Failed to enumerate supported surface present modes"); + goto FREE_COMMAND_BUFFER; + } + VkPresentModeKHR *present_modes = wapp_mem_allocator_alloc(&arena, present_mode_count * sizeof(VkPresentModeKHR)); + if (!present_modes) { + status = set_error(RETURN_FAILED_TO_ALLOCATE_MEMORY_FOR_SURFACE_PRESENT_MODES, + "Failed to allocate memory for surface present modes"); + goto FREE_COMMAND_BUFFER; + } + swapchain_present_mode = VK_PRESENT_MODE_FIFO_KHR; + for (u32 i = 0; i < present_mode_count; ++i) { + if (present_modes[i] == VK_PRESENT_MODE_FIFO_LATEST_READY_KHR) { + swapchain_present_mode = present_modes[i]; + break; + } + } + + VkSwapchainCreateInfoKHR swapchain_info = { + .sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR, + .surface = surface, + .minImageCount = capabilities.minImageCount, + .imageFormat = swapchain_format.format, + .imageColorSpace = swapchain_format.colorSpace, + .imageExtent = capabilities.currentExtent, + .imageArrayLayers = 1, + .imageUsage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT, + .imageSharingMode = VK_SHARING_MODE_EXCLUSIVE, + .queueFamilyIndexCount = 1, + .pQueueFamilyIndices = &queue_family_index, + .preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR, + .compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR, + .presentMode = swapchain_present_mode, + }; + + if (vkCreateSwapchainKHR(device, &swapchain_info, NULL, &swapchain) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_CREATE_SWAPCHAIN, "Failed to create swapchain"); + goto FREE_COMMAND_BUFFER; + } + // }}} + + // Get swapchain images {{{ + u32 image_count; + if (vkGetSwapchainImagesKHR(device, swapchain, &image_count, NULL) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_GET_SWAPCHAIN_IMAGE_COUNT, "Failed to get swapchain images count"); + goto DESTROY_SWAPCHAIN; + } + swapchain_images = wapp_array_alloc_capacity(VkImage, VkImageArray, &arena, image_count, true); + swapchain_image_views = wapp_array_alloc_capacity(VkImageView, VkImageViewArray, &arena, image_count, false); + if (!swapchain_images || !swapchain_image_views) { + status = set_error(RETURN_FAILED_TO_ALLOCATE_SWAPCHAIN_IMAGES_MEMORY, + "Failed to allocate swapchain images memory"); + goto DESTROY_SWAPCHAIN; + } + if (vkGetSwapchainImagesKHR(device, swapchain, (u32 *)&(swapchain_images->count), swapchain_images->items) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_GET_SWAPCHAIN_IMAGES, "Failed to get swapchain images"); + goto DESTROY_SWAPCHAIN; + } + // }}} + + // Create swapchain image views {{{ + for (u32 i = 0; i < swapchain_images->count; ++i) { + VkImageViewCreateInfo view_info = { + .sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO, + .image = *wapp_array_get(VkImage, swapchain_images, i), + .viewType = VK_IMAGE_VIEW_TYPE_2D, + .format = swapchain_format.format, + .subresourceRange = { + .levelCount = 1, + .baseMipLevel = 0, + .layerCount = 1, + .baseArrayLayer = 0, + .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT, + }, + }; + VkImageView view; + if (vkCreateImageView(device, &view_info, NULL, &view) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_CREATE_IMAGE_VIEWS, "Failed to create image views"); + goto DESTROY_IMAGE_VIEWS; + } + wapp_array_append_capped(VkImageView, swapchain_image_views, &view); + } + // }}} + + // Create synchronization objects {{{ + VkFenceCreateInfo fence_info = { + .sType = VK_STRUCTURE_TYPE_FENCE_CREATE_INFO, + .flags = VK_FENCE_CREATE_SIGNALED_BIT, + }; + VkSemaphoreCreateInfo semaphore_info = { + .sType = VK_STRUCTURE_TYPE_SEMAPHORE_CREATE_INFO, + }; + for (u32 i = 0; i < FRAMES_IN_FLIGHT; ++i) { + if (vkCreateFence(device, &fence_info, NULL, wapp_array_get(VkFence, &fences, i)) != VK_SUCCESS || + vkCreateSemaphore(device, &semaphore_info, NULL, wapp_array_get(VkSemaphore, &present_complete_semaphores, i)) != VK_SUCCESS || + vkCreateSemaphore(device, &semaphore_info, NULL, wapp_array_get(VkSemaphore, &render_finished_semaphores, i)) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_CREATE_SYNC_OBJECTS, "Failed to create sync objects"); + goto DESTROY_SYNC_OBJECTS; + } + } + // }}} + + // Create shader module {{{ + FILE *shader_file = fopen(SHADER_FILE, "rb"); + fseek(shader_file, 0, SEEK_END); + u64 length = ftell(shader_file); + fseek(shader_file, 0, SEEK_SET); + + u8 *code = wapp_mem_allocator_alloc(&arena, length * sizeof(u8)); + fread(code, sizeof(u8), length, shader_file); + + VkShaderModuleCreateInfo module_info = { + .sType = VK_STRUCTURE_TYPE_SHADER_MODULE_CREATE_INFO, + .codeSize = length, + .pCode = (u32 *)code, + }; + if (vkCreateShaderModule(device, &module_info, NULL, &shader_module) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_CREATE_SHADER_MODULE, "Failed to create shader module"); + goto DESTROY_SYNC_OBJECTS; + } + + fclose(shader_file); + // }}} + + // Create depth resources {{{ + depth_image_format = VK_FORMAT_D32_SFLOAT_S8_UINT; + VkExtent3D depth_image_extent = { + .width = swapchain_extent.width, + .height = swapchain_extent.height, + .depth = 1 + }; + VkImageCreateInfo depth_info = { + .sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO, + .imageType = VK_IMAGE_TYPE_2D, + .format = depth_image_format, + .extent = depth_image_extent, + .mipLevels = 1, + .arrayLayers = 1, + .samples = VK_SAMPLE_COUNT_1_BIT, + .tiling = VK_IMAGE_TILING_OPTIMAL, + .usage = VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT, + .sharingMode = VK_SHARING_MODE_EXCLUSIVE, + }; + + if (vkCreateImage(device, &depth_info, NULL, &depth_image) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_CREATE_DEPTH_IMAGE, "Failed to create depth image"); + goto DESTROY_DEPTH_RESOURCES; + } + + VkMemoryRequirements depth_memory_requirements = {0}; + vkGetImageMemoryRequirements(device, depth_image, &depth_memory_requirements); + + i32 memory_type_index = get_memory_type(&memory_properties.memoryProperties, + depth_memory_requirements.memoryTypeBits, + VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT); + if (memory_type_index == -1) { + status = set_error(RETURN_FAILED_TO_FIND_SUITABLE_MEMORY_TYPE_FOR_DEPTH_IMAGE, + "Failed to find suitable memory type for depth image"); + goto DESTROY_DEPTH_RESOURCES; + } + + VkMemoryPriorityAllocateInfoEXT allocation_priority = { + .sType = VK_STRUCTURE_TYPE_MEMORY_PRIORITY_ALLOCATE_INFO_EXT, + .priority = 1.0f, + }; + VkMemoryAllocateInfo depth_memory_alloc_info = { + .sType = VK_STRUCTURE_TYPE_MEMORY_ALLOCATE_INFO, + .pNext = &allocation_priority, + .allocationSize = depth_memory_requirements.size, + .memoryTypeIndex = (u32)memory_type_index, + }; + if (vkAllocateMemory(device, &depth_memory_alloc_info, NULL, &depth_image_memory) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_ALLOCATE_DEPTH_IMAGE_MEMORY, "Failed to allocate depth image memory"); + goto DESTROY_DEPTH_RESOURCES; + } + + if (vkBindImageMemory(device, depth_image, depth_image_memory, 0) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_BIND_DEPTH_IMAGE_MEMORY, "Failed to bind depth image memory"); + goto DESTROY_DEPTH_RESOURCES; + } + + VkImageViewCreateInfo depth_view_info = { + .sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO, + .image = depth_image, + .viewType = VK_IMAGE_VIEW_TYPE_2D, + .format = depth_image_format, + .subresourceRange = (VkImageSubresourceRange){ + .aspectMask = VK_IMAGE_ASPECT_DEPTH_BIT, + .baseMipLevel = 0, + .levelCount = 1, + .baseArrayLayer = 0, + .layerCount = 1, + }, + }; + if (vkCreateImageView(device, &depth_view_info, NULL, &depth_image_view) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_CREATE_DEPTH_IMAGE_VIEW, "Failed to create depth image view"); + goto DESTROY_DEPTH_RESOURCES; + } + // }}} + + // Create copying command buffer and fence {{{ + VkFence copy_fence = VK_NULL_HANDLE; + VkFenceCreateInfo copy_fence_info = { + .sType = VK_STRUCTURE_TYPE_FENCE_CREATE_INFO, + }; + if (vkCreateFence(device, ©_fence_info, NULL, ©_fence) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_CREATE_COPY_COMMAND_BUFFER, "Failed to create copy command buffer"); + goto FREE_COPY_BUFFER; + } + + VkCommandBuffer copy_buffer = VK_NULL_HANDLE; + VkCommandBufferAllocateInfo copy_buffer_info = { + .sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_ALLOCATE_INFO, + .commandPool = command_pool, + .level = VK_COMMAND_BUFFER_LEVEL_PRIMARY, + .commandBufferCount = 1, + }; + if (vkAllocateCommandBuffers(device, ©_buffer_info, ©_buffer) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_CREATE_COPY_COMMAND_BUFFER, "Failed to create copy command buffer"); + goto FREE_COPY_BUFFER; + } + // }}} + + VkCommandBufferBeginInfo begin_info = { + .sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO, + .flags = VK_COMMAND_BUFFER_USAGE_ONE_TIME_SUBMIT_BIT, + }; + vkBeginCommandBuffer(copy_buffer, &begin_info); + + // Create vertex buffer {{{ + u64 vertex_buffer_size = sizeof(Vertex) * vertices.count; + VkBuffer vertex_staging = VK_NULL_HANDLE; + VkDeviceMemory vertex_staging_memory = VK_NULL_HANDLE; + if (!create_buffer(&memory_properties.memoryProperties, device, 0, vertex_buffer_size, VK_BUFFER_USAGE_TRANSFER_SRC_BIT, + VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT, + &vertex_staging, &vertex_staging_memory)) { + status = set_error(RETURN_FAILED_TO_CREATE_VERTEX_BUFFER, "Failed to create vertex buffer"); + goto DESTROY_VERTEX_BUFFER; + } + + void *vertex_staging_mapped = NULL; + if (vkMapMemory(device, vertex_staging_memory, 0, vertex_buffer_size, 0, &vertex_staging_mapped) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_CREATE_VERTEX_BUFFER, "Failed to create vertex buffer"); + goto DESTROY_VERTEX_BUFFER; + } + memcpy(vertex_staging_mapped, vertices.items, vertex_buffer_size); + vkUnmapMemory(device, vertex_staging_memory); + + if (!create_buffer(&memory_properties.memoryProperties, device, 0, vertex_buffer_size, + VK_BUFFER_USAGE_VERTEX_BUFFER_BIT | VK_BUFFER_USAGE_TRANSFER_DST_BIT, + VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT, &vertex_buffer, &vertex_buffer_memory)) { + status = set_error(RETURN_FAILED_TO_CREATE_VERTEX_BUFFER, "Failed to create vertex buffer"); + goto DESTROY_VERTEX_BUFFER; + } + + VkBufferCopy vertex_buffer_copy = { + .srcOffset = 0, + .dstOffset = 0, + .size = vertex_buffer_size, + }; + vkCmdCopyBuffer(copy_buffer, vertex_staging, vertex_buffer, 1, &vertex_buffer_copy); + // }}} + + // Create index buffer {{{ + u64 index_buffer_size = sizeof(u32) * indices.count; + VkBuffer index_staging = VK_NULL_HANDLE; + VkDeviceMemory index_staging_memory = VK_NULL_HANDLE; + if (!create_buffer(&memory_properties.memoryProperties, device, 0, index_buffer_size, VK_BUFFER_USAGE_TRANSFER_SRC_BIT, + VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT, + &index_staging, &index_staging_memory)) { + status = set_error(RETURN_FAILED_TO_CREATE_VERTEX_BUFFER, "Failed to create index buffer"); + goto DESTROY_INDEX_BUFFER; + } + + void *index_staging_mapped = NULL; + if (vkMapMemory(device, index_staging_memory, 0, index_buffer_size, 0, &index_staging_mapped) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_CREATE_VERTEX_BUFFER, "Failed to create index buffer"); + goto DESTROY_INDEX_BUFFER; + } + memcpy(index_staging_mapped, indices.items, index_buffer_size); + vkUnmapMemory(device, index_staging_memory); + + if (!create_buffer(&memory_properties.memoryProperties, device, 0, index_buffer_size, + VK_BUFFER_USAGE_INDEX_BUFFER_BIT | VK_BUFFER_USAGE_TRANSFER_DST_BIT, + VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT, &index_buffer, &index_buffer_memory)) { + status = set_error(RETURN_FAILED_TO_CREATE_VERTEX_BUFFER, "Failed to create index buffer"); + goto DESTROY_INDEX_BUFFER; + } + + VkBufferCopy index_buffer_copy = { + .srcOffset = 0, + .dstOffset = 0, + .size = index_buffer_size, + }; + vkCmdCopyBuffer(copy_buffer, index_staging, index_buffer, 1, &index_buffer_copy); + // }}} + + vkEndCommandBuffer(copy_buffer); + + // Submit copying vertex and index buffers {{{ + VkSubmitInfo copy_submit_info = { + .sType = VK_STRUCTURE_TYPE_SUBMIT_INFO, + .commandBufferCount = 1, + .pCommandBuffers = ©_buffer, + }; + if (vkQueueSubmit(queue, 1, ©_submit_info, copy_fence) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_COPY_VERTEX_AND_INDEX_DATA, "Failed to copy vertex and index data"); + goto DESTROY_INDEX_BUFFER; + } + while (vkWaitForFences(device, 1, ©_fence, VK_TRUE, UINT64_MAX) == VK_TIMEOUT) {} + // }}} + + // Create graphics pipeline {{{ + VkPipelineShaderStageCreateInfo shader_stages[SHADER_COUNT] = { + (VkPipelineShaderStageCreateInfo){ + .sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO, + .stage = VK_SHADER_STAGE_VERTEX_BIT, + .module = shader_module, + .pName = "vertMain", + }, + (VkPipelineShaderStageCreateInfo){ + .sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO, + .stage = VK_SHADER_STAGE_FRAGMENT_BIT, + .module = shader_module, + .pName = "fragMain", + }, + }; + VkVertexInputBindingDescription vertex_binding = { + .binding = 0, + .stride = sizeof(Vertex), + .inputRate = VK_VERTEX_INPUT_RATE_VERTEX, + }; + VkVertexInputAttributeDescription vertex_attributes[] = { + (VkVertexInputAttributeDescription){ + .location = 0, + .binding = 0, + .format = VK_FORMAT_R32G32B32_SFLOAT, + .offset = offsetof(Vertex, position), + }, + (VkVertexInputAttributeDescription){ + .location = 1, + .binding = 0, + .format = VK_FORMAT_R32G32B32_SFLOAT, + .offset = offsetof(Vertex, color), + }, + }; + u64 attributes_count = sizeof(vertex_attributes) / sizeof(vertex_attributes[0]); + VkPipelineVertexInputStateCreateInfo vertex_input = { + .sType = VK_STRUCTURE_TYPE_PIPELINE_VERTEX_INPUT_STATE_CREATE_INFO, + .vertexBindingDescriptionCount = 1, + .pVertexBindingDescriptions = &vertex_binding, + .vertexAttributeDescriptionCount = attributes_count, + .pVertexAttributeDescriptions = vertex_attributes, + }; + VkPipelineInputAssemblyStateCreateInfo input_assembly = { + .sType = VK_STRUCTURE_TYPE_PIPELINE_INPUT_ASSEMBLY_STATE_CREATE_INFO, + .topology = VK_PRIMITIVE_TOPOLOGY_TRIANGLE_LIST, + }; + VkPipelineRasterizationStateCreateInfo rasterisation = { + .sType = VK_STRUCTURE_TYPE_PIPELINE_RASTERIZATION_STATE_CREATE_INFO, + .polygonMode = VK_POLYGON_MODE_FILL, + .cullMode = VK_CULL_MODE_BACK_BIT, + .frontFace = VK_FRONT_FACE_CLOCKWISE, + .lineWidth = 1.0f, + }; + VkPipelineMultisampleStateCreateInfo multisampling = { + .sType = VK_STRUCTURE_TYPE_PIPELINE_MULTISAMPLE_STATE_CREATE_INFO, + .rasterizationSamples = VK_SAMPLE_COUNT_1_BIT, + }; + VkPipelineDepthStencilStateCreateInfo depth_stencil = { + .sType = VK_STRUCTURE_TYPE_PIPELINE_DEPTH_STENCIL_STATE_CREATE_INFO, + .depthTestEnable = VK_TRUE, + .depthWriteEnable = VK_TRUE, + .depthCompareOp = VK_COMPARE_OP_LESS, + }; + VkPipelineColorBlendAttachmentState blend_state = { + .colorWriteMask = VK_COLOR_COMPONENT_R_BIT | + VK_COLOR_COMPONENT_G_BIT | + VK_COLOR_COMPONENT_B_BIT | + VK_COLOR_COMPONENT_A_BIT, + }; + VkPipelineColorBlendStateCreateInfo blending = { + .sType = VK_STRUCTURE_TYPE_PIPELINE_COLOR_BLEND_STATE_CREATE_INFO, + .logicOpEnable = VK_TRUE, + .logicOp = VK_LOGIC_OP_COPY, + .attachmentCount = 1, + .pAttachments = &blend_state, + }; + VkDynamicState dynamic_states[2] = {VK_DYNAMIC_STATE_VIEWPORT_WITH_COUNT, VK_DYNAMIC_STATE_SCISSOR_WITH_COUNT}; + VkPipelineDynamicStateCreateInfo dynamic_state = { + .sType = VK_STRUCTURE_TYPE_PIPELINE_DYNAMIC_STATE_CREATE_INFO, + .dynamicStateCount = 2, + .pDynamicStates = dynamic_states, + }; + VkPipelineLayoutCreateInfo layout_info = { + .sType = VK_STRUCTURE_TYPE_PIPELINE_LAYOUT_CREATE_INFO, + }; + if (vkCreatePipelineLayout(device, &layout_info, NULL, &pipeline_layout) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_CREATE_PIPELINE_LAYOUT, "Failed to create pipeline layout"); + goto DESTROY_DEPTH_RESOURCES; + } + VkFormat color_attachment_format = swapchain_format.format; + VkPipelineRenderingCreateInfo rendering_info = { + .sType = VK_STRUCTURE_TYPE_PIPELINE_RENDERING_CREATE_INFO, + .colorAttachmentCount = 1, + .pColorAttachmentFormats = &color_attachment_format, + .depthAttachmentFormat = depth_image_format, + .stencilAttachmentFormat = depth_image_format, + }; + + VkGraphicsPipelineCreateInfo pipeline_info = { + .sType = VK_STRUCTURE_TYPE_GRAPHICS_PIPELINE_CREATE_INFO, + .pNext = &rendering_info, + .stageCount = SHADER_COUNT, + .pStages = shader_stages, + .pVertexInputState = &vertex_input, + .pInputAssemblyState = &input_assembly, + .pRasterizationState = &rasterisation, + .pMultisampleState = &multisampling, + .pDepthStencilState = &depth_stencil, + .pColorBlendState = &blending, + .pDynamicState = &dynamic_state, + .layout = pipeline_layout, + }; + if (vkCreateGraphicsPipelines(device, NULL, 1, &pipeline_info, NULL, &graphics_pipeline) != VK_SUCCESS) { + status = set_error(RETURN_FAILED_TO_CREATE_GRAPHICS_PIPELINE, "Failed to create graphics pipeline"); + goto DESTROY_PIPELINE_LAYOUT; + } + // }}} + + // Main loop {{{ + u64 frame_start, frame_time; + b8 running = true; + SDL_Event event = {0}; + while (running) { + frame_start = SDL_GetTicksNS(); + + while (SDL_PollEvent(&event)) { + switch (event.type) { + case SDL_EVENT_QUIT: { + running = false; + } break; + case SDL_EVENT_KEY_DOWN: { + if (event.key.key == SDLK_ESCAPE) { + running = false; + } + } break; + } + } + + draw_frame(); + + frame_time = SDL_GetTicksNS() - frame_start; + if (frame_time < FRAME_TIME) { SDL_DelayNS(FRAME_TIME - frame_time); } + } + // }}} + + // Cleanup {{{ + vkDeviceWaitIdle(device); + +DESTROY_GRAPHICS_PIPELINE: + vkDestroyPipeline(device, graphics_pipeline, NULL); +DESTROY_PIPELINE_LAYOUT: + vkDestroyPipelineLayout(device, pipeline_layout, NULL); +DESTROY_INDEX_BUFFER: + if (index_staging != VK_NULL_HANDLE) { vkDestroyBuffer(device, index_staging, NULL); } + if (index_staging_memory != VK_NULL_HANDLE) { vkFreeMemory(device, index_staging_memory, NULL); } + if (index_buffer != VK_NULL_HANDLE) { vkDestroyBuffer(device, index_buffer, NULL); } + if (index_buffer_memory != VK_NULL_HANDLE) { vkFreeMemory(device, index_buffer_memory, NULL); } +DESTROY_VERTEX_BUFFER: + if (vertex_staging != VK_NULL_HANDLE) { vkDestroyBuffer(device, vertex_staging, NULL); } + if (vertex_staging_memory != VK_NULL_HANDLE) { vkFreeMemory(device, vertex_staging_memory, NULL); } + if (vertex_buffer != VK_NULL_HANDLE) { vkDestroyBuffer(device, vertex_buffer, NULL); } + if (vertex_buffer_memory != VK_NULL_HANDLE) { vkFreeMemory(device, vertex_buffer_memory, NULL); } +FREE_COPY_BUFFER: + if (copy_fence != VK_NULL_HANDLE) { vkDestroyFence(device, copy_fence, NULL); } + if (copy_buffer != VK_NULL_HANDLE) { vkFreeCommandBuffers(device, command_pool, 1, ©_buffer); } +DESTROY_DEPTH_RESOURCES: + if (depth_image_view != VK_NULL_HANDLE) { vkDestroyImageView(device, depth_image_view, NULL); } + if (depth_image_memory != VK_NULL_HANDLE) { vkFreeMemory(device, depth_image_memory, NULL); } + if (depth_image != VK_NULL_HANDLE) { vkDestroyImage(device, depth_image, NULL); } +DESTROY_SHADER_MODULE: + if (shader_module != VK_NULL_HANDLE) { vkDestroyShaderModule(device, shader_module, NULL); } +DESTROY_SYNC_OBJECTS: + for (u32 i = 0; i < FRAMES_IN_FLIGHT; ++i) { + if (wapp_array_get(VkFence, &fences, i) != VK_NULL_HANDLE) { + vkDestroyFence(device, *wapp_array_get(VkFence, &fences, i), NULL); + } + if (wapp_array_get(VkSemaphore, &present_complete_semaphores, i) != VK_NULL_HANDLE) { + vkDestroySemaphore(device, *wapp_array_get(VkSemaphore, &present_complete_semaphores, i), NULL); + } + if (wapp_array_get(VkSemaphore, &render_finished_semaphores, i) != VK_NULL_HANDLE) { + vkDestroySemaphore(device, *wapp_array_get(VkSemaphore, &render_finished_semaphores, i), NULL); + } + } +DESTROY_IMAGE_VIEWS: + for (u32 i = 0; i < swapchain_images->count; ++i) { + VkImageView view = *wapp_array_get(VkImageView, swapchain_image_views, i); + if (view != VK_NULL_HANDLE) { vkDestroyImageView(device, view, NULL); } + } +DESTROY_SWAPCHAIN: + vkDestroySwapchainKHR(device, swapchain, NULL); +FREE_COMMAND_BUFFER: + for (u32 i = 0; i < FRAMES_IN_FLIGHT; ++i) { + if (command_buffers[i] != VK_NULL_HANDLE) { vkFreeCommandBuffers(device, command_pool, 1, &command_buffers[i]); } + } +DESTROY_COMMAND_POOL: + vkDestroyCommandPool(device, command_pool, NULL); +DESTROY_VULKAN_DEVICE: + vkDestroyDevice(device, NULL); +DESTROY_VULKAN_SURFACE: + SDL_Vulkan_DestroySurface(instance, surface, NULL); +DESTROY_VULKAN_INSTANCE: + vkDestroyInstance(instance, NULL); +SDL_CLEANUP: + SDL_DestroyWindow(window); + SDL_Quit(); + + wapp_mem_arena_allocator_destroy(&arena); + // }}} + + return status; +} + +static inline void draw_frame(void) { + // Ensure queue isn't still rendering using objects for the current frame_index {{{ + while (vkWaitForFences(device, 1, wapp_array_get(VkFence, &fences, frame_index), VK_TRUE, UINT64_MAX) == VK_TIMEOUT) {} + vkResetFences(device, 1, wapp_array_get(VkFence, &fences, frame_index)); + // }}} + + vkAcquireNextImageKHR(device, swapchain, UINT64_MAX, *wapp_array_get(VkSemaphore, &present_complete_semaphores, frame_index), + VK_NULL_HANDLE, &image_index); + + record_command_buffer(); + + // Submit command buffer to queue {{{ + VkPipelineStageFlags wait_stage = VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT; + VkSubmitInfo submit_info = { + .sType = VK_STRUCTURE_TYPE_SUBMIT_INFO, + .waitSemaphoreCount = 1, + .pWaitSemaphores = wapp_array_get(VkSemaphore, &present_complete_semaphores, frame_index), + .commandBufferCount = 1, + .pCommandBuffers = &command_buffers[frame_index], + .pWaitDstStageMask = &wait_stage, + .signalSemaphoreCount = 1, + .pSignalSemaphores = wapp_array_get(VkSemaphore, &render_finished_semaphores, frame_index), + }; + vkQueueSubmit(queue, 1, &submit_info, *wapp_array_get(VkFence, &fences, frame_index)); + // }}} + + // Present the swapchain image when rendering is done {{{ + VkPresentInfoKHR present_info = { + .sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR, + .waitSemaphoreCount = 1, + .pWaitSemaphores = wapp_array_get(VkSemaphore, &render_finished_semaphores, frame_index), + .swapchainCount = 1, + .pSwapchains = &(swapchain), + .pImageIndices = &image_index, + }; + vkQueuePresentKHR(queue, &present_info); + // }}} + + frame_index = (frame_index + 1) % FRAMES_IN_FLIGHT; +} + +static inline void record_command_buffer() { + VkCommandBufferBeginInfo begin_info = { + .sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO, + .flags = VK_COMMAND_BUFFER_USAGE_ONE_TIME_SUBMIT_BIT, + }; + vkBeginCommandBuffer(command_buffers[frame_index], &begin_info); + + // Transition image layout for rendering {{{ + transition_image_layout( + command_buffers[frame_index], + *wapp_array_get(VkImage, swapchain_images, image_index), + VK_IMAGE_LAYOUT_UNDEFINED, + VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL, + VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT, + VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT, + VK_ACCESS_NONE, + VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT, + VK_IMAGE_ASPECT_COLOR_BIT + ); + transition_image_layout( + command_buffers[frame_index], + depth_image, + VK_IMAGE_LAYOUT_UNDEFINED, + VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL, + VK_PIPELINE_STAGE_EARLY_FRAGMENT_TESTS_BIT | VK_PIPELINE_STAGE_LATE_FRAGMENT_TESTS_BIT, + VK_PIPELINE_STAGE_EARLY_FRAGMENT_TESTS_BIT | VK_PIPELINE_STAGE_LATE_FRAGMENT_TESTS_BIT, + VK_ACCESS_DEPTH_STENCIL_ATTACHMENT_WRITE_BIT, + VK_ACCESS_DEPTH_STENCIL_ATTACHMENT_WRITE_BIT, + VK_IMAGE_ASPECT_DEPTH_BIT | VK_IMAGE_ASPECT_STENCIL_BIT + ); + // }}} + + // Dynamic rendering {{{ + vkCmdBindPipeline(command_buffers[frame_index], VK_PIPELINE_BIND_POINT_GRAPHICS, graphics_pipeline); + + VkDeviceSize vertex_buffer_offset = 0; + vkCmdBindVertexBuffers(command_buffers[frame_index], 0, 1, &vertex_buffer, &vertex_buffer_offset); + vkCmdBindIndexBuffer(command_buffers[frame_index], index_buffer, 0, VK_INDEX_TYPE_UINT32); + + VkViewport viewport = { + .x = 0, + .y = 0, + .width = swapchain_extent.width, + .height = swapchain_extent.height, + .minDepth = MIN_DEPTH, + .maxDepth = MAX_DEPTH, + }; + vkCmdSetViewportWithCount(command_buffers[frame_index], 1, &viewport); + + VkRect2D scissor = { + .offset = (VkOffset2D){ .x = 0, .y = 0 }, + .extent = swapchain_extent, + }; + vkCmdSetScissorWithCount(command_buffers[frame_index], 1, &scissor); + + VkClearValue color_clear = { + .color = (VkClearColorValue){ .float32 = {0.0f, 0.0f, 0.0f, 0.0f} } + }; + VkRenderingAttachmentInfo color_attachment = { + .sType = VK_STRUCTURE_TYPE_RENDERING_ATTACHMENT_INFO, + .imageView = *wapp_array_get(VkImageView, swapchain_image_views, image_index), + .imageLayout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL, + .loadOp = VK_ATTACHMENT_LOAD_OP_CLEAR, + .storeOp = VK_ATTACHMENT_STORE_OP_STORE, + .clearValue = color_clear + }; + VkClearValue depth_clear = { + .depthStencil = (VkClearDepthStencilValue){ .depth = MAX_DEPTH } + }; + VkRenderingAttachmentInfo depth_attachment = { + .sType = VK_STRUCTURE_TYPE_RENDERING_ATTACHMENT_INFO, + .imageView = depth_image_view, + .imageLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL, + .loadOp = VK_ATTACHMENT_LOAD_OP_CLEAR, + .storeOp = VK_ATTACHMENT_STORE_OP_DONT_CARE, + .clearValue = depth_clear + }; + VkRect2D render_area = { + .offset = (VkOffset2D){.x = 0, .y = 0}, + .extent = swapchain_extent, + }; + VkRenderingInfo rendering_info = { + .sType = VK_STRUCTURE_TYPE_RENDERING_INFO, + .renderArea = render_area, + .layerCount = 1, + .colorAttachmentCount = 1, + .pColorAttachments = &color_attachment, + .pDepthAttachment = &depth_attachment + }; + vkCmdBeginRendering(command_buffers[frame_index], &rendering_info); + vkCmdDrawIndexed(command_buffers[frame_index], indices.count, 1, 0, 0, 0); + vkCmdEndRendering(command_buffers[frame_index]); + // }}} + + // Transition image layout for presentation {{{ + transition_image_layout( + command_buffers[frame_index], + *wapp_array_get(VkImage, swapchain_images, image_index), + VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL, + VK_IMAGE_LAYOUT_PRESENT_SRC_KHR, + VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT, + VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, + VK_ACCESS_2_COLOR_ATTACHMENT_WRITE_BIT, + VK_ACCESS_2_NONE, + VK_IMAGE_ASPECT_COLOR_BIT + ); + // }}} + + vkEndCommandBuffer(command_buffers[frame_index]); +} + +wapp_intern inline b8 create_buffer(VkPhysicalDeviceMemoryProperties *properties, VkDevice device, + VkBufferCreateFlags flags, VkDeviceSize size, VkBufferUsageFlags usage, + VkMemoryPropertyFlags memory_properties, VkBuffer *buffer, + VkDeviceMemory *buffer_memory) { + VkBufferCreateInfo buffer_info = { + .sType = VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO, + .flags = flags, + .size = size, + .usage = usage, + .sharingMode = VK_SHARING_MODE_EXCLUSIVE, + }; + if (vkCreateBuffer(device, &buffer_info, NULL, buffer) != VK_SUCCESS) { return false; } + + VkBufferMemoryRequirementsInfo2 requirements_info = { + .sType = VK_STRUCTURE_TYPE_BUFFER_MEMORY_REQUIREMENTS_INFO_2, + .buffer = *buffer, + }; + VkMemoryRequirements2 requirements = { + .sType = VK_STRUCTURE_TYPE_MEMORY_REQUIREMENTS_2, + }; + vkGetBufferMemoryRequirements2(device, &requirements_info, &requirements); + + i32 memory_type = get_memory_type(properties, requirements.memoryRequirements.memoryTypeBits, memory_properties); + if (memory_type < 0) { return false; } + + VkMemoryPriorityAllocateInfoEXT priority = { + .sType = VK_STRUCTURE_TYPE_MEMORY_PRIORITY_ALLOCATE_INFO_EXT, + .priority = 0.5f, + }; + VkMemoryAllocateInfo alloc_info = { + .sType = VK_STRUCTURE_TYPE_MEMORY_ALLOCATE_INFO, + .pNext = &priority, + .memoryTypeIndex = memory_type, + .allocationSize = requirements.memoryRequirements.size, + }; + if (vkAllocateMemory(device, &alloc_info, NULL, buffer_memory) != VK_SUCCESS) { return false; } + + if (vkBindBufferMemory(device, *buffer, *buffer_memory, 0) != VK_SUCCESS) { return false; } + + return true; +} + +wapp_intern inline void transition_image_layout( + VkCommandBuffer command_buffer, VkImage image, VkImageLayout old_layout, VkImageLayout new_layout, + VkPipelineStageFlags2 src_stage, VkPipelineStageFlags2 dst_stage, + VkAccessFlags2 src_access, VkAccessFlags2 dst_access, VkImageAspectFlags aspect_mask +) { + VkImageMemoryBarrier2 barrier = { + .sType = VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER_2, + .image = image, + .oldLayout = old_layout, + .newLayout = new_layout, + .srcStageMask = src_stage, + .dstStageMask = dst_stage, + .srcAccessMask = src_access, + .dstAccessMask = dst_access, + .srcQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED, + .dstQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED, + .subresourceRange = { + .aspectMask = aspect_mask, + .baseMipLevel = 0, + .levelCount = 1, + .baseArrayLayer = 0, + .layerCount = 1, + }, + }; + VkDependencyInfo dependency = { + .sType = VK_STRUCTURE_TYPE_DEPENDENCY_INFO, + .imageMemoryBarrierCount = 1, + .pImageMemoryBarriers = &barrier, + }; + vkCmdPipelineBarrier2(command_buffer, &dependency); +} + +wapp_intern inline i32 get_memory_type(const VkPhysicalDeviceMemoryProperties *properties, + u32 memory_type_bits, VkMemoryPropertyFlags flags) { + for (u32 i = 0; i < properties->memoryTypeCount; ++i) { + u32 type = (u32)1 << i; + b8 is_type = (type & memory_type_bits) == type; + if (!is_type) { continue; } + + b8 has_properties = (flags & properties->memoryTypes[i].propertyFlags) == flags; + if (has_properties) { return (i32)i; } + } + + return -1; +} + +wapp_intern inline u32 set_error(u32 code, const char *msg) { + fprintf(stderr, "%s\n", msg); + return code; +} diff --git a/mat.c b/mat.c new file mode 100644 index 0000000..dba8d53 --- /dev/null +++ b/mat.c @@ -0,0 +1,18 @@ +// vim:fileencoding=utf-8:foldmethod=marker + +#include "cglm/vec3.h" +#define CGLM_DEFINE_PRINTS +#define CGLM_FORCE_DEPTH_ZERO_TO_ONE +#include "cglm/cglm.h" + +int main(void) { + vec3 eye = {-3.0f, 2.0f, -4.0f}; + vec3 center = {0.0f, 0.0f, 0.0f}; + vec3 up = {0.0f, 1.0f, 0.0f}; + mat4 view; + mat4 proj; + glm_lookat(eye, center, up, view); + glm_perspective(glm_rad(45.0f), 1600.0f / 1200.0f, 0.1f, 10.0f, proj); + glm_mat4_print(view, stdout); + glm_mat4_print(proj, stdout); +} diff --git a/mat.cc b/mat.cc new file mode 100644 index 0000000..578aa1c --- /dev/null +++ b/mat.cc @@ -0,0 +1,19 @@ +#include +#include +#include +#define GLM_FORCE_DEPTH_ZERO_TO_ONE +#include +#include +#define GLM_ENABLE_EXPERIMENTAL +#include +#include + +int main() { + glm::vec3 eye = {0.2f, 4.0f, 8.0f}; + glm::vec3 center = {0.0f, 0.0f, 0.0f}; + glm::vec3 up = {0.0f, 0.0f, 1.0f}; + glm::mat4 view = glm::lookAt(eye, center, up); + glm::mat4 proj = glm::perspective(glm::radians(45.0f), 1600.0f / 1200.0f, 0.1f, 10.0f); + std::cout << glm::to_string(view) << '\n'; + // std::cout << glm::to_string(proj) << '\n'; +} diff --git a/wapp b/wapp new file mode 160000 index 0000000..5e939a7 --- /dev/null +++ b/wapp @@ -0,0 +1 @@ +Subproject commit 5e939a71586f7ada9baf08d029b1f7902171fe3c