Skip to content
Snippets Groups Projects
Verified Commit 3648d60d authored by Tobias Frisch's avatar Tobias Frisch
Browse files

[#56] Added WIP frustum culling

parent 03a19196
No related branches found
No related tags found
1 merge request!45Resolve "Szene-Repräsentation"
Pipeline #26223 passed
#pragma once
#include <array>
#include <iostream>
#include <glm/vec3.hpp>
namespace vkcv::scene {
......@@ -11,6 +13,7 @@ namespace vkcv::scene {
public:
Bounds();
Bounds(const glm::vec3& min, const glm::vec3& max);
~Bounds() = default;
Bounds(const Bounds& other) = default;
......@@ -21,20 +24,46 @@ namespace vkcv::scene {
void setMin(const glm::vec3& min);
[[nodiscard]]
const glm::vec3& getMin() const;
void setMax(const glm::vec3& max);
[[nodiscard]]
const glm::vec3& getMax() const;
void setCenter(const glm::vec3& center);
[[nodiscard]]
glm::vec3 getCenter() const;
void setSize(const glm::vec3& size);
[[nodiscard]]
glm::vec3 getSize() const;
[[nodiscard]]
std::array<glm::vec3, 8> getCorners() const;
void extend(const glm::vec3& point);
[[nodiscard]]
bool contains(const glm::vec3& point) const;
[[nodiscard]]
bool contains(const Bounds& other) const;
[[nodiscard]]
bool intersects(const Bounds& other) const;
[[nodiscard]]
explicit operator bool() const;
[[nodiscard]]
bool operator!() const;
};
std::ostream& operator << (std::ostream& out, const Bounds& bounds);
}
......@@ -24,7 +24,9 @@ namespace vkcv::scene {
void load(const asset::Scene& scene,
const asset::Mesh& mesh);
void recordDrawcalls(std::vector<glm::mat4>& matrices, std::vector<DrawcallInfo>& drawcalls);
void recordDrawcalls(const glm::mat4& viewProjection,
std::vector<glm::mat4>& matrices,
std::vector<DrawcallInfo>& drawcalls);
public:
~Mesh();
......
......@@ -43,6 +43,9 @@ namespace vkcv::scene {
[[nodiscard]]
const material::Material& getMaterial() const;
[[nodiscard]]
const Bounds& getBounds() const;
explicit operator bool() const;
bool operator!() const;
......
......@@ -23,7 +23,9 @@ namespace vkcv::scene {
void loadMesh(const asset::Scene& asset_scene, const asset::Mesh& asset_mesh);
void recordDrawcalls(std::vector<glm::mat4>& matrices, std::vector<DrawcallInfo>& drawcalls);
void recordDrawcalls(const glm::mat4& viewProjection,
std::vector<glm::mat4>& matrices,
std::vector<DrawcallInfo>& drawcalls);
public:
~Node();
......
......@@ -6,6 +6,11 @@ namespace vkcv::scene {
Bounds::Bounds() :
m_min(glm::vec3(0)),
m_max(glm::vec3(0)) {}
Bounds::Bounds(const glm::vec3 &min, const glm::vec3 &max) :
m_min(min),
m_max(max)
{}
void Bounds::setMin(const glm::vec3 &min) {
m_min = min;
......@@ -42,5 +47,80 @@ namespace vkcv::scene {
glm::vec3 Bounds::getSize() const {
return (m_max - m_min);
}
std::array<glm::vec3, 8> Bounds::getCorners() const {
return {
m_min,
glm::vec3(m_min[0], m_min[1], m_max[2]),
glm::vec3(m_min[0], m_max[1], m_min[2]),
glm::vec3(m_min[0], m_max[1], m_max[2]),
glm::vec3(m_max[0], m_min[1], m_min[2]),
glm::vec3(m_max[0], m_min[1], m_max[2]),
glm::vec3(m_max[0], m_max[1], m_min[2]),
m_max
};
}
void Bounds::extend(const glm::vec3 &point) {
m_min = glm::vec3(
std::min(m_min[0], point[0]),
std::min(m_min[1], point[1]),
std::min(m_min[2], point[2])
);
m_max = glm::vec3(
std::max(m_max[0], point[0]),
std::max(m_max[1], point[1]),
std::max(m_max[2], point[2])
);
}
bool Bounds::contains(const glm::vec3 &point) const {
return (
(point[0] >= m_min[0]) && (point[0] <= m_max[0]) &&
(point[1] >= m_min[1]) && (point[1] <= m_max[1]) &&
(point[2] >= m_min[2]) && (point[2] <= m_max[2])
);
}
bool Bounds::contains(const Bounds &other) const {
return (
(other.m_min[0] >= m_min[0]) && (other.m_max[0] <= m_max[0]) &&
(other.m_min[1] >= m_min[1]) && (other.m_max[1] <= m_max[1]) &&
(other.m_min[2] >= m_min[2]) && (other.m_max[2] <= m_max[2])
);
}
bool Bounds::intersects(const Bounds &other) const {
return (
(other.m_max[0] >= m_min[0]) && (other.m_min[0] <= m_max[0]) &&
(other.m_max[1] >= m_min[1]) && (other.m_min[1] <= m_max[1]) &&
(other.m_max[2] >= m_min[2]) && (other.m_min[2] <= m_max[2])
);
}
Bounds::operator bool() const {
return (
(m_min[0] <= m_max[0]) &&
(m_min[1] <= m_max[1]) &&
(m_min[2] <= m_max[2])
);
}
bool Bounds::operator!() const {
return (
(m_min[0] > m_max[0]) ||
(m_min[1] > m_max[1]) ||
(m_min[2] > m_max[2])
);
}
std::ostream& operator << (std::ostream& out, const Bounds& bounds) {
const auto& min = bounds.getMin();
const auto& max = bounds.getMax();
return out << "[Bounds: (" << min[0] << ", " << min[1] << ", " << min[2] << ") ("
<< max[0] << ", " << max[1] << ", " << max[2] << ") ]";
}
}
......@@ -79,14 +79,50 @@ namespace vkcv::scene {
return *this;
}
void Mesh::recordDrawcalls(std::vector<glm::mat4>& matrices,
static glm::vec3 projectPoint(const glm::mat4& transform, const glm::vec3& point) {
const glm::vec4 position = transform * glm::vec4(point, 1.0f);
return glm::vec3(
position[0] / position[3],
position[1] / position[3],
position[2] / position[3]
);
}
static bool checkFrustum(const Bounds& bounds) {
static Bounds frustum (
glm::vec3(-1.0f, -1.0f, -0.0f),
glm::vec3(+1.0f, +1.0f, +1.0f)
);
return frustum.intersects(bounds);
}
void Mesh::recordDrawcalls(const glm::mat4& viewProjection,
std::vector<glm::mat4>& matrices,
std::vector<DrawcallInfo>& drawcalls) {
for (const auto& part : m_parts) {
matrices.push_back(m_transform);
}
const glm::mat4 transform = viewProjection * m_transform;
for (const auto& drawcall : m_drawcalls) {
drawcalls.push_back(drawcall);
for (size_t i = 0; i < m_parts.size(); i++) {
const MeshPart& part = m_parts[i];
const Bounds& bounds = part.getBounds();
const auto corners = bounds.getCorners();
auto projected = projectPoint(transform, corners[0]);
Bounds aabb (projected, projected);
for (size_t j = 1; j < corners.size(); j++) {
projected = projectPoint(transform, corners[j]);
aabb.extend(projected);
}
if (!checkFrustum(aabb)) {
continue;
}
matrices.push_back(transform);
drawcalls.push_back(m_drawcalls[i]);
}
}
......
......@@ -267,5 +267,9 @@ namespace vkcv::scene {
(!m_indices)
);
}
const Bounds &MeshPart::getBounds() const {
return m_bounds;
}
}
......@@ -55,14 +55,15 @@ namespace vkcv::scene {
return m_nodes.back();
}
void Node::recordDrawcalls(std::vector<glm::mat4>& matrices,
void Node::recordDrawcalls(const glm::mat4& viewProjection,
std::vector<glm::mat4>& matrices,
std::vector<DrawcallInfo>& drawcalls) {
for (auto& mesh : m_meshes) {
mesh.recordDrawcalls(matrices, drawcalls);
mesh.recordDrawcalls(viewProjection, matrices, drawcalls);
}
for (auto& node : m_nodes) {
node.recordDrawcalls(matrices, drawcalls);
node.recordDrawcalls(viewProjection, matrices, drawcalls);
}
}
......
......@@ -74,12 +74,10 @@ namespace vkcv::scene {
std::vector<glm::mat4> matrices;
std::vector<DrawcallInfo> drawcalls;
for (auto& node : m_nodes) {
node.recordDrawcalls(matrices, drawcalls);
}
const glm::mat4 viewProjection = camera.getMVP();
for (auto& matrix : matrices) {
matrix = camera.getMVP() * matrix;
for (auto& node : m_nodes) {
node.recordDrawcalls(viewProjection, matrices, drawcalls);
}
PushConstantData pushConstantData (matrices.data(), sizeof(glm::mat4));
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment