|
|
|
@@ -3,6 +3,7 @@
|
|
|
|
|
#define GLFW_INCLUDE_VULKAN
|
|
|
|
|
#include <GLFW/glfw3.h>
|
|
|
|
|
|
|
|
|
|
#include <fstream>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <iostream>
|
|
|
|
|
#include <stdexcept>
|
|
|
|
@@ -23,6 +24,23 @@ const std::vector<const char*> validationLayers = {
|
|
|
|
|
const bool enableValidationLayers = true;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
static std::vector<char> readFile(const std::string& filename) {
|
|
|
|
|
std::ifstream file(filename, std::ios::ate | std::ios::binary);
|
|
|
|
|
|
|
|
|
|
if (!file.is_open()) {
|
|
|
|
|
throw std::runtime_error("failed to open file!");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
size_t fileSize = (size_t) file.tellg();
|
|
|
|
|
std::vector<char> buffer(fileSize);
|
|
|
|
|
|
|
|
|
|
file.seekg(0);
|
|
|
|
|
file.read(buffer.data(), fileSize);
|
|
|
|
|
file.close();
|
|
|
|
|
|
|
|
|
|
return buffer;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class HelloTriangleApplication {
|
|
|
|
|
public:
|
|
|
|
|
void run() {
|
|
|
|
@@ -55,6 +73,13 @@ private:
|
|
|
|
|
const uint32_t WIDTH = 800;
|
|
|
|
|
const uint32_t HEIGHT = 600;
|
|
|
|
|
|
|
|
|
|
VkPipelineLayout pipelineLayout;
|
|
|
|
|
|
|
|
|
|
std::vector<VkDynamicState> dynamicStates = {
|
|
|
|
|
VK_DYNAMIC_STATE_VIEWPORT,
|
|
|
|
|
VK_DYNAMIC_STATE_SCISSOR
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void initWindow() {
|
|
|
|
|
glfwInit();
|
|
|
|
|
glfwWindowHint(GLFW_CLIENT_API, GLFW_NO_API);
|
|
|
|
@@ -70,6 +95,139 @@ private:
|
|
|
|
|
createLogicalDevice();
|
|
|
|
|
createSwapChain();
|
|
|
|
|
createImageViews();
|
|
|
|
|
|
|
|
|
|
createGraphicsPipeline();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void createGraphicsPipeline() {
|
|
|
|
|
auto vertShaderCode = readFile("shaders/vert.spv");
|
|
|
|
|
auto fragShaderCode = readFile("shaders/frag.spv");
|
|
|
|
|
|
|
|
|
|
VkShaderModule vertShaderModule = createShaderModule(vertShaderCode);
|
|
|
|
|
VkShaderModule fragShaderModule = createShaderModule(fragShaderCode);
|
|
|
|
|
|
|
|
|
|
VkPipelineShaderStageCreateInfo vertShaderStageInfo{};
|
|
|
|
|
vertShaderStageInfo.sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO;
|
|
|
|
|
vertShaderStageInfo.stage = VK_SHADER_STAGE_VERTEX_BIT;
|
|
|
|
|
vertShaderStageInfo.module = vertShaderModule;
|
|
|
|
|
vertShaderStageInfo.pName = "main";
|
|
|
|
|
|
|
|
|
|
VkPipelineShaderStageCreateInfo fragShaderStageInfo{};
|
|
|
|
|
fragShaderStageInfo.sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO;
|
|
|
|
|
fragShaderStageInfo.stage = VK_SHADER_STAGE_FRAGMENT_BIT;
|
|
|
|
|
fragShaderStageInfo.module = fragShaderModule;
|
|
|
|
|
fragShaderStageInfo.pName = "main";
|
|
|
|
|
|
|
|
|
|
VkPipelineShaderStageCreateInfo shaderStages[] = {vertShaderStageInfo, fragShaderStageInfo};
|
|
|
|
|
|
|
|
|
|
VkPipelineVertexInputStateCreateInfo vertexInputInfo{};
|
|
|
|
|
vertexInputInfo.sType = VK_STRUCTURE_TYPE_PIPELINE_VERTEX_INPUT_STATE_CREATE_INFO;
|
|
|
|
|
vertexInputInfo.vertexBindingDescriptionCount = 0;
|
|
|
|
|
vertexInputInfo.pVertexBindingDescriptions = nullptr; // Optional
|
|
|
|
|
vertexInputInfo.vertexAttributeDescriptionCount = 0;
|
|
|
|
|
vertexInputInfo.pVertexAttributeDescriptions = nullptr; // Optional
|
|
|
|
|
|
|
|
|
|
VkPipelineDynamicStateCreateInfo dynamicState{};
|
|
|
|
|
dynamicState.sType = VK_STRUCTURE_TYPE_PIPELINE_DYNAMIC_STATE_CREATE_INFO;
|
|
|
|
|
dynamicState.dynamicStateCount = static_cast<uint32_t>(dynamicStates.size());
|
|
|
|
|
dynamicState.pDynamicStates = dynamicStates.data();
|
|
|
|
|
|
|
|
|
|
VkPipelineInputAssemblyStateCreateInfo inputAssembly{};
|
|
|
|
|
inputAssembly.sType = VK_STRUCTURE_TYPE_PIPELINE_INPUT_ASSEMBLY_STATE_CREATE_INFO;
|
|
|
|
|
inputAssembly.topology = VK_PRIMITIVE_TOPOLOGY_TRIANGLE_LIST;
|
|
|
|
|
inputAssembly.primitiveRestartEnable = VK_FALSE;
|
|
|
|
|
|
|
|
|
|
VkViewport viewport{};
|
|
|
|
|
viewport.x = 0.0f;
|
|
|
|
|
viewport.y = 0.0f;
|
|
|
|
|
viewport.width = (float) swapChainExtent.width;
|
|
|
|
|
viewport.height = (float) swapChainExtent.height;
|
|
|
|
|
viewport.minDepth = 0.0f;
|
|
|
|
|
viewport.maxDepth = 1.0f;
|
|
|
|
|
|
|
|
|
|
VkRect2D scissor{};
|
|
|
|
|
scissor.offset = {0, 0};
|
|
|
|
|
scissor.extent = swapChainExtent;
|
|
|
|
|
|
|
|
|
|
VkPipelineViewportStateCreateInfo viewportState{};
|
|
|
|
|
viewportState.sType = VK_STRUCTURE_TYPE_PIPELINE_VIEWPORT_STATE_CREATE_INFO;
|
|
|
|
|
viewportState.viewportCount = 1;
|
|
|
|
|
viewportState.pViewports = &viewport;
|
|
|
|
|
viewportState.scissorCount = 1;
|
|
|
|
|
viewportState.pScissors = &scissor;
|
|
|
|
|
|
|
|
|
|
VkPipelineRasterizationStateCreateInfo rasterizer{};
|
|
|
|
|
rasterizer.sType = VK_STRUCTURE_TYPE_PIPELINE_RASTERIZATION_STATE_CREATE_INFO;
|
|
|
|
|
rasterizer.depthClampEnable = VK_FALSE;
|
|
|
|
|
rasterizer.rasterizerDiscardEnable = VK_FALSE;
|
|
|
|
|
rasterizer.polygonMode = VK_POLYGON_MODE_FILL;
|
|
|
|
|
rasterizer.lineWidth = 1.0f;
|
|
|
|
|
rasterizer.cullMode = VK_CULL_MODE_BACK_BIT;
|
|
|
|
|
rasterizer.frontFace = VK_FRONT_FACE_CLOCKWISE;
|
|
|
|
|
rasterizer.depthBiasEnable = VK_FALSE;
|
|
|
|
|
rasterizer.depthBiasConstantFactor = 0.0f; // Optional
|
|
|
|
|
rasterizer.depthBiasClamp = 0.0f; // Optional
|
|
|
|
|
rasterizer.depthBiasSlopeFactor = 0.0f; // Optional
|
|
|
|
|
|
|
|
|
|
VkPipelineMultisampleStateCreateInfo multisampling{};
|
|
|
|
|
multisampling.sType = VK_STRUCTURE_TYPE_PIPELINE_MULTISAMPLE_STATE_CREATE_INFO;
|
|
|
|
|
multisampling.sampleShadingEnable = VK_FALSE;
|
|
|
|
|
multisampling.rasterizationSamples = VK_SAMPLE_COUNT_1_BIT;
|
|
|
|
|
multisampling.minSampleShading = 1.0f; // Optional
|
|
|
|
|
multisampling.pSampleMask = nullptr; // Optional
|
|
|
|
|
multisampling.alphaToCoverageEnable = VK_FALSE; // Optional
|
|
|
|
|
multisampling.alphaToOneEnable = VK_FALSE; // Optional
|
|
|
|
|
|
|
|
|
|
VkPipelineColorBlendAttachmentState colorBlendAttachment{};
|
|
|
|
|
colorBlendAttachment.colorWriteMask = VK_COLOR_COMPONENT_R_BIT | VK_COLOR_COMPONENT_G_BIT | VK_COLOR_COMPONENT_B_BIT | VK_COLOR_COMPONENT_A_BIT;
|
|
|
|
|
colorBlendAttachment.blendEnable = VK_FALSE;
|
|
|
|
|
colorBlendAttachment.srcColorBlendFactor = VK_BLEND_FACTOR_ONE; // Optional
|
|
|
|
|
colorBlendAttachment.dstColorBlendFactor = VK_BLEND_FACTOR_ZERO; // Optional
|
|
|
|
|
colorBlendAttachment.colorBlendOp = VK_BLEND_OP_ADD; // Optional
|
|
|
|
|
colorBlendAttachment.srcAlphaBlendFactor = VK_BLEND_FACTOR_ONE; // Optional
|
|
|
|
|
colorBlendAttachment.dstAlphaBlendFactor = VK_BLEND_FACTOR_ZERO; // Optional
|
|
|
|
|
colorBlendAttachment.alphaBlendOp = VK_BLEND_OP_ADD; // Optional
|
|
|
|
|
|
|
|
|
|
VkPipelineColorBlendStateCreateInfo colorBlending{};
|
|
|
|
|
colorBlending.sType = VK_STRUCTURE_TYPE_PIPELINE_COLOR_BLEND_STATE_CREATE_INFO;
|
|
|
|
|
colorBlending.logicOpEnable = VK_FALSE;
|
|
|
|
|
colorBlending.logicOp = VK_LOGIC_OP_COPY; // Optional
|
|
|
|
|
colorBlending.attachmentCount = 1;
|
|
|
|
|
colorBlending.pAttachments = &colorBlendAttachment;
|
|
|
|
|
colorBlending.blendConstants[0] = 0.0f; // Optional
|
|
|
|
|
colorBlending.blendConstants[1] = 0.0f; // Optional
|
|
|
|
|
colorBlending.blendConstants[2] = 0.0f; // Optional
|
|
|
|
|
colorBlending.blendConstants[3] = 0.0f; // Optional
|
|
|
|
|
|
|
|
|
|
VkPipelineLayoutCreateInfo pipelineLayoutInfo{};
|
|
|
|
|
pipelineLayoutInfo.sType = VK_STRUCTURE_TYPE_PIPELINE_LAYOUT_CREATE_INFO;
|
|
|
|
|
pipelineLayoutInfo.setLayoutCount = 0; // Optional
|
|
|
|
|
pipelineLayoutInfo.pSetLayouts = nullptr; // Optional
|
|
|
|
|
pipelineLayoutInfo.pushConstantRangeCount = 0; // Optional
|
|
|
|
|
pipelineLayoutInfo.pPushConstantRanges = nullptr; // Optional
|
|
|
|
|
|
|
|
|
|
if (vkCreatePipelineLayout(device, &pipelineLayoutInfo, nullptr, &pipelineLayout) != VK_SUCCESS) {
|
|
|
|
|
throw std::runtime_error("failed to create pipeline layout!");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
vkDestroyShaderModule(device, fragShaderModule, nullptr);
|
|
|
|
|
vkDestroyShaderModule(device, vertShaderModule, nullptr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
VkShaderModule createShaderModule(const std::vector<char>& code) {
|
|
|
|
|
VkShaderModuleCreateInfo createInfo{};
|
|
|
|
|
createInfo.sType = VK_STRUCTURE_TYPE_SHADER_MODULE_CREATE_INFO;
|
|
|
|
|
createInfo.codeSize = code.size();
|
|
|
|
|
createInfo.pCode = reinterpret_cast<const uint32_t*>(code.data());
|
|
|
|
|
|
|
|
|
|
VkShaderModule shaderModule;
|
|
|
|
|
VkResult result = vkCreateShaderModule(device, &createInfo, nullptr, &shaderModule);
|
|
|
|
|
if (result != VK_SUCCESS) {
|
|
|
|
|
std::cout << "create shader module error: " << result << std::endl;
|
|
|
|
|
throw std::runtime_error("failed to create shader module!");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return shaderModule;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void createSurface() {
|
|
|
|
@@ -491,6 +649,7 @@ private:
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void cleanup() {
|
|
|
|
|
vkDestroyPipelineLayout(device, pipelineLayout, nullptr);
|
|
|
|
|
for (auto imageView : swapChainImageViews) {
|
|
|
|
|
vkDestroyImageView(device, imageView, nullptr);
|
|
|
|
|
}
|
|
|
|
|