COSC-3P93-Project/Step 3/src/engine/math/bvh.cpp

475 lines
20 KiB
C++

/*
* Created by Brett Terpstra 6920201 on 16/11/22.
* Copyright (c) 2022 Brett Terpstra. All Rights Reserved.
*/
#include <engine/math/bvh.h>
#include <queue>
namespace Raytracing {
/*
* Triangle BVH Node Class
* -------------------------------------------------------------------------
*/
/**
* Returns the first bounding tree with objects in it hit by the provided ray.
* @param r the ray to check for intersection with
* @param min the min t allowed in intersection
* @param max the max t value allowed in intersection
* @return BVHHitData, a structure containing the node, AABB intersection and whether we actually hit anything or not.
*/
TriangleBVHNode::BVHHitData TriangleBVHNode::firstHitRayIntersectTraversal(const Ray& r, PRECISION_TYPE min, PRECISION_TYPE max) {
// first we need to check if the ray actually intersects with this node,
auto ourHitData = aabb.intersects(r, min, max);
// if it doesn't we need to immediately return, which prunes all subtrees to this node.
if (!ourHitData.hit)
return {this, ourHitData, false};
// storage for left and right subtree results.
BVHHitData leftHit{};
BVHHitData rightHit{};
// make sure we actually have a left or right node, and if we do traverse them recursively
if (left != nullptr)
leftHit = left->firstHitRayIntersectTraversal(r, min, max);
if (right != nullptr)
rightHit = right->firstHitRayIntersectTraversal(r, min, max);
// once we are either in a bounding box with objects or we are not.
// to ensure that we keep order and return the first bounding box with objects hit by the ray,
// we must check if the left tree is closer than the right tree, or we didn't hit the right tree.
// if we didn't hit the right tree than we can propagate the left tree up
if (leftHit.hit && (leftHit.data.tMax < rightHit.data.tMax || !rightHit.hit))
return leftHit;
// the same goes for the right tree. if we reach this point the left tree didn't get it or was farther than the right tree
// and so we do the same check. make sure that the right tree is closer, or we didn't hit the left tree.
else if (rightHit.hit && (rightHit.data.tMax < leftHit.data.tMax || !leftHit.hit))
return rightHit;
// if we hit neither tree in all likelihood this is a leaf node, and we can return accordingly
// but if this isn't a leaf node there will be problems. So if this isn't a leaf node (doesn't have objects)
// we can pretend like we didn't hit anything.
return {this, ourHitData, !objs.empty()};
}
/*
* BVH Tree Class
* -------------------------------------------------------------------------
*/
/**
* Creates a BVH using the supplied vector of objects
* @param objects objects used to generate the BVH
*/
void BVHTree::addObjects(const std::vector<Object*>& objects) {
if (root != nullptr)
throw std::runtime_error("BVHTree already exists. What are you trying to do?");
// move all the object's aabb's into world position
std::vector<BVHObject> objs;
for (auto* obj: objects) {
// we don't want to store all the AABBs which don't exist: ie spheres
if (obj->getAABB().isEmpty()) {
noAABBObjects.push_back(obj);
continue;
}
BVHObject bvhObject;
// returns a copy of the AABB object and assigns it in to the tree storage object
bvhObject.aabb = obj->getAABB().translate(obj->getPosition());
// which means we don't have to do memory management, since we are using the pointer without ownership or coping now.
bvhObject.ptr = obj;
objs.push_back(bvhObject);
}
root = addObjectsRecursively(objs, {});
}
/**
* Splits the objects into the two distinct spaces provided in the aabbs param based on their intersection
* This is left side based and produces two vectors which are unique.
* @param aabbs the two spaces which to split the objs vector into
* @param objs object vector to be split into aabbs
* @return BVHPartitionedSpace, a structure with two vectors containing the split objects. Is left side based and unique.
*/
BVHPartitionedSpace BVHTree::partition(const std::pair<AABB, AABB>& aabbs, const std::vector<BVHObject>& objs) {
BVHPartitionedSpace space;
for (const auto& obj: objs) {
// if this object doesn't have an AABB, we cannot use a BVH on it. If this ever fails we have a problem with the implementation.
RTAssert(!obj.aabb.isEmpty());
if (aabbs.first.intersects(obj.aabb))
space.left.push_back(obj);
else if (aabbs.second.intersects(obj.aabb))
space.right.push_back(obj);
}
return space;
}
/**
* Returns all the objects intersected by the provided ray. The returned objects are in no particular order.
* @param ray to use in AABB intersection
* @param min min t allowed for intersection search
* @param max max t allowed
* @return a unordered array of objects intersected by ray in this BVH.
*/
std::vector<BVHObject> BVHTree::rayAnyHitIntersect(const Ray& ray, PRECISION_TYPE min, PRECISION_TYPE max) {
std::queue<BVHNode*> nodes{};
std::vector<BVHObject> objects;
nodes.push(root);
while (!nodes.empty()) {
auto* node = nodes.front();
auto AABB = node->aabb;
auto nodeHitData = AABB.intersects(ray, min, max);
if (nodeHitData.hit) {
if (node->left != nullptr)
nodes.push(node->left);
if (node->right != nullptr)
nodes.push(node->right);
if (!node->objs.empty()) {
for (const auto& obj: node->objs)
objects.push_back(obj);
}
}
nodes.pop();
}
return objects;
}
BVHNode* BVHTree::addObjectsRecursively(const std::vector<BVHObject>& objects, const BVHPartitionedSpace& prevSpace) {
// create a volume for the entire world.
// yes, we could use a recursion provided AABB, but that wouldn't be minimum, only half. this ensures that we have a minimum AABB.
AABB world;
for (const auto& obj: objects)
world = world.expand(obj.aabb);
// if we have a single object then we can create a leaf.
if ((objects.size() <= 1 && !objects.empty())) {
return new BVHNode(objects, world, nullptr, nullptr);
} else if (objects.empty()) // should never reach here!!
return nullptr;
// then split and partition the world
auto splitAABBs = world.splitByLongestAxis();
auto partitionedObjs = partition(splitAABBs, objects);
if (prevSpace == partitionedObjs) {
// if we haven't made progress in splitting the world, try inverting the order we add objects first
partitionedObjs = partition({splitAABBs.second, splitAABBs.first}, objects);
// if we fail then try splitting on another axis
if (prevSpace == partitionedObjs) {
// try all axis
for (int i = 0; i < 3; i++){
splitAABBs = world.splitAlongAxis(i == 0 ? X : i == 1 ? Y : Z);
partitionedObjs = partition(splitAABBs, objects);
// and once we have an axis that works we can break.
if (prevSpace != partitionedObjs)
break;
}
}
}
// if we were unable to find a partition that isn't the same as our last, we should create a leaf here
// otherwise we'll get infinite recursion
if (prevSpace == partitionedObjs) {
return new BVHNode(objects, world, nullptr, nullptr);
}
BVHNode* left = nullptr;
BVHNode* right = nullptr;
// don't try to explore nodes which don't have anything in them.
if (!partitionedObjs.left.empty())
left = addObjectsRecursively(partitionedObjs.left, partitionedObjs);
if (!partitionedObjs.right.empty())
right = addObjectsRecursively(partitionedObjs.right, partitionedObjs);
if (left == nullptr && right == nullptr)
return new BVHNode(objects, world, left, right);
else
return new BVHNode({}, world, left, right);
}
/*
* Triangle BVH Tree Class
* -------------------------------------------------------------------------
*/
/**
* Returns the object array from the first bounding tree with objects in it hit by the provided ray.
* @firstHitRayIntersectTraversal for more information.
*/
std::vector<TriangleBVHObject> TriangleBVHTree::rayFirstHitIntersect(const Ray& ray, PRECISION_TYPE min, PRECISION_TYPE max) {
RTAssert(root != nullptr);
auto results = root->firstHitRayIntersectTraversal(ray, min, max);
RTAssert(results.ptr != nullptr);
if (results.hit)
return results.ptr->objs;
else
return {};
}
/**
* Splits the objects into the two distinct spaces provided in the aabbs param based on their intersection
* This is left side based and produces two vectors which are unique.
* @param aabbs the two spaces which to split the objs vector into
* @param objs object vector to be split into aabbs
* @return BVHPartitionedSpace, a structure with two vectors containing the split objects. Is left side based and unique.
*/
TriangleBVHPartitionedSpace TriangleBVHTree::partition(const std::pair<AABB, AABB>& aabbs, const std::vector<TriangleBVHObject>& objs) {
TriangleBVHPartitionedSpace space;
for (const auto& obj: objs) {
// if this object doesn't have an AABB, we cannot use a BVH on it. If this ever fails we have a problem with the implementation.
RTAssert(!obj.aabb.isEmpty());
if (aabbs.first.intersects(obj.aabb))
space.left.push_back(obj);
else if (aabbs.second.intersects(obj.aabb))
space.right.push_back(obj);
}
return space;
}
TriangleBVHNode* TriangleBVHTree::addObjectsRecursively(const std::vector<TriangleBVHObject>& objects, const TriangleBVHPartitionedSpace& prevSpace) {
// create a volume for the entire world.
// yes, we could use a recursion provided AABB, but that wouldn't be minimum, only half. this ensures that we have a minimum AABB.
AABB world;
for (const auto& obj: objects)
world = world.expand(obj.aabb);
// if we have a single object then we can create a leaf.
if ((objects.size() <= 1 && !objects.empty())) {
return new TriangleBVHNode(objects, world, nullptr, nullptr);
} else if (objects.empty()) // should never reach here!!
return nullptr;
// then split and partition the world
auto splitAABBs = world.splitByLongestAxis();
auto partitionedObjs = partition(splitAABBs, objects);
if (prevSpace == partitionedObjs) {
// if we haven't made progress in splitting the world, try inverting the order we add objects first
partitionedObjs = partition({splitAABBs.second, splitAABBs.first}, objects);
// if we fail then try splitting on another axis
if (prevSpace == partitionedObjs) {
// try all axis
for (int i = 0; i < 3; i++){
splitAABBs = world.splitAlongAxis(i == 0 ? X : i == 1 ? Y : Z);
partitionedObjs = partition(splitAABBs, objects);
// and once we have an axis that works we can break.
if (prevSpace != partitionedObjs)
break;
}
}
}
// if we were unable to find a partition that isn't the same as our last, we should create a leaf here
// otherwise we'll get infinite recursion
if (prevSpace == partitionedObjs) {
return new TriangleBVHNode(objects, world, nullptr, nullptr);
}
TriangleBVHNode* left = nullptr;
TriangleBVHNode* right = nullptr;
// don't try to explore nodes which don't have anything in them.
if (!partitionedObjs.left.empty())
left = addObjectsRecursively(partitionedObjs.left, partitionedObjs);
if (!partitionedObjs.right.empty())
right = addObjectsRecursively(partitionedObjs.right, partitionedObjs);
if (left == nullptr && right == nullptr)
return new TriangleBVHNode(objects, world, left, right);
else
return new TriangleBVHNode({}, world, left, right);
}
/**
* Creates a BVH using the supplied vector of objects
* @param objects objects used to generate the BVH
*/
void TriangleBVHTree::addObjects(const std::vector<TriangleBVHObject>& objects) {
if (root != nullptr)
throw std::runtime_error("Triangle BVHTree already exists. What are you trying to do?");
// move all the object's aabb's into world position
std::vector<TriangleBVHObject> objs;
for (const auto& obj: objects) {
// we don't want to store all the AABBs which don't exist
RTAssert(!obj.aabb.isEmpty());
TriangleBVHObject bvhObject;
// returns a copy of the AABB object and assigns it in to the tree storage object
bvhObject.position = obj.position;
bvhObject.aabb = obj.tri->aabb.translate(obj.position);
bvhObject.tri = obj.tri;
objs.push_back(bvhObject);
}
root = addObjectsRecursively(objs, {});
}
/**
* Returns all the objects intersected by the provided ray. The returned objects are in no particular order.
* @param ray to use in AABB intersection
* @param min min t allowed for intersection search
* @param max max t allowed
* @return a unordered array of objects intersected by ray in this BVH.
*/
std::vector<TriangleBVHObject> TriangleBVHTree::rayAnyHitIntersect(const Ray& ray, PRECISION_TYPE min, PRECISION_TYPE max) {
std::queue<TriangleBVHNode*> nodes{};
std::vector<TriangleBVHObject> objects;
nodes.push(root);
while (!nodes.empty()) {
auto* node = nodes.front();
auto AABB = node->aabb;
auto nodeHitData = AABB.intersects(ray, min, max);
if (nodeHitData.hit) {
if (node->left != nullptr)
nodes.push(node->left);
if (node->right != nullptr)
nodes.push(node->right);
if (!node->objs.empty()) {
for (const auto& obj: node->objs)
objects.push_back(obj);
}
}
nodes.pop();
}
return objects;
}
}
/**
* UNUSED CODE:
*/
// #ifdef COMPILE_GUI
// // renders all the debug VAOs on screen.
// void render(Shader& worldShader) {
// ImGui::Begin(("BVH Data "), nullptr, ImGuiWindowFlags_NoCollapse);
// worldShader.use();
// worldShader.setInt("useWhite", 1);
// worldShader.setVec3("color", {1.0, 1.0, 1.0});
// {
// ImGui::BeginChild("left pane", ImVec2(180, 0), true);
// guiNodesRecur(root);
// ImGui::EndChild();
// }
// ImGui::SameLine();
// {
// ImGui::BeginGroup();
// ImGui::BeginChild("item view",
// ImVec2(0, -ImGui::GetFrameHeightWithSpacing()),
// true,
// ImGuiWindowFlags_AlwaysAutoResize); // Leave room for 1 line below us
// drawNodesRecur(worldShader, root);
// ImGui::EndChild();
// ImGui::EndGroup();
// }
// worldShader.setInt("useWhite", 0);
// ImGui::End();
// }
// #endif
//#ifdef COMPILE_GUI
// void drawNodesRecur(Shader& worldShader, BVHNode* node) {
// node->draw(worldShader);
// if (node->left != nullptr)
// drawNodesRecur(worldShader, node->left);
// if (node->right != nullptr)
// drawNodesRecur(worldShader, node->right);
// }
// void guiNodesRecur(BVHNode* node) {
// node->gui();
// if (node->left != nullptr)
// guiNodesRecur(node->left);
// if (node->right != nullptr)
// guiNodesRecur(node->right);
// }
//#endif
// BVH NODE
// static Raytracing::Mat4x4 getTransform(const AABB& _aabb) {
// Raytracing::Mat4x4 transform{};
// auto center = _aabb.getCenter();
// transform.translate(center);
// auto xRadius = _aabb.getXRadius(center) * 2;
// auto yRadius = _aabb.getYRadius(center) * 2;
// auto zRadius = _aabb.getZRadius(center) * 2;
// transform.scale(float(xRadius), float(yRadius), float(zRadius));
// return transform;
// }
// #ifdef COMPILE_GUI
// void draw(Shader& worldShader) {
// worldShader.setVec3("color", {1.0, 1.0, 1.0});
// aabbVAO->bind();
// if (selected == index) {
// if (selected == index && ImGui::BeginListBox("", ImVec2(250, 350))) {
// std::stringstream strs;
// strs << aabb;
// ImGui::Text("%s", strs.str().c_str());
// for (const auto& item: objs) {
// auto pos = item.ptr->getPosition();
// std::stringstream stm;
// stm << item.aabb;
// ImGui::Text("%s,\n\t%s", (std::to_string(pos.x()) + " " + std::to_string(pos.y()) + " " + std::to_string(pos.z())).c_str(),
// stm.str().c_str());
// }
// ImGui::EndListBox();
// }
//
// for (const auto& obj: objs) {
// auto transform = getTransform(obj.aabb);
// worldShader.setMatrix("transform", transform);
// aabbVAO->draw(worldShader);
// }
// auto transform = getTransform(aabb);
// worldShader.setMatrix("transform", transform);
// aabbVAO->draw(worldShader);
//
// /*auto splitAABBs = aabb.splitByLongestAxis();
// transform = getTransform(splitAABBs.second);
// worldShader.setMatrix("transform", transform);
// aabbVAO->draw(worldShader);
// transform = getTransform(splitAABBs.first);
// worldShader.setMatrix("transform", transform);
// aabbVAO->draw(worldShader);*/
// }
// if (hit){
// if (hit == 1)
// worldShader.setVec3("color", {0.0, 0.0, 1.0});
// else if (hit == 2)
// worldShader.setVec3("color", {0.0, 1.0, 0.0});
// else
// worldShader.setVec3("color", {1.0, 0.5, 0.5});
// auto transform = getTransform(aabb);
// worldShader.setMatrix("transform", transform);
// aabbVAO->draw(worldShader);
// }
// }
// void gui() const {
// int c1 = -1;
// int c2 = -1;
// if (left != nullptr)
// c1 = left->index;
// if (right != nullptr)
// c2 = right->index;
// std::string t;
// if (c1 == -1 && c2 == -1)
// t = " LEAF";
// else
// t = " L: " + std::to_string(c1) + " R: " + std::to_string(c2);
// if (ImGui::Selectable(("S: " + std::to_string(objs.size()) + " I: " + std::to_string(index) + t).c_str(), selected == index))
// selected = index;
// }
// #endif