vertex data no longer hard coded
This commit is contained in:
parent
f81d611f0e
commit
7c1dfec943
15 changed files with 132 additions and 120 deletions
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
#include <cassert>
|
#include <cassert>
|
||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
namespace xe {
|
namespace xe {
|
||||||
|
|
||||||
|
|
|
@ -35,16 +35,17 @@ std::shared_ptr<Model> Engine::loadModelFromFile(const std::string &filename) {
|
||||||
return Model::createModelFromFile(xeDevice, filename);
|
return Model::createModelFromFile(xeDevice, filename);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::shared_ptr<Model> Engine::loadModelFromData(std::vector<Model::Vertex> vertices, std::vector<uint32_t> indices) {
|
std::shared_ptr<Model> Engine::loadModelFromData(std::vector<float> vertexData, uint32_t vertexSize, std::vector<uint32_t> indices) {
|
||||||
Model::Builder builder{};
|
Model::Builder builder{};
|
||||||
builder.vertices = vertices;
|
builder.vertexData = vertexData;
|
||||||
|
builder.vertexSize = vertexSize;
|
||||||
if(indices.size() > 0) {
|
if(indices.size() > 0) {
|
||||||
builder.indices = indices;
|
builder.indices = indices;
|
||||||
}
|
}
|
||||||
return std::make_shared<Model>(xeDevice, builder);
|
return std::make_shared<Model>(xeDevice, builder);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::shared_ptr<Image> Engine::loadImage(const std::string &filename) {
|
std::shared_ptr<Image> Engine::loadImageFromFile(const std::string &filename) {
|
||||||
return std::make_shared<Image>(xeDevice, filename);
|
return std::make_shared<Image>(xeDevice, filename);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -26,8 +26,8 @@ class Engine {
|
||||||
Camera& getCamera() {return xeCamera;}
|
Camera& getCamera() {return xeCamera;}
|
||||||
|
|
||||||
std::shared_ptr<Model> loadModelFromFile(const std::string &filename);
|
std::shared_ptr<Model> loadModelFromFile(const std::string &filename);
|
||||||
std::shared_ptr<Model> loadModelFromData(std::vector<Model::Vertex> vertices, std::vector<uint32_t> indices);
|
std::shared_ptr<Model> loadModelFromData(std::vector<float> vertexData, uint32_t vertexSize, std::vector<uint32_t> indices);
|
||||||
std::shared_ptr<Image> loadImage(const std::string &filename);
|
std::shared_ptr<Image> loadImageFromFile(const std::string &filename);
|
||||||
|
|
||||||
bool beginFrame() { return xeRenderer.beginFrame(); }
|
bool beginFrame() { return xeRenderer.beginFrame(); }
|
||||||
void endFrame() { xeRenderer.endFrame(); }
|
void endFrame() { xeRenderer.endFrame(); }
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
#include "xe_model.hpp"
|
#include "xe_model.hpp"
|
||||||
#include "xe_utils.hpp"
|
|
||||||
|
|
||||||
#define TINYOBJLOADER_IMPLEMENTATION
|
#define TINYOBJLOADER_IMPLEMENTATION
|
||||||
#include "xe_obj_loader.hpp"
|
#include "xe_obj_loader.hpp"
|
||||||
|
@ -10,22 +9,12 @@
|
||||||
#include <cassert>
|
#include <cassert>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
#include <unordered_map>
|
#include <unordered_map>
|
||||||
|
#include <iostream>
|
||||||
namespace std {
|
|
||||||
template<>
|
|
||||||
struct hash<xe::Model::Vertex> {
|
|
||||||
size_t operator()(xe::Model::Vertex const &vertex) const {
|
|
||||||
size_t seed = 0;
|
|
||||||
xe::hashCombine(seed, vertex.position, vertex.normal, vertex.uv);
|
|
||||||
return seed;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace xe {
|
namespace xe {
|
||||||
|
|
||||||
Model::Model(Device &device, const Model::Builder &builder) : xeDevice{device} {
|
Model::Model(Device &device, const Model::Builder &builder) : xeDevice{device} {
|
||||||
createVertexBuffers(builder.vertices);
|
createVertexBuffers(builder.vertexData, builder.vertexSize);
|
||||||
createIndexBuffers(builder.indices);
|
createIndexBuffers(builder.indices);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -37,11 +26,10 @@ std::unique_ptr<Model> Model::createModelFromFile(Device &device, const std::str
|
||||||
return std::make_unique<Model>(device, builder);
|
return std::make_unique<Model>(device, builder);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Model::createVertexBuffers(const std::vector<Vertex> &vertices) {
|
void Model::createVertexBuffers(const std::vector<float> &vertexData, uint32_t vertexSize) {
|
||||||
vertexCount = static_cast<uint32_t>(vertices.size());
|
vertexCount = static_cast<uint32_t>(vertexData.size()) / (vertexSize / 4);
|
||||||
assert(vertexCount >= 3 && "Vertex count must be atleast 3");
|
assert(vertexCount >= 3 && "Vertex count must be atleast 3");
|
||||||
VkDeviceSize bufferSize = sizeof(vertices[0]) * vertexCount;
|
VkDeviceSize bufferSize = vertexData.size() * 4;
|
||||||
uint32_t vertexSize = sizeof(vertices[0]);
|
|
||||||
|
|
||||||
Buffer stagingBuffer {
|
Buffer stagingBuffer {
|
||||||
xeDevice,
|
xeDevice,
|
||||||
|
@ -52,7 +40,7 @@ void Model::createVertexBuffers(const std::vector<Vertex> &vertices) {
|
||||||
};
|
};
|
||||||
|
|
||||||
stagingBuffer.map();
|
stagingBuffer.map();
|
||||||
stagingBuffer.writeToBuffer((void *)vertices.data());
|
stagingBuffer.writeToBuffer((void *)vertexData.data());
|
||||||
|
|
||||||
vertexBuffer = std::make_unique<Buffer>(
|
vertexBuffer = std::make_unique<Buffer>(
|
||||||
xeDevice,
|
xeDevice,
|
||||||
|
@ -65,16 +53,16 @@ void Model::createVertexBuffers(const std::vector<Vertex> &vertices) {
|
||||||
xeDevice.copyBuffer(stagingBuffer.getBuffer(), vertexBuffer->getBuffer(), bufferSize);
|
xeDevice.copyBuffer(stagingBuffer.getBuffer(), vertexBuffer->getBuffer(), bufferSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Model::createIndexBuffers(const std::vector<uint32_t> &indices) {
|
void Model::createIndexBuffers(const std::vector<uint32_t> &indexData) {
|
||||||
indexCount = static_cast<uint32_t>(indices.size());
|
indexCount = static_cast<uint32_t>(indexData.size());
|
||||||
hasIndexBuffer = indexCount > 0;
|
hasIndexBuffer = indexCount > 0;
|
||||||
|
|
||||||
if (!hasIndexBuffer) {
|
if (!hasIndexBuffer) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
VkDeviceSize bufferSize = sizeof(indices[0]) * indexCount;
|
VkDeviceSize bufferSize = sizeof(indexData[0]) * indexCount;
|
||||||
uint32_t indexSize = sizeof(indices[0]);
|
uint32_t indexSize = sizeof(indexData[0]);
|
||||||
|
|
||||||
Buffer stagingBuffer {
|
Buffer stagingBuffer {
|
||||||
xeDevice,
|
xeDevice,
|
||||||
|
@ -85,7 +73,7 @@ void Model::createIndexBuffers(const std::vector<uint32_t> &indices) {
|
||||||
};
|
};
|
||||||
|
|
||||||
stagingBuffer.map();
|
stagingBuffer.map();
|
||||||
stagingBuffer.writeToBuffer((void *)indices.data());
|
stagingBuffer.writeToBuffer((void *)indexData.data());
|
||||||
|
|
||||||
indexBuffer = std::make_unique<Buffer>(
|
indexBuffer = std::make_unique<Buffer>(
|
||||||
xeDevice,
|
xeDevice,
|
||||||
|
@ -116,25 +104,6 @@ void Model::draw(VkCommandBuffer commandBuffer) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<VkVertexInputBindingDescription> Model::Vertex::getBindingDescriptions() {
|
|
||||||
std::vector<VkVertexInputBindingDescription> bindingDescriptions(1);
|
|
||||||
bindingDescriptions[0].binding = 0;
|
|
||||||
bindingDescriptions[0].stride = sizeof(Vertex);
|
|
||||||
bindingDescriptions[0].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
|
|
||||||
return bindingDescriptions;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<VkVertexInputAttributeDescription> Model::Vertex::getAttributeDescriptions() {
|
|
||||||
std::vector<VkVertexInputAttributeDescription> attributeDescptions{};
|
|
||||||
|
|
||||||
attributeDescptions.push_back({0, 0, VK_FORMAT_R32G32B32_SFLOAT, offsetof(Vertex, position)});
|
|
||||||
attributeDescptions.push_back({1, 0, VK_FORMAT_R32G32B32_SFLOAT, offsetof(Vertex, color)});
|
|
||||||
attributeDescptions.push_back({2, 0, VK_FORMAT_R32G32B32_SFLOAT, offsetof(Vertex, normal)});
|
|
||||||
attributeDescptions.push_back({3, 0, VK_FORMAT_R32G32_SFLOAT, offsetof(Vertex, uv)});
|
|
||||||
|
|
||||||
return attributeDescptions;
|
|
||||||
}
|
|
||||||
|
|
||||||
void Model::Builder::loadModel(const std::string &filepath) {
|
void Model::Builder::loadModel(const std::string &filepath) {
|
||||||
tinyobj::attrib_t attrib;
|
tinyobj::attrib_t attrib;
|
||||||
std::vector<tinyobj::shape_t> shapes;
|
std::vector<tinyobj::shape_t> shapes;
|
||||||
|
@ -144,50 +113,52 @@ void Model::Builder::loadModel(const std::string &filepath) {
|
||||||
throw std::runtime_error(warn + err);
|
throw std::runtime_error(warn + err);
|
||||||
}
|
}
|
||||||
|
|
||||||
vertices.clear();
|
vertexData.clear();
|
||||||
indices.clear();
|
indices.clear();
|
||||||
|
vertexSize = 0;
|
||||||
|
|
||||||
|
bool vertex, color, normal, uvs;
|
||||||
|
|
||||||
std::unordered_map<Vertex, uint32_t> uniqueVertices{};
|
|
||||||
for (const auto &shape : shapes) {
|
for (const auto &shape : shapes) {
|
||||||
for (const auto &index : shape.mesh.indices) {
|
for (const auto &index : shape.mesh.indices) {
|
||||||
Vertex vertex{};
|
|
||||||
|
|
||||||
if(index.vertex_index >= 0) {
|
if(index.vertex_index >= 0) {
|
||||||
vertex.position = {
|
vertexData.push_back(attrib.vertices[3 * index.vertex_index + 0]);
|
||||||
attrib.vertices[3 * index.vertex_index + 0],
|
vertexData.push_back(attrib.vertices[3 * index.vertex_index + 1]);
|
||||||
attrib.vertices[3 * index.vertex_index + 1],
|
vertexData.push_back(attrib.vertices[3 * index.vertex_index + 2]);
|
||||||
attrib.vertices[3 * index.vertex_index + 2]
|
vertex = true;
|
||||||
};
|
|
||||||
|
|
||||||
vertex.color = {
|
vertexData.push_back(attrib.colors[3 * index.vertex_index + 0]);
|
||||||
attrib.colors[3 * index.vertex_index + 0],
|
vertexData.push_back(attrib.colors[3 * index.vertex_index + 1]);
|
||||||
attrib.colors[3 * index.vertex_index + 1],
|
vertexData.push_back(attrib.colors[3 * index.vertex_index + 2]);
|
||||||
attrib.colors[3 * index.vertex_index + 2]
|
color = true;
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if(index.normal_index >= 0) {
|
if(index.normal_index >= 0) {
|
||||||
vertex.normal = {
|
vertexData.push_back(attrib.normals[3 * index.normal_index + 0]);
|
||||||
attrib.normals[3 * index.normal_index + 0],
|
vertexData.push_back(attrib.normals[3 * index.normal_index + 1]);
|
||||||
attrib.normals[3 * index.normal_index + 1],
|
vertexData.push_back(attrib.normals[3 * index.normal_index + 2]);
|
||||||
attrib.normals[3 * index.normal_index + 2]
|
normal = true;
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if(index.texcoord_index >= 0) {
|
if(index.texcoord_index >= 0) {
|
||||||
vertex.uv = {
|
vertexData.push_back(attrib.texcoords[2 * index.texcoord_index + 0]);
|
||||||
attrib.texcoords[2 * index.texcoord_index + 0],
|
vertexData.push_back(attrib.texcoords[2 * index.texcoord_index + 1]);
|
||||||
attrib.texcoords[2 * index.texcoord_index + 1],
|
uvs = true;
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (uniqueVertices.count(vertex) == 0) {
|
|
||||||
uniqueVertices[vertex] = static_cast<uint32_t>(vertices.size());
|
|
||||||
vertices.push_back(vertex);
|
|
||||||
}
|
|
||||||
indices.push_back(uniqueVertices[vertex]);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(vertex)
|
||||||
|
vertexSize += 12;
|
||||||
|
if(color)
|
||||||
|
vertexSize += 12;
|
||||||
|
if(normal)
|
||||||
|
vertexSize += 12;
|
||||||
|
if(uvs)
|
||||||
|
vertexSize += 8;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -14,22 +14,10 @@ namespace xe {
|
||||||
class Model {
|
class Model {
|
||||||
public:
|
public:
|
||||||
|
|
||||||
struct Vertex {
|
|
||||||
glm::vec3 position;
|
|
||||||
glm::vec3 color;
|
|
||||||
glm::vec3 normal;
|
|
||||||
glm::vec2 uv;
|
|
||||||
|
|
||||||
static std::vector<VkVertexInputBindingDescription> getBindingDescriptions();
|
|
||||||
static std::vector<VkVertexInputAttributeDescription> getAttributeDescriptions();
|
|
||||||
|
|
||||||
bool operator==(const Vertex &other) const {
|
|
||||||
return position == other.position && color == other.color && normal == other.normal && uv == other.uv;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct Builder {
|
struct Builder {
|
||||||
std::vector<Vertex> vertices{};
|
std::vector<float> vertexData{};
|
||||||
|
uint32_t vertexSize;
|
||||||
|
|
||||||
std::vector<uint32_t> indices{};
|
std::vector<uint32_t> indices{};
|
||||||
|
|
||||||
void loadModel(const std::string &filepath);
|
void loadModel(const std::string &filepath);
|
||||||
|
@ -47,8 +35,8 @@ class Model {
|
||||||
void draw(VkCommandBuffer commandBuffer);
|
void draw(VkCommandBuffer commandBuffer);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void createVertexBuffers(const std::vector<Vertex> &vertices);
|
void createVertexBuffers(const std::vector<float> &vertexData, uint32_t vertexSize);
|
||||||
void createIndexBuffers(const std::vector<uint32_t> &indices);
|
void createIndexBuffers(const std::vector<uint32_t> &indexData);
|
||||||
|
|
||||||
Device &xeDevice;
|
Device &xeDevice;
|
||||||
|
|
||||||
|
|
|
@ -16,9 +16,11 @@ namespace xe {
|
||||||
Device &device,
|
Device &device,
|
||||||
const std::string& vertFilepath,
|
const std::string& vertFilepath,
|
||||||
const std::string& fragFilepath,
|
const std::string& fragFilepath,
|
||||||
const PipelineConfigInfo& configInfo)
|
const PipelineConfigInfo& configInfo,
|
||||||
: xeDevice{device} {
|
std::vector<VkVertexInputAttributeDescription> &attributeDescptions,
|
||||||
createGraphicsPipeline(vertFilepath, fragFilepath, configInfo);
|
uint32_t vertexSize
|
||||||
|
) : xeDevice{device} {
|
||||||
|
createGraphicsPipeline(vertFilepath, fragFilepath, configInfo, attributeDescptions, vertexSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
Pipeline::~Pipeline() {
|
Pipeline::~Pipeline() {
|
||||||
|
@ -48,7 +50,10 @@ namespace xe {
|
||||||
void Pipeline::createGraphicsPipeline(
|
void Pipeline::createGraphicsPipeline(
|
||||||
const std::string& vertFilePath,
|
const std::string& vertFilePath,
|
||||||
const std::string& fragFilepath,
|
const std::string& fragFilepath,
|
||||||
const PipelineConfigInfo& configInfo) {
|
const PipelineConfigInfo& configInfo,
|
||||||
|
std::vector<VkVertexInputAttributeDescription> &attributeDescptions,
|
||||||
|
uint32_t vertexSize
|
||||||
|
) {
|
||||||
|
|
||||||
assert(
|
assert(
|
||||||
configInfo.pipelineLayout != VK_NULL_HANDLE &&
|
configInfo.pipelineLayout != VK_NULL_HANDLE &&
|
||||||
|
@ -78,8 +83,13 @@ namespace xe {
|
||||||
shaderStages[1].pNext = nullptr;
|
shaderStages[1].pNext = nullptr;
|
||||||
shaderStages[1].pSpecializationInfo = nullptr;
|
shaderStages[1].pSpecializationInfo = nullptr;
|
||||||
|
|
||||||
auto bindingDescriptions = Model::Vertex::getBindingDescriptions();
|
VkVertexInputBindingDescription bindingDescription;
|
||||||
auto attributeDescptions = Model::Vertex::getAttributeDescriptions();
|
bindingDescription.binding = 0;
|
||||||
|
bindingDescription.stride = vertexSize;
|
||||||
|
bindingDescription.inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
|
||||||
|
|
||||||
|
std::vector<VkVertexInputBindingDescription> bindingDescriptions{bindingDescription};
|
||||||
|
|
||||||
VkPipelineVertexInputStateCreateInfo vertexInputInfo{};
|
VkPipelineVertexInputStateCreateInfo vertexInputInfo{};
|
||||||
vertexInputInfo.sType = VK_STRUCTURE_TYPE_PIPELINE_VERTEX_INPUT_STATE_CREATE_INFO;
|
vertexInputInfo.sType = VK_STRUCTURE_TYPE_PIPELINE_VERTEX_INPUT_STATE_CREATE_INFO;
|
||||||
vertexInputInfo.vertexAttributeDescriptionCount = static_cast<uint32_t>(attributeDescptions.size());
|
vertexInputInfo.vertexAttributeDescriptionCount = static_cast<uint32_t>(attributeDescptions.size());
|
||||||
|
|
|
@ -33,7 +33,10 @@ class Pipeline {
|
||||||
Device &device,
|
Device &device,
|
||||||
const std::string& vertFilepath,
|
const std::string& vertFilepath,
|
||||||
const std::string& fragFilepath,
|
const std::string& fragFilepath,
|
||||||
const PipelineConfigInfo& configInfo);
|
const PipelineConfigInfo& configInfo,
|
||||||
|
std::vector<VkVertexInputAttributeDescription> &attributeDescptions,
|
||||||
|
uint32_t vertexSize
|
||||||
|
);
|
||||||
~Pipeline();
|
~Pipeline();
|
||||||
|
|
||||||
Pipeline(const Pipeline&) = delete;
|
Pipeline(const Pipeline&) = delete;
|
||||||
|
@ -48,7 +51,10 @@ class Pipeline {
|
||||||
void createGraphicsPipeline(
|
void createGraphicsPipeline(
|
||||||
const std::string& vertFilePath,
|
const std::string& vertFilePath,
|
||||||
const std::string& fragFilepath,
|
const std::string& fragFilepath,
|
||||||
const PipelineConfigInfo& configInfo);
|
const PipelineConfigInfo& configInfo,
|
||||||
|
std::vector<VkVertexInputAttributeDescription> &attributeDescptions,
|
||||||
|
uint32_t vertexSize
|
||||||
|
);
|
||||||
|
|
||||||
void createShaderModule(const std::vector<char>& code, VkShaderModule* shaderModule);
|
void createShaderModule(const std::vector<char>& code, VkShaderModule* shaderModule);
|
||||||
|
|
||||||
|
|
|
@ -23,7 +23,9 @@ RenderSystem::RenderSystem(
|
||||||
std::map<uint32_t, uint32_t> uniformBindings,
|
std::map<uint32_t, uint32_t> uniformBindings,
|
||||||
std::map<uint32_t, Image*> imageBindings,
|
std::map<uint32_t, Image*> imageBindings,
|
||||||
uint32_t pushCunstantDataSize,
|
uint32_t pushCunstantDataSize,
|
||||||
bool cullingEnabled
|
bool cullingEnabled,
|
||||||
|
std::vector<VkVertexInputAttributeDescription> attributeDescptions,
|
||||||
|
uint32_t vertexSize
|
||||||
) : xeDevice{xeEngine.xeDevice},
|
) : xeDevice{xeEngine.xeDevice},
|
||||||
xeRenderer{xeEngine.xeRenderer},
|
xeRenderer{xeEngine.xeRenderer},
|
||||||
xeDescriptorPool{xeEngine.xeDescriptorPool},
|
xeDescriptorPool{xeEngine.xeDescriptorPool},
|
||||||
|
@ -35,10 +37,9 @@ RenderSystem::RenderSystem(
|
||||||
createUniformBuffers();
|
createUniformBuffers();
|
||||||
createDescriptorSets();
|
createDescriptorSets();
|
||||||
createPipelineLayout();
|
createPipelineLayout();
|
||||||
createPipeline(xeRenderer.getSwapChainRenderPass(), vert, frag, cullingEnabled);
|
createPipeline(xeRenderer.getSwapChainRenderPass(), vert, frag, cullingEnabled, attributeDescptions, vertexSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
RenderSystem::~RenderSystem() {
|
RenderSystem::~RenderSystem() {
|
||||||
vkDestroyPipelineLayout(xeDevice.device(), pipelineLayout, nullptr);
|
vkDestroyPipelineLayout(xeDevice.device(), pipelineLayout, nullptr);
|
||||||
vkDestroySampler(xeDevice.device(), textureSampler, nullptr);
|
vkDestroySampler(xeDevice.device(), textureSampler, nullptr);
|
||||||
|
@ -164,7 +165,7 @@ void RenderSystem::createPipelineLayout() {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void RenderSystem::createPipeline(VkRenderPass renderPass, std::string vert, std::string frag, bool cullingEnabled) {
|
void RenderSystem::createPipeline(VkRenderPass renderPass, std::string vert, std::string frag, bool cullingEnabled, std::vector<VkVertexInputAttributeDescription> attributeDescptions, uint32_t vertexSize) {
|
||||||
assert(pipelineLayout != nullptr && "Cannot create pipeline before pipeline layout");
|
assert(pipelineLayout != nullptr && "Cannot create pipeline before pipeline layout");
|
||||||
|
|
||||||
PipelineConfigInfo pipelineConfig{};
|
PipelineConfigInfo pipelineConfig{};
|
||||||
|
@ -178,7 +179,9 @@ void RenderSystem::createPipeline(VkRenderPass renderPass, std::string vert, std
|
||||||
xeDevice,
|
xeDevice,
|
||||||
vert,
|
vert,
|
||||||
frag,
|
frag,
|
||||||
pipelineConfig
|
pipelineConfig,
|
||||||
|
attributeDescptions,
|
||||||
|
vertexSize
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -11,6 +11,7 @@
|
||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <map>
|
#include <map>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
namespace xe {
|
namespace xe {
|
||||||
|
|
||||||
|
@ -21,6 +22,21 @@ class RenderSystem {
|
||||||
public:
|
public:
|
||||||
Builder(Engine &xeEngine, std::string vert, std::string frag) : xeEngine{xeEngine}, vert{vert}, frag{frag} {}
|
Builder(Engine &xeEngine, std::string vert, std::string frag) : xeEngine{xeEngine}, vert{vert}, frag{frag} {}
|
||||||
|
|
||||||
|
Builder& addVertexBinding(uint32_t binding, uint32_t dimension, uint32_t offset){
|
||||||
|
if(dimension == 1)
|
||||||
|
attributeDescptions.push_back({binding, 0, VK_FORMAT_R32_SFLOAT, offset});
|
||||||
|
if(dimension == 2)
|
||||||
|
attributeDescptions.push_back({binding, 0, VK_FORMAT_R32G32_SFLOAT, offset});
|
||||||
|
if(dimension == 3)
|
||||||
|
attributeDescptions.push_back({binding, 0, VK_FORMAT_R32G32B32_SFLOAT, offset});
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
Builder& setVertexSize(uint32_t size) {
|
||||||
|
vertexSize = size;
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
Builder& addPushConstant(uint32_t size) {
|
Builder& addPushConstant(uint32_t size) {
|
||||||
pushCunstantDataSize = size;
|
pushCunstantDataSize = size;
|
||||||
return *this;
|
return *this;
|
||||||
|
@ -42,7 +58,7 @@ class RenderSystem {
|
||||||
}
|
}
|
||||||
|
|
||||||
std::unique_ptr<RenderSystem> build() {
|
std::unique_ptr<RenderSystem> build() {
|
||||||
return std::make_unique<RenderSystem>(xeEngine, std::move(vert), std::move(frag), std::move(uniformBindings), std::move(imageBindings), std::move(pushCunstantDataSize), std::move(cullingEnabled));
|
return std::make_unique<RenderSystem>(xeEngine, std::move(vert), std::move(frag), std::move(uniformBindings), std::move(imageBindings), std::move(pushCunstantDataSize), std::move(cullingEnabled), std::move(attributeDescptions), std::move(vertexSize));
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -51,6 +67,9 @@ class RenderSystem {
|
||||||
std::map<uint32_t, Image*> imageBindings{};
|
std::map<uint32_t, Image*> imageBindings{};
|
||||||
uint32_t pushCunstantDataSize{0};
|
uint32_t pushCunstantDataSize{0};
|
||||||
|
|
||||||
|
std::vector<VkVertexInputAttributeDescription> attributeDescptions{};
|
||||||
|
uint32_t vertexSize;
|
||||||
|
|
||||||
std::string vert;
|
std::string vert;
|
||||||
std::string frag;
|
std::string frag;
|
||||||
|
|
||||||
|
@ -66,7 +85,9 @@ class RenderSystem {
|
||||||
std::map<uint32_t, uint32_t> uniformBindings,
|
std::map<uint32_t, uint32_t> uniformBindings,
|
||||||
std::map<uint32_t, Image*> imageBindings,
|
std::map<uint32_t, Image*> imageBindings,
|
||||||
uint32_t pushCunstantDataSize,
|
uint32_t pushCunstantDataSize,
|
||||||
bool cullingEnabled
|
bool cullingEnabled,
|
||||||
|
std::vector<VkVertexInputAttributeDescription> attributeDescptions,
|
||||||
|
uint32_t vertexSize
|
||||||
);
|
);
|
||||||
|
|
||||||
~RenderSystem();
|
~RenderSystem();
|
||||||
|
@ -89,7 +110,7 @@ class RenderSystem {
|
||||||
void createDescriptorSets();
|
void createDescriptorSets();
|
||||||
void updateDescriptorSet(int frameIndex, bool allocate);
|
void updateDescriptorSet(int frameIndex, bool allocate);
|
||||||
void createPipelineLayout();
|
void createPipelineLayout();
|
||||||
void createPipeline(VkRenderPass renderPass, std::string vert, std::string frag, bool cullingEnabled);
|
void createPipeline(VkRenderPass renderPass, std::string vert, std::string frag, bool cullingEnabled, std::vector<VkVertexInputAttributeDescription> attributeDescptions, uint32_t vertexSize);
|
||||||
|
|
||||||
bool boundPipeline{false};
|
bool boundPipeline{false};
|
||||||
bool boundDescriptor{false};
|
bool boundDescriptor{false};
|
||||||
|
|
|
@ -1,11 +0,0 @@
|
||||||
#pragma once
|
|
||||||
|
|
||||||
namespace xe {
|
|
||||||
|
|
||||||
template <typename T, typename... Rest>
|
|
||||||
void hashCombine(std::size_t& seed, const T& v, const Rest&... rest) {
|
|
||||||
seed ^= std::hash<T>{}(v) + 0x9e3779b9 + (seed << 6) + (seed >> 2);
|
|
||||||
(hashCombine(seed, rest), ...);
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
|
@ -10,6 +10,11 @@ layout (binding = 0) uniform GlobalUbo {
|
||||||
vec3 directionToLight;
|
vec3 directionToLight;
|
||||||
} ubo;
|
} ubo;
|
||||||
|
|
||||||
|
layout (binding = 2) uniform Deez {
|
||||||
|
mat4 projectionViewMatrix;
|
||||||
|
vec3 directionToLight;
|
||||||
|
} deez;
|
||||||
|
|
||||||
layout (binding = 1) uniform sampler2D texSampler;
|
layout (binding = 1) uniform sampler2D texSampler;
|
||||||
|
|
||||||
layout(push_constant) uniform Push {
|
layout(push_constant) uniform Push {
|
||||||
|
|
|
@ -13,6 +13,11 @@ layout (binding = 0) uniform GlobalUbo {
|
||||||
vec3 directionToLight;
|
vec3 directionToLight;
|
||||||
} ubo;
|
} ubo;
|
||||||
|
|
||||||
|
layout (binding = 2) uniform Deez {
|
||||||
|
mat4 projectionViewMatrix;
|
||||||
|
vec3 directionToLight;
|
||||||
|
} deez;
|
||||||
|
|
||||||
layout (push_constant) uniform Push {
|
layout (push_constant) uniform Push {
|
||||||
mat4 modelMatrix;
|
mat4 modelMatrix;
|
||||||
mat4 normalMatrix;
|
mat4 normalMatrix;
|
||||||
|
|
|
@ -27,8 +27,8 @@ FirstApp::~FirstApp() {}
|
||||||
|
|
||||||
void FirstApp::run() {
|
void FirstApp::run() {
|
||||||
|
|
||||||
std::shared_ptr<xe::Image> image = xeEngine.loadImage("res/image/texture.png");
|
std::shared_ptr<xe::Image> image = xeEngine.loadImageFromFile("res/image/texture.png");
|
||||||
std::shared_ptr<xe::Image> image2 = xeEngine.loadImage("res/image/ltt.");
|
std::shared_ptr<xe::Image> image2 = xeEngine.loadImageFromFile("res/image/scaly.png");
|
||||||
|
|
||||||
SimpleRenderer renderer{xeEngine, image.get()};
|
SimpleRenderer renderer{xeEngine, image.get()};
|
||||||
|
|
||||||
|
|
|
@ -7,6 +7,11 @@ namespace app {
|
||||||
|
|
||||||
SimpleRenderer::SimpleRenderer(xe::Engine &xeEngine, xe::Image *xeImage) {
|
SimpleRenderer::SimpleRenderer(xe::Engine &xeEngine, xe::Image *xeImage) {
|
||||||
xeRenderSystem = xe::RenderSystem::Builder(xeEngine, "res/shaders/simple_shader.vert.spv", "res/shaders/simple_shader.frag.spv")
|
xeRenderSystem = xe::RenderSystem::Builder(xeEngine, "res/shaders/simple_shader.vert.spv", "res/shaders/simple_shader.frag.spv")
|
||||||
|
.addVertexBinding(0, 3, 0)
|
||||||
|
.addVertexBinding(1, 3, 12)
|
||||||
|
.addVertexBinding(2, 3, 24)
|
||||||
|
.addVertexBinding(3, 2, 36)
|
||||||
|
.setVertexSize(sizeof(Vertex))
|
||||||
.addPushConstant(sizeof(PushConstant))
|
.addPushConstant(sizeof(PushConstant))
|
||||||
.addUniformBinding(0, sizeof(UniformBuffer))
|
.addUniformBinding(0, sizeof(UniformBuffer))
|
||||||
.addTextureBinding(1, xeImage)
|
.addTextureBinding(1, xeImage)
|
||||||
|
|
|
@ -15,6 +15,13 @@ struct PushConstant {
|
||||||
alignas(16) glm::mat4 normalMatrix{1.f};
|
alignas(16) glm::mat4 normalMatrix{1.f};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct Vertex {
|
||||||
|
glm::vec3 position;
|
||||||
|
glm::vec3 color;
|
||||||
|
glm::vec3 normal;
|
||||||
|
glm::vec2 uv;
|
||||||
|
};
|
||||||
|
|
||||||
class SimpleRenderer {
|
class SimpleRenderer {
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
Loading…
Reference in a new issue