2017-03-01 01:46:03 +00:00
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
2017-03-07 14:58:22 +00:00
|
|
|
// 2D collision tree for 1D shadowmap lights
|
2017-03-01 01:46:03 +00:00
|
|
|
// Copyright(C) 2017 Magnus Norddahl
|
|
|
|
// All rights reserved.
|
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
|
|
// along with this program. If not, see http://www.gnu.org/licenses/
|
|
|
|
//
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "gl/system/gl_system.h"
|
|
|
|
#include "gl/shaders/gl_shader.h"
|
|
|
|
#include "gl/dynlights/gl_lightbsp.h"
|
|
|
|
#include "gl/system/gl_interface.h"
|
|
|
|
#include "r_state.h"
|
2017-03-07 14:58:22 +00:00
|
|
|
#include "g_levellocals.h"
|
2017-03-01 01:46:03 +00:00
|
|
|
|
|
|
|
int FLightBSP::GetNodesBuffer()
|
|
|
|
{
|
|
|
|
UpdateBuffers();
|
|
|
|
return NodesBuffer;
|
|
|
|
}
|
|
|
|
|
2017-03-07 14:58:22 +00:00
|
|
|
int FLightBSP::GetLinesBuffer()
|
2017-03-01 01:46:03 +00:00
|
|
|
{
|
|
|
|
UpdateBuffers();
|
2017-03-07 14:58:22 +00:00
|
|
|
return LinesBuffer;
|
2017-03-01 01:46:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void FLightBSP::UpdateBuffers()
|
|
|
|
{
|
|
|
|
if (numnodes != NumNodes || numsegs != NumSegs) // To do: there is probably a better way to detect a map change than this..
|
|
|
|
Clear();
|
|
|
|
|
|
|
|
if (NodesBuffer == 0)
|
|
|
|
GenerateBuffers();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FLightBSP::GenerateBuffers()
|
|
|
|
{
|
2017-03-07 14:58:22 +00:00
|
|
|
if (!Shape)
|
|
|
|
Shape.reset(new Level2DShape());
|
2017-03-01 01:46:03 +00:00
|
|
|
UploadNodes();
|
|
|
|
UploadSegs();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FLightBSP::UploadNodes()
|
|
|
|
{
|
|
|
|
#if 0
|
2017-03-07 14:58:22 +00:00
|
|
|
if (Shape->nodes.Size() > 0)
|
2017-03-01 01:46:03 +00:00
|
|
|
{
|
|
|
|
FILE *file = fopen("nodes.txt", "wb");
|
2017-03-07 14:58:22 +00:00
|
|
|
fwrite(&Shape->nodes[0], sizeof(GPUNode) * Shape->nodes.Size(), 1, file);
|
2017-03-01 01:46:03 +00:00
|
|
|
fclose(file);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
int oldBinding = 0;
|
|
|
|
glGetIntegerv(GL_SHADER_STORAGE_BUFFER_BINDING, &oldBinding);
|
|
|
|
|
|
|
|
glGenBuffers(1, (GLuint*)&NodesBuffer);
|
|
|
|
glBindBuffer(GL_SHADER_STORAGE_BUFFER, NodesBuffer);
|
2017-03-07 14:58:22 +00:00
|
|
|
glBufferData(GL_SHADER_STORAGE_BUFFER, sizeof(GPUNode) * Shape->nodes.Size(), &Shape->nodes[0], GL_STATIC_DRAW);
|
2017-03-01 01:46:03 +00:00
|
|
|
glBindBuffer(GL_SHADER_STORAGE_BUFFER, oldBinding);
|
|
|
|
|
|
|
|
NumNodes = numnodes;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FLightBSP::UploadSegs()
|
|
|
|
{
|
|
|
|
#if 0
|
2017-03-07 23:34:08 +00:00
|
|
|
if (Shape->lines.Size() > 0)
|
2017-03-01 01:46:03 +00:00
|
|
|
{
|
2017-03-07 14:58:22 +00:00
|
|
|
FILE *file = fopen("lines.txt", "wb");
|
2017-03-07 23:34:08 +00:00
|
|
|
fwrite(&Shape->lines[0], sizeof(GPULine) * Shape->lines.Size(), 1, file);
|
2017-03-01 01:46:03 +00:00
|
|
|
fclose(file);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
int oldBinding = 0;
|
|
|
|
glGetIntegerv(GL_SHADER_STORAGE_BUFFER_BINDING, &oldBinding);
|
|
|
|
|
2017-03-07 14:58:22 +00:00
|
|
|
glGenBuffers(1, (GLuint*)&LinesBuffer);
|
|
|
|
glBindBuffer(GL_SHADER_STORAGE_BUFFER, LinesBuffer);
|
2017-03-07 23:34:08 +00:00
|
|
|
glBufferData(GL_SHADER_STORAGE_BUFFER, sizeof(GPULine) * Shape->lines.Size(), &Shape->lines[0], GL_STATIC_DRAW);
|
2017-03-01 01:46:03 +00:00
|
|
|
glBindBuffer(GL_SHADER_STORAGE_BUFFER, oldBinding);
|
|
|
|
|
|
|
|
NumSegs = numsegs;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FLightBSP::Clear()
|
|
|
|
{
|
|
|
|
if (NodesBuffer != 0)
|
|
|
|
{
|
|
|
|
glDeleteBuffers(1, (GLuint*)&NodesBuffer);
|
|
|
|
NodesBuffer = 0;
|
|
|
|
}
|
2017-03-07 14:58:22 +00:00
|
|
|
if (LinesBuffer != 0)
|
|
|
|
{
|
|
|
|
glDeleteBuffers(1, (GLuint*)&LinesBuffer);
|
|
|
|
LinesBuffer = 0;
|
|
|
|
}
|
|
|
|
Shape.reset();
|
|
|
|
}
|
|
|
|
|
2017-03-07 23:34:08 +00:00
|
|
|
bool FLightBSP::ShadowTest(const DVector3 &light, const DVector3 &pos)
|
|
|
|
{
|
|
|
|
return Shape->RayTest(light, pos) >= 1.0f;
|
|
|
|
}
|
|
|
|
|
2017-03-07 14:58:22 +00:00
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
Level2DShape::Level2DShape()
|
|
|
|
{
|
2017-03-07 23:34:08 +00:00
|
|
|
TArray<int> line_elements;
|
2017-03-07 14:58:22 +00:00
|
|
|
TArray<FVector2> centroids;
|
|
|
|
for (unsigned int i = 0; i < level.lines.Size(); i++)
|
|
|
|
{
|
|
|
|
if (level.lines[i].backsector)
|
|
|
|
{
|
|
|
|
centroids.Push(FVector2(0.0f, 0.0f));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-03-07 23:34:08 +00:00
|
|
|
line_elements.Push(i);
|
2017-03-07 14:58:22 +00:00
|
|
|
|
|
|
|
FVector2 v1 = { (float)level.lines[i].v1->fX(), (float)level.lines[i].v1->fY() };
|
|
|
|
FVector2 v2 = { (float)level.lines[i].v2->fX(), (float)level.lines[i].v2->fY() };
|
|
|
|
centroids.Push((v1 + v2) * 0.5f);
|
|
|
|
}
|
|
|
|
|
|
|
|
TArray<int> work_buffer;
|
2017-03-07 23:34:08 +00:00
|
|
|
work_buffer.Resize(line_elements.Size() * 2);
|
|
|
|
root = Subdivide(&line_elements[0], (int)line_elements.Size(), ¢roids[0], &work_buffer[0]);
|
|
|
|
|
|
|
|
lines.Resize(level.lines.Size());
|
|
|
|
for (unsigned int i = 0; i < level.lines.Size(); i++)
|
|
|
|
{
|
|
|
|
const auto &line = level.lines[i];
|
|
|
|
auto &gpuseg = lines[i];
|
|
|
|
|
|
|
|
gpuseg.x = (float)line.v1->fX();
|
|
|
|
gpuseg.y = (float)line.v1->fY();
|
|
|
|
gpuseg.dx = (float)line.v2->fX() - gpuseg.x;
|
|
|
|
gpuseg.dy = (float)line.v2->fY() - gpuseg.y;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
double Level2DShape::RayTest(const DVector3 &ray_start, const DVector3 &ray_end)
|
|
|
|
{
|
|
|
|
DVector2 raydelta = ray_end - ray_start;
|
|
|
|
double raydist2 = raydelta | raydelta;
|
|
|
|
DVector2 raynormal = DVector2(raydelta.Y, -raydelta.X);
|
|
|
|
double rayd = raynormal | ray_start;
|
|
|
|
if (raydist2 < 1.0)
|
|
|
|
return 1.0f;
|
|
|
|
|
|
|
|
double t = 1.0;
|
|
|
|
|
|
|
|
int stack[16];
|
|
|
|
int stack_pos = 1;
|
|
|
|
stack[0] = nodes.Size() - 1;
|
|
|
|
while (stack_pos > 0)
|
|
|
|
{
|
|
|
|
int node_index = stack[stack_pos - 1];
|
|
|
|
|
|
|
|
if (!OverlapRayAABB(ray_start, ray_end, nodes[node_index]))
|
|
|
|
{
|
|
|
|
stack_pos--;
|
|
|
|
}
|
|
|
|
else if (nodes[node_index].line_index != -1) // isLeaf(node_index)
|
|
|
|
{
|
|
|
|
t = MIN(IntersectRayLine(ray_start, ray_end, nodes[node_index].line_index, raydelta, rayd, raydist2), t);
|
|
|
|
stack_pos--;
|
|
|
|
}
|
|
|
|
else if (stack_pos == 16)
|
|
|
|
{
|
|
|
|
stack_pos--; // stack overflow
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
stack[stack_pos - 1] = nodes[node_index].left;
|
|
|
|
stack[stack_pos] = nodes[node_index].right;
|
|
|
|
stack_pos++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Level2DShape::OverlapRayAABB(const DVector2 &ray_start2d, const DVector2 &ray_end2d, const GPUNode &node)
|
|
|
|
{
|
|
|
|
// To do: simplify test to use a 2D test
|
|
|
|
DVector3 ray_start = DVector3(ray_start2d, 0.0);
|
|
|
|
DVector3 ray_end = DVector3(ray_end2d, 0.0);
|
|
|
|
DVector3 aabb_min = DVector3(node.aabb_left, node.aabb_top, -1.0);
|
|
|
|
DVector3 aabb_max = DVector3(node.aabb_right, node.aabb_bottom, 1.0);
|
|
|
|
|
|
|
|
DVector3 c = (ray_start + ray_end) * 0.5f;
|
|
|
|
DVector3 w = ray_end - c;
|
|
|
|
DVector3 h = (aabb_max - aabb_min) * 0.5f; // aabb.extents();
|
|
|
|
|
|
|
|
c -= (aabb_max + aabb_min) * 0.5f; // aabb.center();
|
|
|
|
|
|
|
|
DVector3 v = DVector3(abs(w.X), abs(w.Y), abs(w.Z));
|
|
|
|
|
|
|
|
if (abs(c.X) > v.X + h.X || abs(c.Y) > v.Y + h.Y || abs(c.Z) > v.Z + h.Z)
|
|
|
|
return false; // disjoint;
|
|
|
|
|
|
|
|
if (abs(c.Y * w.Z - c.Z * w.Y) > h.Y * v.Z + h.Z * v.Y ||
|
|
|
|
abs(c.X * w.Z - c.Z * w.X) > h.X * v.Z + h.Z * v.X ||
|
|
|
|
abs(c.X * w.Y - c.Y * w.X) > h.X * v.Y + h.Y * v.X)
|
|
|
|
return false; // disjoint;
|
|
|
|
|
|
|
|
return true; // overlap;
|
|
|
|
}
|
|
|
|
|
|
|
|
double Level2DShape::IntersectRayLine(const DVector2 &ray_start, const DVector2 &ray_end, int line_index, const DVector2 &raydelta, double rayd, double raydist2)
|
|
|
|
{
|
|
|
|
const double epsilon = 0.0000001;
|
|
|
|
const GPULine &line = lines[line_index];
|
|
|
|
|
|
|
|
DVector2 raynormal = DVector2(raydelta.Y, -raydelta.X);
|
|
|
|
|
|
|
|
DVector2 line_pos(line.x, line.y);
|
|
|
|
DVector2 line_delta(line.dx, line.dy);
|
|
|
|
|
|
|
|
double den = raynormal | line_delta;
|
|
|
|
if (abs(den) > epsilon)
|
|
|
|
{
|
|
|
|
double t_line = (rayd - (raynormal | line_pos)) / den;
|
|
|
|
if (t_line >= 0.0 && t_line <= 1.0)
|
|
|
|
{
|
|
|
|
DVector2 linehitdelta = line_pos + line_delta * t_line - ray_start;
|
|
|
|
double t = (raydelta | linehitdelta) / raydist2;
|
|
|
|
return t > 0.0 ? t : 1.0;
|
|
|
|
}
|
|
|
|
}
|
2017-03-07 14:58:22 +00:00
|
|
|
|
2017-03-07 23:34:08 +00:00
|
|
|
return 1.0;
|
2017-03-07 14:58:22 +00:00
|
|
|
}
|
|
|
|
|
2017-03-07 23:34:08 +00:00
|
|
|
int Level2DShape::Subdivide(int *lines, int num_lines, const FVector2 *centroids, int *work_buffer)
|
2017-03-07 14:58:22 +00:00
|
|
|
{
|
|
|
|
if (num_lines == 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
// Find bounding box and median of the lines
|
|
|
|
FVector2 median = FVector2(0.0f, 0.0f);
|
|
|
|
FVector2 aabb_min, aabb_max;
|
|
|
|
aabb_min.X = (float)level.lines[lines[0]].v1->fX();
|
|
|
|
aabb_min.Y = (float)level.lines[lines[0]].v1->fY();
|
|
|
|
aabb_max = aabb_min;
|
|
|
|
for (int i = 0; i < num_lines; i++)
|
|
|
|
{
|
|
|
|
float x1 = (float)level.lines[lines[i]].v1->fX();
|
|
|
|
float y1 = (float)level.lines[lines[i]].v1->fY();
|
|
|
|
float x2 = (float)level.lines[lines[i]].v2->fX();
|
|
|
|
float y2 = (float)level.lines[lines[i]].v2->fY();
|
|
|
|
|
|
|
|
aabb_min.X = MIN(aabb_min.X, x1);
|
|
|
|
aabb_min.X = MIN(aabb_min.X, x2);
|
|
|
|
aabb_min.Y = MIN(aabb_min.Y, y1);
|
|
|
|
aabb_min.Y = MIN(aabb_min.Y, y2);
|
|
|
|
aabb_max.X = MAX(aabb_max.X, x1);
|
|
|
|
aabb_max.X = MAX(aabb_max.X, x2);
|
|
|
|
aabb_max.Y = MAX(aabb_max.Y, y1);
|
|
|
|
aabb_max.Y = MAX(aabb_max.Y, y2);
|
|
|
|
|
|
|
|
median += centroids[lines[i]];
|
|
|
|
}
|
|
|
|
median /= (float)num_lines;
|
|
|
|
|
|
|
|
if (num_lines == 1) // Leaf node
|
|
|
|
{
|
|
|
|
nodes.Push(GPUNode(aabb_min, aabb_max, lines[0]));
|
|
|
|
return (int)nodes.Size() - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find the longest axis
|
|
|
|
float axis_lengths[2] =
|
|
|
|
{
|
|
|
|
aabb_max.X - aabb_min.X,
|
|
|
|
aabb_max.Y - aabb_min.Y
|
|
|
|
};
|
|
|
|
|
|
|
|
int axis_order[2] = { 0, 1 };
|
|
|
|
FVector2 axis_plane[2] = { FVector2(1.0f, 0.0f), FVector2(0.0f, 1.0f) };
|
|
|
|
std::sort(axis_order, axis_order + 2, [&](int a, int b) { return axis_lengths[a] > axis_lengths[b]; });
|
|
|
|
|
|
|
|
// Try split at longest axis, then if that fails the next longest, and then the remaining one
|
|
|
|
int left_count, right_count;
|
|
|
|
FVector2 axis;
|
|
|
|
for (int attempt = 0; attempt < 2; attempt++)
|
|
|
|
{
|
|
|
|
// Find the split plane for axis
|
|
|
|
FVector2 axis = axis_plane[axis_order[attempt]];
|
|
|
|
FVector3 plane(axis, -(median | axis));
|
|
|
|
|
|
|
|
// Split lines into two
|
|
|
|
left_count = 0;
|
|
|
|
right_count = 0;
|
|
|
|
for (int i = 0; i < num_lines; i++)
|
|
|
|
{
|
|
|
|
int line_index = lines[i];
|
|
|
|
|
|
|
|
float side = FVector3(centroids[lines[i]], 1.0f) | plane;
|
|
|
|
if (side >= 0.0f)
|
|
|
|
{
|
|
|
|
work_buffer[left_count] = line_index;
|
|
|
|
left_count++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
work_buffer[num_lines + right_count] = line_index;
|
|
|
|
right_count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (left_count != 0 && right_count != 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if something went wrong when splitting and do a random split instead
|
|
|
|
if (left_count == 0 || right_count == 0)
|
2017-03-01 01:46:03 +00:00
|
|
|
{
|
2017-03-07 14:58:22 +00:00
|
|
|
left_count = num_lines / 2;
|
|
|
|
right_count = num_lines - left_count;
|
2017-03-01 01:46:03 +00:00
|
|
|
}
|
2017-03-07 14:58:22 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
// Move result back into lines list:
|
|
|
|
for (int i = 0; i < left_count; i++)
|
|
|
|
lines[i] = work_buffer[i];
|
|
|
|
for (int i = 0; i < right_count; i++)
|
|
|
|
lines[i + left_count] = work_buffer[num_lines + i];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create child nodes:
|
|
|
|
int left_index = -1;
|
|
|
|
int right_index = -1;
|
|
|
|
if (left_count > 0)
|
2017-03-07 23:34:08 +00:00
|
|
|
left_index = Subdivide(lines, left_count, centroids, work_buffer);
|
2017-03-07 14:58:22 +00:00
|
|
|
if (right_count > 0)
|
2017-03-07 23:34:08 +00:00
|
|
|
right_index = Subdivide(lines + left_count, right_count, centroids, work_buffer);
|
2017-03-07 14:58:22 +00:00
|
|
|
|
|
|
|
nodes.Push(GPUNode(aabb_min, aabb_max, left_index, right_index));
|
|
|
|
return (int)nodes.Size() - 1;
|
2017-03-01 01:46:03 +00:00
|
|
|
}
|