i think its done
This commit is contained in:
parent
198f7e1269
commit
4bb7c3fcbe
|
@ -2,12 +2,13 @@
|
|||
#include "Application.hpp"
|
||||
|
||||
#include <stdexcept>
|
||||
#include <iostream>
|
||||
|
||||
#include "backends/imgui_impl_glfw.h"
|
||||
#include "backends/imgui_impl_opengl3.h"
|
||||
|
||||
Application::Application() :
|
||||
window(new Window(1280, 720, "Mandelbrot")), canvas(nullptr), backgroundColor{ 0.1f, 0.01f, 0.19f }
|
||||
window(new Window(1280, 720, "Julia Sets")), canvas(nullptr)
|
||||
{
|
||||
window->MakeContextCurrent();
|
||||
|
||||
|
@ -28,6 +29,27 @@ Application::Application() :
|
|||
ImGui_ImplOpenGL3_Init("#version 460 core");
|
||||
|
||||
ImGui::StyleColorsDark();
|
||||
|
||||
data.lastMousePos = { 0.0, 0.0 };
|
||||
data.mouseDelta = { 0.0, 0.0 };
|
||||
data.wheel = { 0.0, 0.0 };
|
||||
|
||||
GLFWwindow* nativeHandle = window->GetHandle();
|
||||
glfwSetWindowUserPointer(nativeHandle, (void*)&data);
|
||||
|
||||
if (glfwRawMouseMotionSupported())
|
||||
glfwSetInputMode(nativeHandle, GLFW_RAW_MOUSE_MOTION, GLFW_TRUE);
|
||||
|
||||
|
||||
glfwSetScrollCallback(nativeHandle,
|
||||
[] (GLFWwindow* window, double x, double y)
|
||||
{
|
||||
WindowData* data = (WindowData*)glfwGetWindowUserPointer(window);
|
||||
|
||||
data->wheel.x = x;
|
||||
data->wheel.y = y;
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
Application::~Application()
|
||||
|
@ -41,8 +63,10 @@ void Application::Launch()
|
|||
{
|
||||
while (!window->ShouldClose())
|
||||
{
|
||||
canvas->CalculateJuliaSet();
|
||||
|
||||
glfwPollEvents();
|
||||
glClearColor(backgroundColor[0], backgroundColor[1], backgroundColor[2], 1.0f);
|
||||
glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
|
||||
glClear(GL_COLOR_BUFFER_BIT);
|
||||
|
||||
ImGui_ImplOpenGL3_NewFrame();
|
||||
|
@ -51,8 +75,59 @@ void Application::Launch()
|
|||
|
||||
canvas->Render();
|
||||
|
||||
ImGui::Begin("Settings");
|
||||
ImGui::SliderFloat3("Background Color", backgroundColor, 0.0f, 1.0f);
|
||||
JuliaProperties& props = canvas->GetProperties();
|
||||
int width, height;
|
||||
window->GetWindowSize(width, height);
|
||||
props.aspectRatio = (float)height / (float)width;
|
||||
|
||||
double mouseX, mouseY;
|
||||
glfwGetCursorPos(window->GetHandle(), &mouseX, &mouseY);
|
||||
|
||||
data.mouseDelta.x = mouseX - data.lastMousePos.x;
|
||||
data.mouseDelta.y = mouseY - data.lastMousePos.y;
|
||||
|
||||
data.lastMousePos.x = mouseX;
|
||||
data.lastMousePos.y = mouseY;
|
||||
|
||||
float xSize = props.xBounds[1] - props.xBounds[0];
|
||||
|
||||
if (glfwGetMouseButton(window->GetHandle(), GLFW_MOUSE_BUTTON_LEFT))
|
||||
{
|
||||
float stepSize = xSize / (float)width;
|
||||
|
||||
props.xBounds[0] -= data.mouseDelta.x * stepSize;
|
||||
props.xBounds[1] -= data.mouseDelta.x * stepSize;
|
||||
|
||||
props.yCenter += data.mouseDelta.y * stepSize;
|
||||
}
|
||||
|
||||
if (data.wheel.y != 0.0)
|
||||
{
|
||||
props.xBounds[0] += data.wheel.y * (xSize / 10.0f);
|
||||
props.xBounds[1] -= data.wheel.y * (xSize / 10.0f);
|
||||
}
|
||||
|
||||
data.wheel = { 0.0, 0.0 };
|
||||
|
||||
const WorkProperties& workProps = canvas->GetWorkProperties();
|
||||
|
||||
ImGui::Begin("Julia Set Properties");
|
||||
|
||||
ImGui::SliderInt("max iterations", (int*)&props.maxIterations, 10, 1000);
|
||||
ImGui::SliderFloat("max color", &props.iterationColorCutoff, 10, 1000);
|
||||
ImGui::SliderInt("texture width", (int*)&props.textureWidth, 480, 2560);
|
||||
|
||||
ImGui::SliderFloat2("c param", props.c, -1.5f, 1.5f);
|
||||
|
||||
if (ImGui::Button(props.doublePrecision ? "Double Precision" : "Single Precision"))
|
||||
props.doublePrecision = !props.doublePrecision;
|
||||
|
||||
ImGui::Separator();
|
||||
|
||||
ImGui::Text("Max work groups - x: %i, y: %i, z: %i", workProps.groupCount[0], workProps.groupCount[1], workProps.groupCount[2]);
|
||||
ImGui::Text("Max group sizes - x: %i, y: %i, z: %i", workProps.groupSize[0], workProps.groupSize[1], workProps.groupSize[2]);
|
||||
ImGui::Text("Max invocations - %i", workProps.maxInvocations);
|
||||
|
||||
ImGui::End();
|
||||
|
||||
ImGui::Render();
|
||||
|
|
|
@ -3,6 +3,14 @@
|
|||
#include "Window.hpp"
|
||||
#include "Canvas.hpp"
|
||||
|
||||
struct WindowData
|
||||
{
|
||||
struct
|
||||
{
|
||||
double x, y;
|
||||
} mouseDelta, lastMousePos, wheel;
|
||||
};
|
||||
|
||||
class Application
|
||||
{
|
||||
public:
|
||||
|
@ -15,5 +23,5 @@ private:
|
|||
Window* window;
|
||||
Canvas* canvas;
|
||||
|
||||
float backgroundColor[3];
|
||||
WindowData data;
|
||||
};
|
|
@ -4,7 +4,7 @@
|
|||
cmake_minimum_required (VERSION 3.8)
|
||||
|
||||
# Add source to this project's executable.
|
||||
add_executable (Mandelbrot "main.cpp" "Window.cpp" "Application.cpp" "Canvas.cpp")
|
||||
add_executable (Mandelbrot "main.cpp" "Window.cpp" "Application.cpp" "Canvas.cpp" "Shader.cpp")
|
||||
|
||||
target_sources(Mandelbrot PRIVATE
|
||||
${IMGUI_SOURCE_FILES}
|
||||
|
|
228
src/Canvas.cpp
228
src/Canvas.cpp
|
@ -2,20 +2,41 @@
|
|||
|
||||
#include <string>
|
||||
#include <stdexcept>
|
||||
#include <complex>
|
||||
#include <regex>
|
||||
|
||||
#include <glad/glad.h>
|
||||
|
||||
Canvas::Canvas() :
|
||||
vao(0), vbo(0), shader(0)
|
||||
double Map(double fromMin, double fromMax, double toMin, double toMax, double val)
|
||||
{
|
||||
return (val - fromMin) * (toMax - toMin) / (fromMax - fromMin) + toMin;
|
||||
}
|
||||
|
||||
Canvas::Canvas() :
|
||||
vao(0), vbo(0), texture(0)
|
||||
{
|
||||
properties.xBounds[0] = -2.5f;
|
||||
properties.xBounds[1] = 2.5f;
|
||||
properties.yCenter = 0.0f;
|
||||
properties.aspectRatio = 9.0f / 16.0f;
|
||||
properties.textureWidth = 1920;
|
||||
properties.maxIterations = 100;
|
||||
properties.iterationColorCutoff = 100.0f;
|
||||
properties.c[0] = -0.835;
|
||||
properties.c[1] = -0.2321;
|
||||
properties.doublePrecision = false;
|
||||
|
||||
CreateVertexArrayObject();
|
||||
CreateShaderProgram();
|
||||
CreateTexture();
|
||||
|
||||
CreateCompueShader();
|
||||
}
|
||||
|
||||
Canvas::~Canvas()
|
||||
{
|
||||
if (shader)
|
||||
glDeleteProgram(shader);
|
||||
if (texture)
|
||||
glDeleteTextures(1, &texture);
|
||||
|
||||
if (vbo)
|
||||
glDeleteBuffers(1, &vbo);
|
||||
|
@ -26,19 +47,60 @@ Canvas::~Canvas()
|
|||
|
||||
void Canvas::Render()
|
||||
{
|
||||
glUseProgram(shader);
|
||||
glBindVertexArray(vao);
|
||||
glMemoryBarrier(GL_SHADER_IMAGE_ACCESS_BARRIER_BIT);
|
||||
|
||||
shader.Use();
|
||||
glBindTexture(GL_TEXTURE_2D, texture);
|
||||
|
||||
glBindVertexArray(vao);
|
||||
glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
|
||||
}
|
||||
|
||||
void Canvas::CalculateJuliaSet()
|
||||
{
|
||||
glMemoryBarrier(GL_SHADER_IMAGE_ACCESS_BARRIER_BIT);
|
||||
|
||||
int width = properties.textureWidth;
|
||||
int height = properties.textureWidth * properties.aspectRatio;
|
||||
|
||||
float yLength = (properties.xBounds[1] - properties.xBounds[0]) * properties.aspectRatio;
|
||||
float yMin = properties.yCenter - 0.5f * yLength;
|
||||
float yMax = properties.yCenter + 0.5f * yLength;
|
||||
|
||||
glBindTexture(GL_TEXTURE_2D, texture);
|
||||
|
||||
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
||||
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
||||
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
||||
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
||||
|
||||
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA32F, width, height, 0, GL_RGBA, GL_FLOAT, nullptr);
|
||||
glGenerateMipmap(GL_TEXTURE_2D);
|
||||
glBindImageTexture(0, texture, 0, GL_FALSE, 0, GL_WRITE_ONLY, GL_RGBA32F);
|
||||
|
||||
if (properties.doublePrecision)
|
||||
doubleComputeShader.Use();
|
||||
else
|
||||
computeShader.Use();
|
||||
|
||||
glUniform2f(1, properties.xBounds[0], properties.xBounds[1]);
|
||||
glUniform2f(2, yMin, yMax);
|
||||
glUniform2f(3, properties.c[0], properties.c[1]);
|
||||
glUniform1i(4, properties.maxIterations);
|
||||
glUniform1f(5, properties.iterationColorCutoff);
|
||||
|
||||
glDispatchCompute(width, height, 1);
|
||||
|
||||
// delete[] image;
|
||||
}
|
||||
|
||||
void Canvas::CreateVertexArrayObject()
|
||||
{
|
||||
float vertices[4 * 2] = {
|
||||
-0.9f, -0.9f,
|
||||
-0.9f, 0.9f,
|
||||
0.9f, 0.9f,
|
||||
0.9f, -0.9f
|
||||
float vertices[4 * (2 + 2)] = {
|
||||
-1.0f, -1.0f, 0.0f, 0.0f,
|
||||
-1.0f, 1.0f, 0.0f, 1.0f,
|
||||
1.0f, 1.0f, 1.0f, 1.0f,
|
||||
1.0f, -1.0f, 1.0f, 0.0f
|
||||
};
|
||||
|
||||
glGenVertexArrays(1, &vao);
|
||||
|
@ -48,77 +110,127 @@ void Canvas::CreateVertexArrayObject()
|
|||
glBindBuffer(GL_ARRAY_BUFFER, vbo);
|
||||
glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), (const void*)&vertices, GL_STATIC_DRAW);
|
||||
|
||||
glVertexAttribPointer(0, 2, GL_FLOAT, GL_FALSE, 2 * sizeof(float), (const void*)0);
|
||||
glVertexAttribPointer(0, 2, GL_FLOAT, GL_FALSE, 4 * sizeof(float), (const void*)0);
|
||||
glEnableVertexAttribArray(0);
|
||||
|
||||
glVertexAttribPointer(1, 2, GL_FLOAT, GL_FALSE, 4 * sizeof(float), (const void*)(2 * sizeof(float)));
|
||||
glEnableVertexAttribArray(1);
|
||||
}
|
||||
|
||||
void Canvas::CreateShaderProgram()
|
||||
{
|
||||
GLint result;
|
||||
char infoLog[512];
|
||||
|
||||
shader = glCreateProgram();
|
||||
|
||||
GLuint vertexShader = glCreateShader(GL_VERTEX_SHADER);
|
||||
std::string vertexShaderSource = R"(
|
||||
#version 460 core
|
||||
|
||||
layout (location = 0) in vec2 pos;
|
||||
layout (location = 1) in vec2 uv;
|
||||
|
||||
out vec2 uvCoord;
|
||||
|
||||
void main()
|
||||
{
|
||||
uvCoord = uv;
|
||||
gl_Position = vec4(pos, 0.0f, 1.0f);
|
||||
}
|
||||
)";
|
||||
const char* shaderSourceCString = vertexShaderSource.c_str();
|
||||
glShaderSource(vertexShader, 1, &shaderSourceCString, NULL);
|
||||
glCompileShader(vertexShader);
|
||||
shader.AttachVertexShader(vertexShaderSource);
|
||||
|
||||
glGetShaderiv(vertexShader, GL_COMPILE_STATUS, &result);
|
||||
if (!result)
|
||||
{
|
||||
glGetShaderInfoLog(vertexShader, 512, NULL, infoLog);
|
||||
glDeleteShader(vertexShader);
|
||||
throw std::runtime_error("Failed to compile vertex shader\n" + std::string(infoLog));
|
||||
}
|
||||
|
||||
GLuint fragmentShader = glCreateShader(GL_FRAGMENT_SHADER);
|
||||
std::string fragmentShaderSource = R"(
|
||||
#version 460 core
|
||||
|
||||
in vec2 uvCoord;
|
||||
out vec4 FragColor;
|
||||
|
||||
uniform sampler2D canvas;
|
||||
|
||||
void main()
|
||||
{
|
||||
FragColor = vec4(0.1f, 0.3f, 0.3f, 1.0f);
|
||||
FragColor = texture(canvas, uvCoord);
|
||||
}
|
||||
)";
|
||||
shaderSourceCString = fragmentShaderSource.c_str();
|
||||
glShaderSource(fragmentShader, 1, &shaderSourceCString, NULL);
|
||||
glCompileShader(fragmentShader);
|
||||
shader.AttachFragmentShader(fragmentShaderSource);
|
||||
|
||||
glAttachShader(shader, vertexShader);
|
||||
glAttachShader(shader, fragmentShader);
|
||||
glGetShaderiv(fragmentShader, GL_COMPILE_STATUS, &result);
|
||||
if (!result)
|
||||
{
|
||||
glGetShaderInfoLog(fragmentShader, 512, NULL, infoLog);
|
||||
glDeleteShader(fragmentShader);
|
||||
glDeleteShader(vertexShader);
|
||||
throw std::runtime_error("Failed to compile fragment shader\n" + std::string(infoLog));
|
||||
}
|
||||
|
||||
glLinkProgram(shader);
|
||||
|
||||
glGetProgramiv(shader, GL_LINK_STATUS, &result);
|
||||
if (!result)
|
||||
{
|
||||
glGetProgramInfoLog(shader, 512, NULL, infoLog);
|
||||
glDeleteShader(fragmentShader);
|
||||
glDeleteShader(vertexShader);
|
||||
throw std::runtime_error("Failed to link shader program\n" + std::string(infoLog));
|
||||
}
|
||||
|
||||
glDeleteShader(fragmentShader);
|
||||
glDeleteShader(vertexShader);
|
||||
shader.Link();
|
||||
}
|
||||
|
||||
void Canvas::CreateCompueShader()
|
||||
{
|
||||
QueryWorkGroupInfo();
|
||||
|
||||
std::string shaderSource = R"(
|
||||
#version 460 core
|
||||
|
||||
layout(local_size_x = 1, local_size_y = 1) in;
|
||||
layout(rgba32f, binding = 0) uniform image2D img_out;
|
||||
layout(location = 1) uniform vec2 xDomain;
|
||||
layout(location = 2) uniform vec2 yDomain;
|
||||
layout(location = 3) uniform vec2 c;
|
||||
layout(location = 4) uniform int maxIterations;
|
||||
layout(location = 5) uniform float iterationColorCutoff;
|
||||
|
||||
double map(double fromMin, double fromMax, double toMin, double toMax, double val)
|
||||
{
|
||||
return (val - fromMin) * (toMax - toMin) / (fromMax - fromMin) + toMin;
|
||||
}
|
||||
|
||||
dvec2 complexMul(dvec2 a, dvec2 b)
|
||||
{
|
||||
return dvec2(a.x * b.x - a.y * b.y, a.x * b.y + a.y * b.x);
|
||||
}
|
||||
|
||||
void main()
|
||||
{
|
||||
vec4 pixel = vec4(0.0f, 0.05f, 0.2f, 1.0f);
|
||||
|
||||
ivec2 pixel_coords = ivec2(gl_GlobalInvocationID.xy);
|
||||
ivec2 image_size = ivec2(gl_NumWorkGroups.xy);
|
||||
|
||||
double threshold = 0.5f * (sqrt(4 * length(c) + 1) + 1);
|
||||
|
||||
dvec2 z = dvec2(
|
||||
map(0, image_size.x, xDomain.x, xDomain.y, pixel_coords.x),
|
||||
map(0, image_size.y, yDomain.x, yDomain.y, pixel_coords.y)
|
||||
);
|
||||
|
||||
for(int i = 0; i < maxIterations; i++)
|
||||
{
|
||||
if(length(z) > threshold)
|
||||
{
|
||||
pixel.x = float(i) / iterationColorCutoff;
|
||||
break;
|
||||
}
|
||||
|
||||
z = complexMul(z, z) + c;
|
||||
}
|
||||
|
||||
imageStore(img_out, pixel_coords, pixel);
|
||||
}
|
||||
)";
|
||||
doubleComputeShader.AttachComputeShader(shaderSource);
|
||||
doubleComputeShader.Link();
|
||||
|
||||
shaderSource = std::regex_replace(shaderSource, std::regex("double"), "float");
|
||||
shaderSource = std::regex_replace(shaderSource, std::regex("dvec2"), "vec2");
|
||||
computeShader.AttachComputeShader(shaderSource);
|
||||
computeShader.Link();
|
||||
}
|
||||
|
||||
void Canvas::CreateTexture()
|
||||
{
|
||||
glGenTextures(1, &texture);
|
||||
|
||||
// CalculateJuliaSet();
|
||||
}
|
||||
|
||||
void Canvas::QueryWorkGroupInfo()
|
||||
{
|
||||
glGetIntegeri_v(GL_MAX_COMPUTE_WORK_GROUP_COUNT, 0, &workProperties.groupCount[0]);
|
||||
glGetIntegeri_v(GL_MAX_COMPUTE_WORK_GROUP_COUNT, 1, &workProperties.groupCount[1]);
|
||||
glGetIntegeri_v(GL_MAX_COMPUTE_WORK_GROUP_COUNT, 2, &workProperties.groupCount[2]);
|
||||
|
||||
glGetIntegeri_v(GL_MAX_COMPUTE_WORK_GROUP_SIZE, 0, &workProperties.groupSize[0]);
|
||||
glGetIntegeri_v(GL_MAX_COMPUTE_WORK_GROUP_SIZE, 1, &workProperties.groupSize[1]);
|
||||
glGetIntegeri_v(GL_MAX_COMPUTE_WORK_GROUP_SIZE, 2, &workProperties.groupSize[2]);
|
||||
|
||||
glGetIntegerv(GL_MAX_COMPUTE_WORK_GROUP_INVOCATIONS, &workProperties.maxInvocations);
|
||||
}
|
||||
|
|
|
@ -1,6 +1,26 @@
|
|||
#pragma once
|
||||
|
||||
#include <cstdint>
|
||||
#include "Shader.hpp"
|
||||
|
||||
struct JuliaProperties
|
||||
{
|
||||
float xBounds[2];
|
||||
float yCenter;
|
||||
float aspectRatio;
|
||||
uint32_t maxIterations;
|
||||
float iterationColorCutoff;
|
||||
uint32_t textureWidth;
|
||||
float c[2];
|
||||
bool doublePrecision;
|
||||
};
|
||||
|
||||
struct WorkProperties
|
||||
{
|
||||
int groupCount[3];
|
||||
int groupSize[3];
|
||||
int maxInvocations;
|
||||
};
|
||||
|
||||
class Canvas
|
||||
{
|
||||
|
@ -9,12 +29,23 @@ public:
|
|||
~Canvas();
|
||||
|
||||
void Render();
|
||||
void CalculateJuliaSet();
|
||||
inline JuliaProperties& GetProperties() { return properties; }
|
||||
inline const WorkProperties& GetWorkProperties() { return workProperties; }
|
||||
|
||||
private:
|
||||
void CreateVertexArrayObject();
|
||||
void CreateShaderProgram();
|
||||
void CreateCompueShader();
|
||||
void CreateTexture();
|
||||
|
||||
void QueryWorkGroupInfo();
|
||||
|
||||
private:
|
||||
uint32_t vao, vbo;
|
||||
uint32_t shader;
|
||||
Shader shader, computeShader, doubleComputeShader;
|
||||
uint32_t texture;
|
||||
|
||||
JuliaProperties properties;
|
||||
WorkProperties workProperties;
|
||||
};
|
117
src/Shader.cpp
Normal file
117
src/Shader.cpp
Normal file
|
@ -0,0 +1,117 @@
|
|||
#include "Shader.hpp"
|
||||
|
||||
#include <stdexcept>
|
||||
#include <glad/glad.h>
|
||||
|
||||
Shader::Shader() :
|
||||
program(0), vertexShader(0), fragmentShader(0), computeShader(0)
|
||||
{
|
||||
program = glCreateProgram();
|
||||
}
|
||||
|
||||
Shader::~Shader()
|
||||
{
|
||||
if (vertexShader)
|
||||
glDeleteShader(vertexShader);
|
||||
|
||||
if (fragmentShader)
|
||||
glDeleteShader(fragmentShader);
|
||||
|
||||
if (computeShader)
|
||||
glDeleteShader(computeShader);
|
||||
|
||||
glDeleteProgram(program);
|
||||
}
|
||||
|
||||
void Shader::AttachVertexShader(const std::string& vertexSource)
|
||||
{
|
||||
GLint result;
|
||||
char infoLog[512];
|
||||
|
||||
vertexShader = glCreateShader(GL_VERTEX_SHADER);
|
||||
const char* shaderSourceCString = vertexSource.c_str();
|
||||
glShaderSource(vertexShader, 1, &shaderSourceCString, NULL);
|
||||
glCompileShader(vertexShader);
|
||||
|
||||
glGetShaderiv(vertexShader, GL_COMPILE_STATUS, &result);
|
||||
if (!result)
|
||||
{
|
||||
glGetShaderInfoLog(vertexShader, 512, NULL, infoLog);
|
||||
glDeleteShader(vertexShader);
|
||||
throw std::runtime_error("Failed to compile vertex shader\n" + std::string(infoLog));
|
||||
}
|
||||
|
||||
glAttachShader(program, vertexShader);
|
||||
}
|
||||
|
||||
void Shader::AttachFragmentShader(const std::string& fragmentSource)
|
||||
{
|
||||
GLint result;
|
||||
char infoLog[512];
|
||||
|
||||
fragmentShader = glCreateShader(GL_FRAGMENT_SHADER);
|
||||
const char* shaderSourceCString = fragmentSource.c_str();
|
||||
glShaderSource(fragmentShader, 1, &shaderSourceCString, NULL);
|
||||
glCompileShader(fragmentShader);
|
||||
|
||||
glGetShaderiv(fragmentShader, GL_COMPILE_STATUS, &result);
|
||||
if (!result)
|
||||
{
|
||||
glGetShaderInfoLog(fragmentShader, 512, NULL, infoLog);
|
||||
glDeleteShader(fragmentShader);
|
||||
throw std::runtime_error("Failed to compile fragment shader\n" + std::string(infoLog));
|
||||
}
|
||||
|
||||
glAttachShader(program, fragmentShader);
|
||||
}
|
||||
|
||||
void Shader::AttachComputeShader(const std::string& computeSource)
|
||||
{
|
||||
GLint result;
|
||||
char infoLog[512];
|
||||
|
||||
computeShader = glCreateShader(GL_COMPUTE_SHADER);
|
||||
const char* shaderSourceCString = computeSource.c_str();
|
||||
glShaderSource(computeShader, 1, &shaderSourceCString, NULL);
|
||||
glCompileShader(computeShader);
|
||||
|
||||
glGetShaderiv(computeShader, GL_COMPILE_STATUS, &result);
|
||||
if (!result)
|
||||
{
|
||||
glGetShaderInfoLog(computeShader, 512, NULL, infoLog);
|
||||
glDeleteShader(computeShader);
|
||||
throw std::runtime_error("Failed to compile compute shader\n" + std::string(infoLog));
|
||||
}
|
||||
|
||||
glAttachShader(program, computeShader);
|
||||
}
|
||||
|
||||
void Shader::Link()
|
||||
{
|
||||
GLint result;
|
||||
char infoLog[512];
|
||||
|
||||
glLinkProgram(program);
|
||||
glGetProgramiv(program, GL_LINK_STATUS, &result);
|
||||
if (!result)
|
||||
{
|
||||
glGetProgramInfoLog(program, 512, NULL, infoLog);
|
||||
glDeleteShader(fragmentShader);
|
||||
glDeleteShader(vertexShader);
|
||||
throw std::runtime_error("Failed to link shader program\n" + std::string(infoLog));
|
||||
}
|
||||
|
||||
if (vertexShader)
|
||||
glDeleteShader(vertexShader);
|
||||
|
||||
if (fragmentShader)
|
||||
glDeleteShader(fragmentShader);
|
||||
|
||||
if (computeShader)
|
||||
glDeleteShader(computeShader);
|
||||
}
|
||||
|
||||
void Shader::Use()
|
||||
{
|
||||
glUseProgram(program);
|
||||
}
|
21
src/Shader.hpp
Normal file
21
src/Shader.hpp
Normal file
|
@ -0,0 +1,21 @@
|
|||
#pragma once
|
||||
|
||||
#include <string>
|
||||
|
||||
class Shader
|
||||
{
|
||||
public:
|
||||
Shader();
|
||||
~Shader();
|
||||
|
||||
void AttachVertexShader(const std::string& vertexSource);
|
||||
void AttachFragmentShader(const std::string& fragmentSource);
|
||||
void AttachComputeShader(const std::string& computeSource);
|
||||
void Link();
|
||||
|
||||
void Use();
|
||||
|
||||
private:
|
||||
int program;
|
||||
int vertexShader, fragmentShader, computeShader;
|
||||
};
|
|
@ -11,11 +11,15 @@ public:
|
|||
~Window();
|
||||
|
||||
inline bool ShouldClose() { return glfwWindowShouldClose(handle); }
|
||||
inline void SetShouldClose(int value) { return glfwSetWindowShouldClose(handle, value); }
|
||||
inline void MakeContextCurrent() { glfwMakeContextCurrent(handle); }
|
||||
void Display();
|
||||
inline void GetWindowSize(int& width, int& height) { glfwGetWindowSize(handle, &width, &height); }
|
||||
|
||||
void Display();
|
||||
void InitImGui();
|
||||
|
||||
inline GLFWwindow* GetHandle() { return handle; }
|
||||
|
||||
private:
|
||||
GLFWwindow* handle;
|
||||
};
|
Loading…
Reference in a new issue