Files
Np_Term/impl/vulkan_engine/vulkan/physics.cpp

151 lines
5.4 KiB
C++
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

#include "vulkan_engine/vulkan/physics.h"
#include <iostream>
#include <limits>
namespace veng {
void Physics::invokeOnColisionEvent(
gsl::not_null<utils::ThreadPool*> thread_pool,
std::unordered_map<std::string, std::shared_ptr<Model>>& models) {
constexpr std::float_t EPSILON = std::numeric_limits<std::float_t>::epsilon();
for (std::unordered_map<std::string, std::shared_ptr<Model>>::iterator
iter_A = models.begin();
iter_A != models.end(); iter_A++) {
for (auto iter_B = std::next(iter_A); iter_B != models.end(); iter_B++) {
auto model_A = iter_A->second;
std::lock_guard<std::mutex> Alock(model_A->modding);
auto model_B = iter_B->second;
std::lock_guard<std::mutex> Block(model_B->modding);
if (!model_A->colision || !model_B->colision) continue;
std::float_t distance =
glm::distance(model_A->position, model_B->position);
std::float_t modelA_radius = model_A->radius * model_A->scale.x;
std::float_t modelB_radius = model_B->radius * model_B->scale.x;
if (distance <= modelA_radius + modelB_radius) {
model_A = iter_A->second;
model_B = iter_B->second;
if (model_A->OnColision)
thread_pool->enqueueJob(model_A->OnColision, model_A, model_B);
model_A = iter_A->second;
model_B = iter_B->second;
if (model_B->OnColision)
thread_pool->enqueueJob(model_B->OnColision, model_A, model_B);
}
}
}
}
void Physics::invokeOnColisionEvent(
gsl::not_null<utils::ThreadPool*> thread_pool,
std::unordered_map<utils::Snowflake, std::shared_ptr<Model>>& models) {
constexpr std::float_t EPSILON = std::numeric_limits<std::float_t>::epsilon();
for (std::unordered_map<utils::Snowflake, std::shared_ptr<Model>>::iterator
iter_A = models.begin();
iter_A != models.end(); iter_A++) {
for (auto iter_B = std::next(iter_A); iter_B != models.end(); iter_B++) {
auto model_A = iter_A->second;
std::lock_guard<std::mutex> Alock(model_A->modding);
auto model_B = iter_B->second;
std::lock_guard<std::mutex> Block(model_B->modding);
if (!model_A->colision || !model_B->colision) continue;
std::float_t distance =
glm::distance(model_A->position, model_B->position);
std::float_t modelA_radius = model_A->radius * model_A->scale.x;
std::float_t modelB_radius = model_B->radius * model_B->scale.x;
if (distance <= modelA_radius + modelB_radius) {
model_A = iter_A->second;
model_B = iter_B->second;
if (model_A->OnColision)
thread_pool->enqueueJob(model_A->OnColision, model_A, model_B);
model_A = iter_A->second;
model_B = iter_B->second;
if (model_B->OnColision)
thread_pool->enqueueJob(model_B->OnColision, model_A, model_B);
}
}
}
}
bool Physics::RayTrace(const glm::vec3& rayOrigin, const glm::vec3& rayDir,
const glm::vec3& v0, const glm::vec3& v1,
const glm::vec3& v2, std::float_t& outDistance) {
constexpr std::float_t EPSILON = std::numeric_limits<std::float_t>::epsilon();
// 삼각형 엣지와 노멀 계산
glm::vec3 edge1 = v1 - v0;
glm::vec3 edge2 = v2 - v0;
glm::vec3 normal = glm::cross(edge1, edge2);
// 평행 여부 판단
glm::vec3 h = glm::cross(rayDir, edge2);
std::float_t a = glm::dot(edge1, h);
if (fabs(a) < EPSILON) {
// 광선 방향과 삼각형 평면이 거의 평행 → coplanar 검사
// 시작점이 평면 위에 있는지
std::float_t distToPlane = glm::dot(normal, rayOrigin - v0);
if (fabs(distToPlane) < EPSILON) {
// 평면 위에 있다면, 점이 삼각형 내부에 있는지 검사
auto pointInTri = [&](const glm::vec3& P) {
// 엣지마다 P가 같은 반대 방향 노멀 쪽에 있는지
glm::vec3 c0 = glm::cross(v1 - v0, P - v0);
glm::vec3 c1 = glm::cross(v2 - v1, P - v1);
glm::vec3 c2 = glm::cross(v0 - v2, P - v2);
std::float_t f0 = glm::dot(normal, c0);
std::float_t f1 = glm::dot(normal, c1);
std::float_t f2 = glm::dot(normal, c2);
return (f0 >= EPSILON && f1 >= EPSILON && f2 >= EPSILON);
};
if (pointInTri(rayOrigin)) {
outDistance = 0.0f;
return true;
}
}
return false;
}
// 기존 MöllerTrumbore 알고리즘
std::float_t f = 1.0f / a;
glm::vec3 s = rayOrigin - v0;
std::float_t u = f * glm::dot(s, h);
if (u < 0.0f || u > 1.0f) return false;
glm::vec3 q = glm::cross(s, edge1);
std::float_t v = f * glm::dot(rayDir, q);
if (v < 0.0f || u + v > 1.0f) return false;
std::float_t t = f * glm::dot(edge2, q);
if (t > EPSILON) {
outDistance = t;
return true;
}
return false;
}
bool Physics::IsPointInsideMesh_(const glm::vec3& point,
const std::vector<veng::Vertex>& vertices,
const std::vector<std::uint32_t>& indices) {
glm::vec3 rayDir = glm::vec3(1.0f, 0.0f, 0.0f); // X+ 방향 광선
int intersectionCount = 0;
for (size_t i = 0; i < indices.size(); i += 3) {
const glm::vec3& v0 = vertices[indices[i + 0]].position;
const glm::vec3& v1 = vertices[indices[i + 1]].position;
const glm::vec3& v2 = vertices[indices[i + 2]].position;
std::float_t t;
if (RayTrace(point, rayDir, v0, v1, v2, t)) {
intersectionCount++;
}
}
return (intersectionCount % 2 == 1); // 홀수면 내부}
}
} // namespace veng