improved how model matrices are done

This commit is contained in:
Jay Robson 2024-03-06 22:32:54 +11:00
parent 1b317e9ff6
commit e2f8eefe5b
24 changed files with 287 additions and 179 deletions

BIN
assets/scene.blend (Stored with Git LFS)

Binary file not shown.

BIN
assets/scene.glb (Stored with Git LFS)

Binary file not shown.

View File

@ -1,19 +1,32 @@
#version 460 core #version 460 core
layout (location = 2) in vec4 aPos; layout (location = 2) in vec3 aPos;
layout (location = 4) in vec4 aColour; layout (location = 4) in vec4 aColour;
layout (location = 5) in vec3 aMaterial; layout (location = 5) in vec3 aMaterial;
layout (location = 6) in float aTransformIndex; layout (location = 6) in float aTransformIndex;
uniform mat4 model; layout (binding = 3) readonly buffer TransformBuffer
{
mat4 transforms[];
};
uniform mat4 camera; uniform mat4 camera;
out flat int should_ignore; out flat int should_ignore;
void main() mat4 load_model_mat(int index)
{ {
gl_Position = camera * model * aPos; return index < 0 ? mat4(1.f) : transforms[index];
should_ignore = int(aTransformIndex >= 0.f || aMaterial[2] > 0.f || aColour.a < 1.f); }
void main()
{
vec4 pos = vec4(aPos, 1.f);
mat4 model = load_model_mat(int(aTransformIndex));
mat4 mv = camera * model;
gl_Position = mv * pos;
should_ignore = int(aMaterial[2] > 0.f || aColour.a < 1.f);
} }

View File

@ -4,17 +4,16 @@
layout (location = 0) in sampler2D aTex; layout (location = 0) in sampler2D aTex;
layout (location = 1) in vec2 aTexPos; layout (location = 1) in vec2 aTexPos;
layout (location = 2) in vec4 aPos; layout (location = 2) in vec3 aPos;
layout (location = 3) in vec3 aNormal; layout (location = 3) in vec3 aNormal;
layout (location = 4) in vec4 aColour; layout (location = 4) in vec4 aColour;
layout (location = 5) in vec3 aMaterial; layout (location = 5) in vec3 aMaterial;
layout (location = 6) in float aTransformIndex; layout (location = 6) in float aTransformIndex;
uniform mat4 model;
uniform mat4 camera; uniform mat4 camera;
uniform mat4 projection; uniform mat4 projection;
layout (binding = 3) readonly buffer TransformBuffer layout (binding = 3) readonly buffer TransformBuffer1
{ {
mat4 transforms[]; mat4 transforms[];
}; };
@ -29,19 +28,25 @@ out VS_OUT {
out flat sampler2D frag_tex; out flat sampler2D frag_tex;
mat4 load_model_mat(int index)
{
return index < 0 ? mat4(1.f) : transforms[index];
}
void main() void main()
{ {
mat4 m = (aTransformIndex >= 0.f ? transforms[int(aTransformIndex)] : mat4(1.f)) * model; vec4 pos = vec4(aPos, 1.f);
mat4 mvp = camera * m; mat4 model = load_model_mat(int(aTransformIndex));
vec4 pos = mvp * aPos; mat4 mv = camera * model;
mat4 mvp = projection * mv;
vout.normal = mat3(m) * aNormal; vout.normal = mat3(model) * aNormal;
vout.pos = (m * aPos).xyz; vout.pos = (model * pos).xyz;
vout.colour = aColour; vout.colour = aColour;
vout.tex_pos = aTexPos; vout.tex_pos = aTexPos;
vout.material = aMaterial; vout.material = aMaterial;
frag_tex = aTex; frag_tex = aTex;
gl_Position = projection * pos; gl_Position = mvp * pos;
} }

View File

@ -11,9 +11,8 @@ using namespace Sim::Graphics::Equipment;
Reactor::Reactor(const Model& model) Reactor::Reactor(const Model& model)
{ {
g_control_rod = model.load("visual_control_rod_lift"); g_control_rod_lift = model.load("visual_control_rod_lift");
g_control_rod.set_transform_id(); g_control_rod_base = model.load("visual_control_rod_base");
g_control_rod.add(model.load("visual_control_rod_base"));
} }
void Reactor::remesh_static(Mesh& rmesh) void Reactor::remesh_static(Mesh& rmesh)
@ -39,7 +38,15 @@ void Reactor::remesh_static(Mesh& rmesh)
if(r->get_colour()[3] != 0) if(r->get_colour()[3] != 0)
{ {
rmesh.add(g_control_rod, glm::translate(glm::mat4(1), glm::vec3(ox, oy, 0))); Mesh m1, m2;
m1.add(g_control_rod_base, glm::translate(glm::mat4(1), glm::vec3(ox, oy, 0)));
m2.add(g_control_rod_lift, glm::translate(glm::mat4(1), glm::vec3(ox, oy, 0)));
m1.bake_transforms();
m2.bake_transforms();
m2.set_blank_transform();
rmesh.add(m1);
rmesh.add(m2);
} }
} }
} }

View File

@ -8,7 +8,7 @@ namespace Sim::Graphics::Equipment
class Reactor : public MeshGen class Reactor : public MeshGen
{ {
Mesh g_control_rod; Mesh g_control_rod_lift, g_control_rod_base;
public: public:

View File

@ -25,7 +25,7 @@ void Arrays::vertex_attrib_pointers()
glVertexAttribPointer(1, 2, GL_FLOAT, false, sizeof(v), ptr_diff(&v.texpos, &v)); glVertexAttribPointer(1, 2, GL_FLOAT, false, sizeof(v), ptr_diff(&v.texpos, &v));
glEnableVertexAttribArray(1); glEnableVertexAttribArray(1);
glVertexAttribPointer(2, 4, GL_FLOAT, false, sizeof(v), ptr_diff(&v.pos, &v)); glVertexAttribPointer(2, 3, GL_FLOAT, false, sizeof(v), ptr_diff(&v.pos, &v));
glEnableVertexAttribArray(2); glEnableVertexAttribArray(2);
glVertexAttribPointer(3, 3, GL_FLOAT, false, sizeof(v), ptr_diff(&v.normal, &v)); glVertexAttribPointer(3, 3, GL_FLOAT, false, sizeof(v), ptr_diff(&v.normal, &v));

View File

@ -10,7 +10,7 @@ struct Vertex
{ {
unsigned long texid = 0; unsigned long texid = 0;
glm::vec2 texpos = {0, 0}; glm::vec2 texpos = {0, 0};
glm::vec4 pos = {0, 0, 0, 1}; glm::vec3 pos = {0, 0, 0};
glm::vec3 normal = {0, 0, 0}; glm::vec3 normal = {0, 0, 0};
glm::vec4 colour = {1, 1, 1, 1}; glm::vec4 colour = {1, 1, 1, 1};
glm::vec3 material = {0, 0, 0}; glm::vec3 material = {0, 0, 0};

View File

@ -101,7 +101,7 @@ void Font::init()
FT_Done_FreeType(ft); FT_Done_FreeType(ft);
} }
void Mesh::load_text(const char* text, double size) Mesh& Mesh::load_text(const char* text, double size)
{ {
std::vector<Arrays::Vertex> vertices; std::vector<Arrays::Vertex> vertices;
std::vector<unsigned int> indices; std::vector<unsigned int> indices;
@ -113,7 +113,7 @@ void Mesh::load_text(const char* text, double size)
{ {
this->vertices.clear(); this->vertices.clear();
this->indices.clear(); this->indices.clear();
return; return *this;
} }
for(unsigned int i = 0; text[i] != '\0'; i++) for(unsigned int i = 0; text[i] != '\0'; i++)
@ -144,10 +144,10 @@ void Mesh::load_text(const char* text, double size)
float ex = sx + ch.size.x * size; float ex = sx + ch.size.x * size;
float ey = sy + ch.size.y * size; float ey = sy + ch.size.y * size;
vertices.push_back(Arrays::Vertex{.texid=ch.handle, .texpos={0, 0}, .pos={sx, sy, 0, 1}, .normal={0, 0, -1}, .material={0, 0, 1}}); vertices.push_back(Arrays::Vertex{.texid=ch.handle, .texpos={0, 0}, .pos={sx, sy, 0}, .normal={0, 0, -1}, .material={0, 0, 1}});
vertices.push_back(Arrays::Vertex{.texid=ch.handle, .texpos={0, 1}, .pos={sx, ey, 0, 1}, .normal={0, 0, -1}, .material={0, 0, 1}}); vertices.push_back(Arrays::Vertex{.texid=ch.handle, .texpos={0, 1}, .pos={sx, ey, 0}, .normal={0, 0, -1}, .material={0, 0, 1}});
vertices.push_back(Arrays::Vertex{.texid=ch.handle, .texpos={1, 0}, .pos={ex, sy, 0, 1}, .normal={0, 0, -1}, .material={0, 0, 1}}); vertices.push_back(Arrays::Vertex{.texid=ch.handle, .texpos={1, 0}, .pos={ex, sy, 0}, .normal={0, 0, -1}, .material={0, 0, 1}});
vertices.push_back(Arrays::Vertex{.texid=ch.handle, .texpos={1, 1}, .pos={ex, ey, 0, 1}, .normal={0, 0, -1}, .material={0, 0, 1}}); vertices.push_back(Arrays::Vertex{.texid=ch.handle, .texpos={1, 1}, .pos={ex, ey, 0}, .normal={0, 0, -1}, .material={0, 0, 1}});
indices.insert(indices.end(), &index[0], &index[6]); indices.insert(indices.end(), &index[0], &index[6]);
at += 4; at += 4;
@ -156,9 +156,12 @@ void Mesh::load_text(const char* text, double size)
this->vertices = std::move(vertices); this->vertices = std::move(vertices);
this->indices = std::move(indices); this->indices = std::move(indices);
this->transforms.clear();
return *this;
} }
void Mesh::load_text(const char* text, double size, glm::vec2 align) Mesh& Mesh::load_text(const char* text, double size, glm::vec2 align)
{ {
glm::vec2 max; glm::vec2 max;
@ -184,5 +187,7 @@ void Mesh::load_text(const char* text, double size, glm::vec2 align)
v.pos.x -= align.x; v.pos.x -= align.x;
v.pos.y -= align.y; v.pos.y -= align.y;
} }
return *this;
} }

View File

@ -6,6 +6,7 @@
#include "arrays.hpp" #include "arrays.hpp"
#include "../shader.hpp" #include "../shader.hpp"
#include "../camera.hpp" #include "../camera.hpp"
#include "../../util/streams.hpp"
using namespace Sim::Graphics; using namespace Sim::Graphics;
@ -32,40 +33,58 @@ GLMesh::GLMesh(GLMesh&& o)
vbo = o.vbo; vbo = o.vbo;
ebo = o.ebo; ebo = o.ebo;
vao = o.vao; vao = o.vao;
size = o.size; ssbo = o.ssbo;
model_matrix = o.model_matrix;
o.vbo = 0; o.vbo = 0;
o.ebo = 0; o.ebo = 0;
o.vao = 0; o.vao = 0;
o.ssbo = 0;
size = o.size;
ssbo_size = o.ssbo_size;
} }
GLMesh::~GLMesh() GLMesh::~GLMesh()
{ {
if(vbo) glDeleteBuffers(1, &vbo); if(vbo) glDeleteBuffers(1, &vbo);
if(ebo) glDeleteBuffers(1, &ebo); if(ebo) glDeleteBuffers(1, &ebo);
if(ssbo) glDeleteBuffers(1, &ssbo);
if(vao) glDeleteVertexArrays(1, &vao); if(vao) glDeleteVertexArrays(1, &vao);
} }
void GLMesh::bind() void GLMesh::bind(bool bind_ssbo)
{ {
init(this); init(this);
glBindVertexArray(vao); glBindVertexArray(vao);
}
void GLMesh::uniform() if(ssbo_size > 0 && bind_ssbo)
{ {
glUniformMatrix4fv(Shader::ACTIVE->get("model"), 1, false, &model_matrix[0][0]); glBindBuffer(GL_SHADER_STORAGE_BUFFER, ssbo);
glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 3, ssbo);
}
} }
void GLMesh::set(const Mesh& m, int mode) void GLMesh::set(const Mesh& m, int mode, bool send_ssbo)
{ {
glBindBuffer(GL_ARRAY_BUFFER, vbo); glBindBuffer(GL_ARRAY_BUFFER, vbo);
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, ebo); glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, ebo);
glBufferData(GL_ARRAY_BUFFER, m.vertices.size() * sizeof(m.vertices[0]), &m.vertices[0], mode); glBufferData(GL_ARRAY_BUFFER, m.vertices.size() * sizeof(m.vertices[0]), &m.vertices[0], mode);
glBufferData(GL_ELEMENT_ARRAY_BUFFER, m.indices.size() * sizeof(m.indices[0]), &m.indices[0], mode); glBufferData(GL_ELEMENT_ARRAY_BUFFER, m.indices.size() * sizeof(m.indices[0]), &m.indices[0], mode);
this->size = m.indices.size(); this->size = m.indices.size();
this->ssbo_size = m.transforms.size();
if(ssbo_size > 0 && send_ssbo)
{
if(ssbo == 0)
{
glGenBuffers(1, &ssbo);
}
glBindBuffer(GL_SHADER_STORAGE_BUFFER, ssbo);
glBufferData(GL_SHADER_STORAGE_BUFFER, m.transforms.size() * sizeof(m.transforms[0]), &m.transforms[0], mode);
}
} }
void GLMesh::render() void GLMesh::render()

View File

@ -13,9 +13,9 @@ namespace Sim::Graphics
struct GLMesh struct GLMesh
{ {
unsigned int vao = 0, vbo = 0, ebo = 0, size = 0; unsigned int vao = 0, vbo = 0, ebo = 0, ssbo = 0;
int ssbo_size = 0;
glm::mat4 model_matrix {1.0f}; int size = 0;
constexpr GLMesh() { } constexpr GLMesh() { }
@ -23,9 +23,8 @@ struct GLMesh
GLMesh(const GLMesh& o) = delete; GLMesh(const GLMesh& o) = delete;
~GLMesh(); ~GLMesh();
void bind(); void bind(bool bind_ssbo = true);
void uniform(); void set(const Mesh& m, int mode, bool send_ssbo = true);
void set(const Mesh& m, int mode);
void render(int type); void render(int type);
void render(); void render();
}; };

View File

@ -15,79 +15,122 @@ Mesh::Mesh()
} }
void Mesh::set_transform_id() Mesh& Mesh::set_texture_id(unsigned int id)
{
for(unsigned int i = 0; i < vertices.size(); i++)
{
vertices[i].transform_id = 0;
}
max_transform_id = 0;
}
void Mesh::set_texture_id(unsigned int id)
{ {
for(unsigned int i = 0; i < vertices.size(); i++) for(unsigned int i = 0; i < vertices.size(); i++)
{ {
vertices[i].texid = id; vertices[i].texid = id;
} }
return *this;
} }
void Mesh::add(const Mesh& o, glm::mat4 mat) Mesh& Mesh::set_blank_transform()
{ {
unsigned int off = vertices.size(); transforms = {glm::mat4(1)};
float t_off = max_transform_id + 1;
for(unsigned int i = 0; i < vertices.size(); i++)
{
vertices[i].transform_id = 0;
}
return *this;
}
Mesh& Mesh::add(const Mesh& o, glm::mat4 mat, bool bake)
{
int off = vertices.size();
glm::mat3 mat3(mat);
vertices.reserve(vertices.size() + o.vertices.size());
indices.reserve(indices.size() + o.indices.size()); indices.reserve(indices.size() + o.indices.size());
vertices.reserve(vertices.size() + o.vertices.size());
for(unsigned int i = 0; i < o.vertices.size(); i++) if(bake)
{
for(int i = 0; i < o.vertices.size(); i++)
{
Arrays::Vertex v = o.vertices[i];
int t_id = (int)v.transform_id;
glm::mat4 t_mat = t_id >= 0 ? transforms[t_id] : glm::mat4(1);
t_mat = mat * t_mat;
v.pos = t_mat * glm::vec4(v.pos, 1);
v.normal = glm::normalize(glm::mat3(t_mat) * v.normal);
v.transform_id = -1;
vertices.push_back(v);
}
for(int i = 0; i < o.indices.size(); i++)
{
indices.push_back(o.indices[i] + off);
}
return *this;
}
glm::mat3 mat3(mat);
float t_off = transforms.size();
float t_new = -1;
if(mat != glm::mat4(1))
{
t_new = transforms.size() + o.transforms.size();
}
transforms.reserve(transforms.size() + o.transforms.size() + (t_new >= 0 ? 1 : 0));
for(int i = 0; i < o.vertices.size(); i++)
{ {
Arrays::Vertex v = o.vertices[i]; Arrays::Vertex v = o.vertices[i];
v.normal = mat3 * v.normal;
v.pos = mat * v.pos;
if(v.transform_id >= 0) if(v.transform_id >= 0)
{ {
v.transform_id += t_off; v.transform_id += t_off;
max_transform_id = std::max(max_transform_id, v.transform_id); }
else
{
v.transform_id = t_new;
} }
vertices.push_back(v); vertices.push_back(v);
} }
for(unsigned int i = 0; i < o.indices.size(); i++) for(int i = 0; i < o.transforms.size(); i++)
{
transforms.push_back(o.transforms[i] * mat);
}
for(int i = 0; i < o.indices.size(); i++)
{ {
indices.push_back(o.indices[i] + off); indices.push_back(o.indices[i] + off);
} }
if(t_new >= 0)
{
transforms.push_back(mat);
} }
void Mesh::add(const Mesh& o) return *this;
{
add(o, glm::mat4(1));
} }
void Mesh::set_vertices(const Arrays::Vertex* data, size_t size) Mesh& Mesh::bake_transforms()
{ {
vertices.clear(); for(unsigned int i = 0; i < vertices.size(); i++)
vertices.reserve(size); {
int id = (int)vertices[i].transform_id;
for(unsigned int i = 0; i < size; i++) if(id >= 0)
{ {
vertices.push_back(data[i]); glm::mat4 transform = transforms[id];
vertices[i].pos = glm::vec3(transform * glm::vec4(vertices[i].pos, 1));
vertices[i].normal = glm::normalize(glm::mat3(transform) * vertices[i].normal);
vertices[i].transform_id = -1;
} }
} }
void Mesh::set_indices(const unsigned int* data, size_t size) transforms.clear();
{
indices.clear();
indices.reserve(size);
for(unsigned int i = 0; i < size; i++) return *this;
{
indices.push_back(data[i]);
}
} }
typedef glm::vec<3, double> vec3; typedef glm::vec<3, double> vec3;
@ -158,12 +201,22 @@ bool Mesh::check_intersect(vec3 pos, vec3 path) const
for(unsigned int i = 0; i < indices.size(); i += 3) for(unsigned int i = 0; i < indices.size(); i += 3)
{ {
vec3 v[3] = { Arrays::Vertex verts[3] = {
vec3(this->vertices[indices[i]].pos), vertices[indices[i]],
vec3(this->vertices[indices[i + 1]].pos), vertices[indices[i + 1]],
vec3(this->vertices[indices[i + 2]].pos) vertices[indices[i + 2]]
}; };
vec3 v[3];
for(int j = 0; j < 3; j++)
{
int t_id = (int)verts[j].transform_id;
glm::mat4 t_mat = t_id >= 0 ? transforms[t_id] : glm::mat4(1);
v[j] = vec3(t_mat * glm::vec4(verts[j].pos, 1));
}
vec3 ipoint; vec3 ipoint;
vec3 normal = glm::normalize(glm::cross(v[1] - v[0], v[2] - v[0])); vec3 normal = glm::normalize(glm::cross(v[1] - v[0], v[2] - v[0]));
double d = glm::dot(normal, path); double d = glm::dot(normal, path);
@ -226,12 +279,22 @@ vec3 Mesh::calc_intersect(vec3 pos, vec3 path) const
for(unsigned int i = 0; i < indices.size(); i += 3) for(unsigned int i = 0; i < indices.size(); i += 3)
{ {
vec3 v[3] = { Arrays::Vertex verts[3] = {
vec3(this->vertices[indices[i]].pos), vertices[indices[i]],
vec3(this->vertices[indices[i + 1]].pos), vertices[indices[i + 1]],
vec3(this->vertices[indices[i + 2]].pos) vertices[indices[i + 2]]
}; };
vec3 v[3];
for(int j = 0; j < 3; j++)
{
int t_id = (int)verts[j].transform_id;
glm::mat4 t_mat = t_id >= 0 ? transforms[t_id] : glm::mat4(1);
v[j] = vec3(t_mat * glm::vec4(verts[j].pos, 1));
}
if(calc_intercept_vert(v, pos, path, path_n, l)) if(calc_intercept_vert(v, pos, path, path_n, l))
{ {
changing = true; changing = true;
@ -265,4 +328,3 @@ Mesh Mesh::to_lines() const
return m; return m;
} }

View File

@ -19,18 +19,16 @@ struct Mesh
{ {
std::vector<Arrays::Vertex> vertices; std::vector<Arrays::Vertex> vertices;
std::vector<unsigned int> indices; std::vector<unsigned int> indices;
float max_transform_id = -1; std::vector<glm::mat4> transforms;
Mesh(); Mesh();
void set_transform_id(); Mesh& bake_transforms();
void set_texture_id(unsigned int id); Mesh& set_blank_transform();
void set_vertices(const Arrays::Vertex* data, size_t size); Mesh& set_texture_id(unsigned int id);
void set_indices(const unsigned int* data, size_t size); Mesh& load_text(const char* text, double size);
void load_text(const char* text, double size); Mesh& load_text(const char* text, double size, glm::vec2 align);
void load_text(const char* text, double size, glm::vec2 align); Mesh& add(const Mesh& o, glm::mat4 mat = glm::mat4(1), bool bake = false);
void add(const Mesh& o, glm::mat4 mat);
void add(const Mesh& o);
Mesh to_lines() const; Mesh to_lines() const;
bool check_focus() const; bool check_focus() const;

View File

@ -23,6 +23,7 @@ struct ProcState
std::string base; std::string base;
std::vector<Arrays::Vertex> vertices; std::vector<Arrays::Vertex> vertices;
std::vector<unsigned int> indices; std::vector<unsigned int> indices;
std::vector<glm::mat4> transforms;
std::unordered_map<const aiTexture*, unsigned int> handles; std::unordered_map<const aiTexture*, unsigned int> handles;
}; };
@ -51,7 +52,7 @@ static unsigned int proc_texture(const ProcState& state, aiMaterial* mat, const
return Texture::load(state.base + "/" + filename); return Texture::load(state.base + "/" + filename);
} }
static void proc_mesh(ProcState& state, glm::mat4 mat, aiMesh* mesh, const aiScene* scene) static void proc_mesh(ProcState& state, aiMesh* mesh, const aiScene* scene)
{ {
aiMaterial* material = scene->mMaterials[mesh->mMaterialIndex]; aiMaterial* material = scene->mMaterials[mesh->mMaterialIndex];
aiString name; aiString name;
@ -114,7 +115,6 @@ static void proc_mesh(ProcState& state, glm::mat4 mat, aiMesh* mesh, const aiSce
unsigned int handle = proc_texture(state, material, scene, aiTextureType_BASE_COLOR, 0); unsigned int handle = proc_texture(state, material, scene, aiTextureType_BASE_COLOR, 0);
unsigned int offset = state.offset; unsigned int offset = state.offset;
glm::mat3 mat3(mat);
if(!handle) if(!handle)
{ {
@ -131,7 +131,8 @@ static void proc_mesh(ProcState& state, glm::mat4 mat, aiMesh* mesh, const aiSce
Arrays::Vertex vertex; Arrays::Vertex vertex;
auto [x, y, z] = mesh->mVertices[i]; auto [x, y, z] = mesh->mVertices[i];
vertex.pos = glm::vec4(x, y, z, 1) * mat; vertex.pos = glm::vec3(x, y, z);
vertex.transform_id = state.transforms.size();
vertex.material = matv; vertex.material = matv;
vertex.texid = handle; vertex.texid = handle;
vertex.colour = cb; vertex.colour = cb;
@ -139,7 +140,7 @@ static void proc_mesh(ProcState& state, glm::mat4 mat, aiMesh* mesh, const aiSce
if(mesh->HasNormals()) if(mesh->HasNormals())
{ {
auto [x, y, z] = mesh->mNormals[i]; auto [x, y, z] = mesh->mNormals[i];
vertex.normal = glm::vec3(x, y, z) * mat3; vertex.normal = glm::vec3(x, y, z);
} }
/*if(mesh->HasTangentsAndBitangents()) /*if(mesh->HasTangentsAndBitangents())
@ -207,7 +208,12 @@ static void proc_node(ProcState& state, glm::mat4 mat, aiNode* node, const aiSce
for(size_t i = 0; i < node->mNumMeshes; i++) for(size_t i = 0; i < node->mNumMeshes; i++)
{ {
aiMesh* mesh = scene->mMeshes[node->mMeshes[i]]; aiMesh* mesh = scene->mMeshes[node->mMeshes[i]];
proc_mesh(state, mat, mesh, scene); proc_mesh(state, mesh, scene);
}
if(node->mNumMeshes > 0)
{
state.transforms.push_back(glm::transpose(mat));
} }
} }
@ -296,6 +302,7 @@ Mesh Model::load(const char* name, glm::mat4 mat) const
mesh.vertices = std::move(state.vertices); mesh.vertices = std::move(state.vertices);
mesh.indices = std::move(state.indices); mesh.indices = std::move(state.indices);
mesh.transforms = std::move(state.transforms);
return mesh; return mesh;
} }

View File

@ -137,22 +137,21 @@ void Core::remesh_static(Mesh& rmesh)
{ {
Mesh mesh; Mesh mesh;
mesh.load_text("Reactor Core", 0.04); mesh.load_text("Reactor Core", 0.04);
rmesh.add(mesh, mat); rmesh.add(mesh, mat, true);
} }
static Mesh add_dot(glm::mat4 model_mat, glm::vec4 colour) static Mesh add_dot(glm::mat4 model_mat, glm::vec4 colour)
{ {
unsigned int indices[] = {0, 1, 3, 0, 3, 2}; Mesh mesh;
Arrays::Vertex vertices[] = {
{.texid=Texture::handle_white, .texpos={0, 0}, .pos=model_mat * glm::vec4(-0.75, -0.75, 0, 1), .normal={0, 0, -1}, .colour=colour, .material={0, 0, 1}}, mesh.indices = {0, 1, 3, 0, 3, 2};
{.texid=Texture::handle_white, .texpos={0, 1}, .pos=model_mat * glm::vec4(-0.75, 0.75, 0, 1), .normal={0, 0, -1}, .colour=colour, .material={0, 0, 1}}, mesh.vertices = {
{.texid=Texture::handle_white, .texpos={1, 0}, .pos=model_mat * glm::vec4( 0.75, -0.75, 0, 1), .normal={0, 0, -1}, .colour=colour, .material={0, 0, 1}}, {.texid=Texture::handle_white, .texpos={0, 0}, .pos=glm::vec3(model_mat * glm::vec4(-0.75, -0.75, 0, 1)), .normal={0, 0, -1}, .colour=colour, .material={0, 0, 1}},
{.texid=Texture::handle_white, .texpos={1, 1}, .pos=model_mat * glm::vec4( 0.75, 0.75, 0, 1), .normal={0, 0, -1}, .colour=colour, .material={0, 0, 1}}, {.texid=Texture::handle_white, .texpos={0, 1}, .pos=glm::vec3(model_mat * glm::vec4(-0.75, 0.75, 0, 1)), .normal={0, 0, -1}, .colour=colour, .material={0, 0, 1}},
{.texid=Texture::handle_white, .texpos={1, 0}, .pos=glm::vec3(model_mat * glm::vec4( 0.75, -0.75, 0, 1)), .normal={0, 0, -1}, .colour=colour, .material={0, 0, 1}},
{.texid=Texture::handle_white, .texpos={1, 1}, .pos=glm::vec3(model_mat * glm::vec4( 0.75, 0.75, 0, 1)), .normal={0, 0, -1}, .colour=colour, .material={0, 0, 1}},
}; };
Mesh mesh;
mesh.set_indices(indices, 6);
mesh.set_vertices(vertices, 4);
return mesh; return mesh;
} }

View File

@ -62,10 +62,6 @@ PrimaryLoop::PrimaryLoop(const Model& model)
m_switch_pump = model.load("click_pump_switch_1"); m_switch_pump = model.load("click_pump_switch_1");
m_switch_bypass = model.load("click_bypass_switch"); m_switch_bypass = model.load("click_bypass_switch");
m_switch_inlet = model.load("click_inlet_switch"); m_switch_inlet = model.load("click_inlet_switch");
g_switch_pump.set_transform_id();
g_switch_bypass.set_transform_id();
g_switch_inlet.set_transform_id();
} }
void PrimaryLoop::remesh_static(Mesh& rmesh) void PrimaryLoop::remesh_static(Mesh& rmesh)
@ -86,7 +82,7 @@ void PrimaryLoop::remesh_static(Mesh& rmesh)
ss << "Level\n"; ss << "Level\n";
mesh.load_text(ss.str().c_str(), 0.04); mesh.load_text(ss.str().c_str(), 0.04);
rmesh.add(mesh, mat); rmesh.add(mesh, mat, true);
rmesh.add(g_switch_pump); rmesh.add(g_switch_pump);
rmesh.add(g_switch_bypass); rmesh.add(g_switch_bypass);

View File

@ -28,9 +28,6 @@ SecondaryLoop::SecondaryLoop(const Model& model)
m_joystick_turbine_inlet = model.load("click_inlet_joystick"); m_joystick_turbine_inlet = model.load("click_inlet_joystick");
m_switch_2 = model.load("click_pump_switch_2"); m_switch_2 = model.load("click_pump_switch_2");
m_switch_3 = model.load("click_pump_switch_3"); m_switch_3 = model.load("click_pump_switch_3");
g_switch_2.set_transform_id();
g_switch_3.set_transform_id();
} }
void SecondaryLoop::update(double dt) void SecondaryLoop::update(double dt)
@ -56,7 +53,7 @@ void SecondaryLoop::remesh_static(Mesh& rmesh)
ss << "Power\nSpeed\nFlow\n\n"; ss << "Power\nSpeed\nFlow\n\n";
mesh.load_text(ss.str().c_str(), 0.04); mesh.load_text(ss.str().c_str(), 0.04);
rmesh.add(mesh, mat); rmesh.add(mesh, mat, true);
rmesh.add(g_switch_2); rmesh.add(g_switch_2);
rmesh.add(g_switch_3); rmesh.add(g_switch_3);
} }

View File

@ -28,12 +28,6 @@ Turbine::Turbine(const Model& model)
g_switch_breaker = model.load("visual_breaker_switch"); g_switch_breaker = model.load("visual_breaker_switch");
m_switch_breaker = model.load("click_breaker_switch"); m_switch_breaker = model.load("click_breaker_switch");
g_dial_phase.set_transform_id();
g_dial_voltage.set_transform_id();
g_dial_power.set_transform_id();
g_dial_frequency.set_transform_id();
g_switch_breaker.set_transform_id();
} }
void Turbine::update(double dt) void Turbine::update(double dt)
@ -55,22 +49,12 @@ void Turbine::get_static_transforms(std::vector<glm::mat4>& transforms)
if(rpm > 3570 && rpm < 3630) if(rpm > 3570 && rpm < 3630)
{ {
mat_phase = glm::translate(mat_phase, glm::vec3(6.35, 3.949, 1.35));
mat_phase = glm::rotate(mat_phase, float(sys.loop.generator.get_phase_diff()), glm::vec3(0, 1, 0)); mat_phase = glm::rotate(mat_phase, float(sys.loop.generator.get_phase_diff()), glm::vec3(0, 1, 0));
mat_phase = glm::translate(mat_phase, glm::vec3(-6.35, -3.949, -1.35));
} }
mat_voltage = glm::translate(mat_voltage, glm::vec3(6.95, 3.949, 1.95));
mat_voltage = glm::rotate(mat_voltage, float(Util::Math::map(sys.loop.generator.get_voltage(), 0, 24e3, 0, M_PI)), glm::vec3(0, 1, 0)); mat_voltage = glm::rotate(mat_voltage, float(Util::Math::map(sys.loop.generator.get_voltage(), 0, 24e3, 0, M_PI)), glm::vec3(0, 1, 0));
mat_voltage = glm::translate(mat_voltage, glm::vec3(-6.95, -3.949, -1.95));
mat_power = glm::translate(mat_power, glm::vec3(6.35, 3.949, 1.95));
mat_power = glm::rotate(mat_power, float(Util::Math::map(sys.loop.generator.get_power(), 0, 600e6, 0, M_PI)), glm::vec3(0, 1, 0)); mat_power = glm::rotate(mat_power, float(Util::Math::map(sys.loop.generator.get_power(), 0, 600e6, 0, M_PI)), glm::vec3(0, 1, 0));
mat_power = glm::translate(mat_power, glm::vec3(-6.35, -3.949, -1.95));
mat_frequency = glm::translate(mat_frequency, glm::vec3(6.95, 3.949, 1.35));
mat_frequency = glm::rotate(mat_frequency, float(Util::Math::map(sys.loop.generator.get_frequency(), 0, 120, 0, M_PI)), glm::vec3(0, 1, 0)); mat_frequency = glm::rotate(mat_frequency, float(Util::Math::map(sys.loop.generator.get_frequency(), 0, 120, 0, M_PI)), glm::vec3(0, 1, 0));
mat_frequency = glm::translate(mat_frequency, glm::vec3(-6.95, -3.949, -1.35));
transforms.push_back(mat_phase); transforms.push_back(mat_phase);
transforms.push_back(mat_voltage); transforms.push_back(mat_voltage);
@ -91,7 +75,7 @@ void Turbine::remesh_static(Mesh& rmesh)
ss << "Heat\nPressure\nSpeed\n\n"; ss << "Heat\nPressure\nSpeed\n\n";
mesh.load_text(ss.str().c_str(), 0.04); mesh.load_text(ss.str().c_str(), 0.04);
rmesh.add(mesh, mat); rmesh.add(mesh, mat, true);
rmesh.add(g_dial_phase); rmesh.add(g_dial_phase);
rmesh.add(g_dial_voltage); rmesh.add(g_dial_voltage);

View File

@ -38,7 +38,7 @@ void Vessel::remesh_static(Mesh& rmesh)
ss << "Control Rods\nMin\nMax\nSpeed\n"; ss << "Control Rods\nMin\nMax\nSpeed\n";
mesh.load_text(ss.str().c_str(), 0.04); mesh.load_text(ss.str().c_str(), 0.04);
rmesh.add(mesh, mat); rmesh.add(mesh, mat, true);
} }
void Vessel::remesh_slow(Mesh& rmesh) void Vessel::remesh_slow(Mesh& rmesh)

View File

@ -30,16 +30,15 @@ void UI::init()
Mesh m; Mesh m;
unsigned int handle = Texture::handle_white; unsigned int handle = Texture::handle_white;
const unsigned int indices[] = {0, 1, 3, 0, 3, 2}; m.indices = {0, 1, 3, 0, 3, 2};
const Arrays::Vertex vertices[] = { m.vertices = {
{.texid=handle, .texpos={0, 0}, .pos={-1, -1, 0, 1}, .normal={0, 0, -1}, .colour={1, 1, 1, 1}, .material={0, 0, 1}}, {.texid=handle, .texpos={0, 0}, .pos={-1, -1, 0}, .normal={0, 0, -1}, .colour={1, 1, 1, 1}, .material={0, 0, 1}},
{.texid=handle, .texpos={0, 1}, .pos={-1, 1, 0, 1}, .normal={0, 0, -1}, .colour={1, 1, 1, 1}, .material={0, 0, 1}}, {.texid=handle, .texpos={0, 1}, .pos={-1, 1, 0}, .normal={0, 0, -1}, .colour={1, 1, 1, 1}, .material={0, 0, 1}},
{.texid=handle, .texpos={1, 0}, .pos={ 1, -1, 0, 1}, .normal={0, 0, -1}, .colour={1, 1, 1, 1}, .material={0, 0, 1}}, {.texid=handle, .texpos={1, 0}, .pos={ 1, -1, 0}, .normal={0, 0, -1}, .colour={1, 1, 1, 1}, .material={0, 0, 1}},
{.texid=handle, .texpos={1, 1}, .pos={ 1, 1, 0, 1}, .normal={0, 0, -1}, .colour={1, 1, 1, 1}, .material={0, 0, 1}}, {.texid=handle, .texpos={1, 1}, .pos={ 1, 1, 0}, .normal={0, 0, -1}, .colour={1, 1, 1, 1}, .material={0, 0, 1}},
}; };
m.set_indices(indices, 6); m.bake_transforms();
m.set_vertices(vertices, 4);
gm_ui.bind(); gm_ui.bind();
gm_ui.set(m, GL_STATIC_DRAW); gm_ui.set(m, GL_STATIC_DRAW);
@ -73,13 +72,9 @@ void UI::render()
glUniformMatrix4fv(Shader::MAIN["camera"], 1, false, &mat_camera[0][0]); glUniformMatrix4fv(Shader::MAIN["camera"], 1, false, &mat_camera[0][0]);
gm_ui.bind(); gm_ui.bind();
gm_ui.uniform();
gm_ui.render(); gm_ui.render();
gm_dynamic_slow[gm_dynamic_slow_at].bind(); gm_dynamic_slow[gm_dynamic_slow_at].bind();
gm_dynamic_slow[gm_dynamic_slow_at].uniform();
gm_dynamic_slow[gm_dynamic_slow_at].render(); gm_dynamic_slow[gm_dynamic_slow_at].render();
w_clock.render();
} }

View File

@ -15,6 +15,7 @@
#include "../mesh/arrays.hpp" #include "../mesh/arrays.hpp"
#include "../resize.hpp" #include "../resize.hpp"
#include "../../system.hpp" #include "../../system.hpp"
#include "../../util/streams.hpp"
using namespace Sim::Graphics::Widget; using namespace Sim::Graphics::Widget;
@ -44,7 +45,3 @@ void Clock::remesh_slow(Mesh& rmesh)
rmesh.add(m, glm::translate(glm::mat4(1), glm::vec3(-wsize + glm::vec2(2, 2), 0))); rmesh.add(m, glm::translate(glm::mat4(1), glm::vec3(-wsize + glm::vec2(2, 2), 0)));
} }
void Clock::render()
{
}

View File

@ -12,7 +12,6 @@ struct Clock
void update(double dt); void update(double dt);
void remesh_slow(Mesh& rmesh); void remesh_slow(Mesh& rmesh);
void render();
}; };
}; };

View File

@ -44,13 +44,13 @@ static bool win_should_close = false;
static unsigned int ssbo_lights; static unsigned int ssbo_lights;
static unsigned int ssbo_shadow_maps; static unsigned int ssbo_shadow_maps;
static unsigned int ssbo_transforms[SSBO_TRANSFORMS_LEN]; static unsigned int ssbo_transforms[SSBO_TRANSFORMS_LEN];
static double secs_wait_at = 0; static unsigned int wait_at = 0;
static double secs_wait_now = 0;
static int gm_dynamic_slow_at = 0; static int gm_dynamic_slow_at = 0;
static int ssbo_transforms_at = 0; static int ssbo_transforms_at = 0;
static Mesh g_scene; static Mesh g_scene;
static std::vector<glm::mat4> g_scene_transforms;
static GLMesh gm_scene; static GLMesh gm_scene;
static GLMesh gm_transparent; static GLMesh gm_transparent;
@ -89,8 +89,9 @@ void remesh_static()
equipment->remesh_static(mesh); equipment->remesh_static(mesh);
} }
gm_scene.bind(); gm_scene.bind(false);
gm_scene.set(mesh, GL_STATIC_DRAW); gm_scene.set(mesh, GL_STATIC_DRAW, false);
g_scene_transforms = std::move(mesh.transforms);
std::cout << "Remeshed static\n"; std::cout << "Remeshed static\n";
} }
@ -171,10 +172,12 @@ void Window::create()
Model model("../assets", "scene.glb"); Model model("../assets", "scene.glb");
m_transparent = model.load("visual_water"); m_transparent = model.load("visual_water");
m_transparent.bake_transforms();
g_scene.add(model.load("cr")); g_scene.add(model.load("cr"));
g_scene.add(model.load("cb")); g_scene.add(model.load("cb"));
g_scene.add(model.load("hw")); g_scene.add(model.load("hw"));
g_scene.bake_transforms();
Camera::init(model); Camera::init(model);
@ -263,6 +266,8 @@ void update_slow()
equipment->remesh_slow(mesh); equipment->remesh_slow(mesh);
} }
mesh.bake_transforms();
gm_dynamic_slow[gm_dynamic_slow_at].bind(); gm_dynamic_slow[gm_dynamic_slow_at].bind();
gm_dynamic_slow[gm_dynamic_slow_at].set(mesh, GL_DYNAMIC_DRAW); gm_dynamic_slow[gm_dynamic_slow_at].set(mesh, GL_DYNAMIC_DRAW);
gm_dynamic_slow_at = (gm_dynamic_slow_at + 1) % 2; gm_dynamic_slow_at = (gm_dynamic_slow_at + 1) % 2;
@ -273,7 +278,6 @@ void update_slow()
void Window::reload() void Window::reload()
{ {
remesh_static(); remesh_static();
render_shadow_map();
} }
void Window::update(double dt) void Window::update(double dt)
@ -295,33 +299,39 @@ void Window::update(double dt)
equipment->get_static_transforms(transforms); equipment->get_static_transforms(transforms);
} }
for(int i = 0; i < transforms.size(); i++)
{
transforms[i] = g_scene_transforms[i] * transforms[i];
}
if(transforms.size() != g_scene_transforms.size())
{
std::cerr << "Transforms size mismatch! " << transforms.size() << " != " << g_scene_transforms.size() << "\n";
close();
}
UI::update(dt); UI::update(dt);
glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 3, ssbo_transforms[ssbo_transforms_at]);
ssbo_transforms_at = (ssbo_transforms_at + 1) % SSBO_TRANSFORMS_LEN;
glBindBuffer(GL_SHADER_STORAGE_BUFFER, ssbo_transforms[ssbo_transforms_at]); glBindBuffer(GL_SHADER_STORAGE_BUFFER, ssbo_transforms[ssbo_transforms_at]);
glBufferData(GL_SHADER_STORAGE_BUFFER, transforms.size() * sizeof(transforms[0]), &transforms[0], GL_DYNAMIC_DRAW); glBufferData(GL_SHADER_STORAGE_BUFFER, transforms.size() * sizeof(transforms[0]), &transforms[0], GL_DYNAMIC_DRAW);
ssbo_transforms_at = (ssbo_transforms_at + 1) % SSBO_TRANSFORMS_LEN;
secs_wait_now += dt; if(wait_at++ % 4 == 0)
if(secs_wait_now > secs_wait_at + 1.0/30.0)
{ {
secs_wait_at += 1.0/30.0;
update_slow(); update_slow();
} }
} }
void Window::render_scene() void Window::render_scene()
{ {
gm_scene.bind(); gm_scene.bind(false);
gm_scene.uniform(); glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 3, ssbo_transforms[ssbo_transforms_at]);
gm_scene.render(); gm_scene.render();
gm_dynamic_slow[gm_dynamic_slow_at].bind(); gm_dynamic_slow[gm_dynamic_slow_at].bind();
gm_dynamic_slow[gm_dynamic_slow_at].uniform();
gm_dynamic_slow[gm_dynamic_slow_at].render(); gm_dynamic_slow[gm_dynamic_slow_at].render();
gm_transparent.bind(); gm_transparent.bind();
gm_transparent.uniform();
gm_transparent.render(); gm_transparent.render();
Focus::render(); Focus::render();
@ -329,6 +339,8 @@ void Window::render_scene()
void Window::render() void Window::render()
{ {
render_shadow_map();
glm::vec<2, int> size = Resize::get_size(); glm::vec<2, int> size = Resize::get_size();
glm::vec3 camera_pos = Camera::get_pos(); glm::vec3 camera_pos = Camera::get_pos();
glm::mat4 mat_camera = Camera::get_matrix(); glm::mat4 mat_camera = Camera::get_matrix();

View File

@ -37,3 +37,17 @@ std::ostream& operator<<(std::ostream& o, const glm::vec<N, T>& v)
return o; return o;
} }
template <int N, int M, typename T>
std::ostream& operator<<(std::ostream& o, const glm::mat<N, M, T>& m)
{
o << "{\n";
for(int i = 0; i < N - 1; i++)
{
o << " " << m[i] << ",\n";
}
o << " " << m[N - 1] << "}";
return o;
}