From 51689af51feffd8a5d37e9dc4497a6cf408126a3 Mon Sep 17 00:00:00 2001 From: wheremyfoodat Date: Thu, 15 Sep 2022 14:59:44 +0300 Subject: [PATCH] Update opengl.hpp, start with ELFs --- CMakeLists.txt | 3 +- include/emulator.hpp | 6 +- include/memory.hpp | 4 + include/opengl.hpp | 810 ++++++++++++++++++---------------- src/core/CPU/cpu_dynarmic.cpp | 2 +- src/core/elf.cpp | 24 + src/emulator.cpp | 8 + src/main.cpp | 4 + 8 files changed, 469 insertions(+), 392 deletions(-) create mode 100644 src/core/elf.cpp diff --git a/CMakeLists.txt b/CMakeLists.txt index c6d0f6af..eee48248 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -30,6 +30,7 @@ set(Boost_INCLUDE_DIR "${CMAKE_SOURCE_DIR}/third_party/boost") set(Boost_NO_SYSTEM_PATHS ON) add_library(boost INTERFACE) target_include_directories(boost SYSTEM INTERFACE ${Boost_INCLUDE_DIR}) +add_compile_definitions(NOMINMAX) if(CMAKE_SYSTEM_PROCESSOR STREQUAL "AMD64" OR CMAKE_SYSTEM_PROCESSOR STREQUAL "x86-64") set(DYNARMIC_TESTS OFF) @@ -42,7 +43,7 @@ else() message(FATAL_ERROR "THIS IS NOT x64 WAIT FOR THE KVM IMPLEMENTATION") endif() -set(SOURCE_FILES src/main.cpp src/emulator.cpp src/core/CPU/cpu_dynarmic.cpp src/core/memory.cpp) +set(SOURCE_FILES src/main.cpp src/emulator.cpp src/core/CPU/cpu_dynarmic.cpp src/core/memory.cpp src/core/elf.cpp) set(HEADER_FILES include/emulator.hpp include/helpers.hpp include/opengl.hpp include/termcolor.hpp include/cpu.hpp include/cpu_dynarmic.hpp include/memory.hpp ) diff --git a/include/emulator.hpp b/include/emulator.hpp index 088dda2e..1c27e495 100644 --- a/include/emulator.hpp +++ b/include/emulator.hpp @@ -1,5 +1,7 @@ #pragma once -#define NOMINMAX // Windows why + +#include + #include "cpu.hpp" #include "helpers.hpp" #include "opengl.hpp" @@ -25,4 +27,6 @@ public: void reset(); void run(); void runFrame(); + + bool loadELF(std::filesystem::path& path); }; \ No newline at end of file diff --git a/include/memory.hpp b/include/memory.hpp index a00ef4ff..8dbdf498 100644 --- a/include/memory.hpp +++ b/include/memory.hpp @@ -1,4 +1,6 @@ #pragma once +#include +#include #include #include "helpers.hpp" @@ -15,4 +17,6 @@ public: Memory(); void* getReadPointer(u32 address); void* getWritePointer(u32 address); + + std::optional loadELF(std::filesystem::path& path); }; \ No newline at end of file diff --git a/include/opengl.hpp b/include/opengl.hpp index 88f889c8..50911f01 100644 --- a/include/opengl.hpp +++ b/include/opengl.hpp @@ -32,458 +32,490 @@ namespace OpenGL { -// Workaround for using static_assert inside constexpr if -// https://stackoverflow.com/questions/53945490/how-to-assert-that-a-constexpr-if-else-clause-never-happen -template -constexpr std::false_type AlwaysFalse{}; + // Workaround for using static_assert inside constexpr if + // https://stackoverflow.com/questions/53945490/how-to-assert-that-a-constexpr-if-else-clause-never-happen + template + constexpr std::false_type AlwaysFalse{}; -struct VertexArray { - GLuint m_handle = 0; + struct VertexArray { + GLuint m_handle = 0; - void create() { - if (m_handle == 0) { - glGenVertexArrays(1, &m_handle); + void create() { + if (m_handle == 0) { + glGenVertexArrays(1, &m_handle); + } } - } - VertexArray(bool shouldCreate = false) { - if (shouldCreate) { + VertexArray(bool shouldCreate = false) { + if (shouldCreate) { + create(); + } + } + + ~VertexArray() { glDeleteVertexArrays(1, &m_handle); } + GLuint handle() { return m_handle; } + bool exists() { return m_handle != 0; } + void bind() { glBindVertexArray(m_handle); } + + template + void setAttributeFloat(GLuint index, GLint size, GLsizei stride, const void* offset, bool normalized = false) { + if constexpr (std::is_same()) { + glVertexAttribPointer(index, size, GL_FLOAT, normalized, stride, offset); + } + else if constexpr (std::is_same()) { + glVertexAttribPointer(index, size, GL_BYTE, normalized, stride, offset); + } + else if constexpr (std::is_same()) { + glVertexAttribPointer(index, size, GL_UNSIGNED_BYTE, normalized, stride, offset); + } + else if constexpr (std::is_same()) { + glVertexAttribPointer(index, size, GL_SHORT, normalized, stride, offset); + } + else if constexpr (std::is_same()) { + glVertexAttribPointer(index, size, GL_UNSIGNED_SHORT, normalized, stride, offset); + } + else if constexpr (std::is_same()) { + glVertexAttribPointer(index, size, GL_INT, normalized, stride, offset); + } + else if constexpr (std::is_same()) { + glVertexAttribPointer(index, size, GL_UNSIGNED_INT, normalized, stride, offset); + } + else { + static_assert(AlwaysFalse, "Unimplemented type for OpenGL::setAttributeFloat"); + } + } + + template + void setAttributeInt(GLuint index, GLint size, GLsizei stride, const void* offset) { + if constexpr (std::is_same()) { + glVertexAttribIPointer(index, size, GL_BYTE, stride, offset); + } + else if constexpr (std::is_same()) { + glVertexAttribIPointer(index, size, GL_UNSIGNED_BYTE, stride, offset); + } + else if constexpr (std::is_same()) { + glVertexAttribIPointer(index, size, GL_SHORT, stride, offset); + } + else if constexpr (std::is_same()) { + glVertexAttribIPointer(index, size, GL_UNSIGNED_SHORT, stride, offset); + } + else if constexpr (std::is_same()) { + glVertexAttribIPointer(index, size, GL_INT, stride, offset); + } + else if constexpr (std::is_same()) { + glVertexAttribIPointer(index, size, GL_UNSIGNED_INT, stride, offset); + } + else { + static_assert(AlwaysFalse, "Unimplemented type for OpenGL::setAttributeInt"); + } + } + + template + void setAttributeFloat(GLuint index, GLint size, GLsizei stride, size_t offset, bool normalized = false) { + setAttributeFloat(index, size, stride, reinterpret_cast(offset), normalized); + } + + template + void setAttributeInt(GLuint index, GLint size, GLsizei stride, size_t offset) { + setAttributeInt(index, size, stride, reinterpret_cast(offset)); + } + + void enableAttribute(GLuint index) { glEnableVertexAttribArray(index); } + void disableAttribute(GLuint index) { glDisableVertexAttribArray(index); } + }; + + enum FramebufferTypes { + DrawFramebuffer = GL_DRAW_FRAMEBUFFER, + ReadFramebuffer = GL_READ_FRAMEBUFFER, + DrawAndReadFramebuffer = GL_FRAMEBUFFER + }; + + struct Texture { + GLuint m_handle = 0; + int m_width, m_height; + GLenum m_binding; + int m_samples = 1; // For MSAA + + void create(int width, int height, GLint internalFormat, GLenum binding = GL_TEXTURE_2D, int samples = 1) { + m_width = width; + m_height = height; + m_binding = binding; + + glGenTextures(1, &m_handle); + bind(); + + if (binding == GL_TEXTURE_2D_MULTISAMPLE) { + if (!glTexStorage2DMultisample) { // Assert that glTexStorage2DMultisample has been loaded + throw std::runtime_error("MSAA is not supported on this platform"); + } + + int maxSampleCount; + glGetIntegerv(GL_MAX_INTEGER_SAMPLES, &maxSampleCount); + if (samples > maxSampleCount) { + samples = maxSampleCount; + } + + glTexStorage2DMultisample(GL_TEXTURE_2D_MULTISAMPLE, samples, internalFormat, width, height, GL_TRUE); + } + else { + glTexStorage2D(binding, 1, internalFormat, width, height); + } + } + + void createMSAA(int width, int height, GLint internalFormat, int samples) { + create(width, height, internalFormat, GL_TEXTURE_2D_MULTISAMPLE, samples); + } + + ~Texture() { glDeleteTextures(1, &m_handle); } + GLuint handle() { return m_handle; } + bool exists() { return m_handle != 0; } + void bind() { glBindTexture(m_binding, m_handle); } + int width() { return m_width; } + int height() { return m_height; } + }; + + struct Framebuffer { + GLuint m_handle = 0; + GLenum m_textureType; // GL_TEXTURE_2D or GL_TEXTURE_2D_MULTISAMPLE + + void create() { + if (m_handle == 0) { + glGenFramebuffers(1, &m_handle); + } + } + + Framebuffer(bool shouldCreate = false) { + if (shouldCreate) { + create(); + } + } + + ~Framebuffer() { glDeleteFramebuffers(1, &m_handle); } + GLuint handle() { return m_handle; } + bool exists() { return m_handle != 0; } + void bind(GLenum target) { glBindFramebuffer(target, m_handle); } + void bind(FramebufferTypes target) { bind(static_cast(target)); } + + void createWithTexture(Texture& tex, GLenum mode = GL_FRAMEBUFFER, GLenum textureType = GL_TEXTURE_2D) { + m_textureType = textureType; create(); + bind(mode); + glFramebufferTexture2D(mode, GL_COLOR_ATTACHMENT0, textureType, tex.handle(), 0); } - } - ~VertexArray() { glDeleteVertexArrays(1, &m_handle); } - GLuint handle() { return m_handle; } - bool exists() { return m_handle != 0; } - void bind() { glBindVertexArray(m_handle); } - - template - void setAttributeFloat(GLuint index, GLint size, GLsizei stride, const void* offset, bool normalized = false) { - if constexpr (std::is_same()) { - glVertexAttribPointer(index, size, GL_FLOAT, normalized, stride, offset); - } else if constexpr (std::is_same()) { - glVertexAttribPointer(index, size, GL_BYTE, normalized, stride, offset); - } else if constexpr (std::is_same()) { - glVertexAttribPointer(index, size, GL_UNSIGNED_BYTE, normalized, stride, offset); - } else if constexpr (std::is_same()) { - glVertexAttribPointer(index, size, GL_SHORT, normalized, stride, offset); - } else if constexpr (std::is_same()) { - glVertexAttribPointer(index, size, GL_UNSIGNED_SHORT, normalized, stride, offset); - } else if constexpr (std::is_same()) { - glVertexAttribPointer(index, size, GL_INT, normalized, stride, offset); - } else if constexpr (std::is_same()) { - glVertexAttribPointer(index, size, GL_UNSIGNED_INT, normalized, stride, offset); - } else { - static_assert(AlwaysFalse, "Unimplemented type for OpenGL::setAttributeFloat"); + void createWithReadTexture(Texture& tex, GLenum textureType = GL_TEXTURE_2D) { + createWithTexture(tex, GL_READ_FRAMEBUFFER, textureType); } - } - - template - void setAttributeInt(GLuint index, GLint size, GLsizei stride, const void* offset) { - if constexpr (std::is_same()) { - glVertexAttribIPointer(index, size, GL_BYTE, stride, offset); - } else if constexpr (std::is_same()) { - glVertexAttribIPointer(index, size, GL_UNSIGNED_BYTE, stride, offset); - } else if constexpr (std::is_same()) { - glVertexAttribIPointer(index, size, GL_SHORT, stride, offset); - } else if constexpr (std::is_same()) { - glVertexAttribIPointer(index, size, GL_UNSIGNED_SHORT, stride, offset); - } else if constexpr (std::is_same()) { - glVertexAttribIPointer(index, size, GL_INT, stride, offset); - } else if constexpr (std::is_same()) { - glVertexAttribIPointer(index, size, GL_UNSIGNED_INT, stride, offset); - } else { - static_assert(AlwaysFalse, "Unimplemented type for OpenGL::setAttributeInt"); + void createWithDrawTexture(Texture& tex, GLenum textureType = GL_TEXTURE_2D) { + createWithTexture(tex, GL_DRAW_FRAMEBUFFER, textureType); } - } - template - void setAttributeFloat(GLuint index, GLint size, GLsizei stride, size_t offset, bool normalized = false) { - setAttributeFloat(index, size, stride, reinterpret_cast(offset), normalized); - } + void createWithTextureMSAA(Texture& tex, GLenum mode = GL_FRAMEBUFFER) { + m_textureType = GL_TEXTURE_2D_MULTISAMPLE; + create(); + bind(mode); + glFramebufferTexture2D(mode, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D_MULTISAMPLE, tex.handle(), 0); + } + }; - template - void setAttributeInt(GLuint index, GLint size, GLsizei stride, size_t offset) { - setAttributeInt(index, size, stride, reinterpret_cast(offset)); - } + enum ShaderType { + Fragment = GL_FRAGMENT_SHADER, + Vertex = GL_VERTEX_SHADER, + Geometry = GL_GEOMETRY_SHADER, + Compute = GL_COMPUTE_SHADER, + TessControl = GL_TESS_CONTROL_SHADER, + TessEvaluation = GL_TESS_EVALUATION_SHADER + }; - void enableAttribute(GLuint index) { glEnableVertexAttribArray(index); } - void disableAttribute(GLuint index) { glDisableVertexAttribArray(index); } -}; + struct Shader { + GLuint m_handle = 0; -enum FramebufferTypes { - DrawFramebuffer = GL_DRAW_FRAMEBUFFER, - ReadFramebuffer = GL_READ_FRAMEBUFFER, - DrawAndReadFramebuffer = GL_FRAMEBUFFER -}; + Shader() {} + Shader(const std::string_view source, ShaderType type) { create(source, static_cast(type)); } -struct Texture { - GLuint m_handle = 0; - int m_width, m_height; - GLenum m_binding; - int m_samples = 1; // For MSAA + // Returns whether compilation failed or not + bool create(const std::string_view source, GLenum type) { + m_handle = glCreateShader(type); + const GLchar* const sources[1] = { source.data() }; - void create(int width, int height, GLint internalFormat, GLenum binding = GL_TEXTURE_2D, int samples = 1) { - m_width = width; - m_height = height; - m_binding = binding; + glShaderSource(m_handle, 1, sources, nullptr); + glCompileShader(m_handle); - glGenTextures(1, &m_handle); - bind(); + GLint success; + glGetShaderiv(m_handle, GL_COMPILE_STATUS, &success); + if (success == GL_FALSE) { + char buf[4096]; + glGetShaderInfoLog(m_handle, 4096, nullptr, buf); + fprintf(stderr, "Failed to compile shader\nError: %s\n", buf); + glDeleteShader(m_handle); - if (binding == GL_TEXTURE_2D_MULTISAMPLE) { - if (!glTexStorage2DMultisample) { // Assert that glTexStorage2DMultisample has been loaded - throw std::runtime_error("MSAA is not supported on this platform"); + m_handle = 0; } - int maxSampleCount; - glGetIntegerv(GL_MAX_INTEGER_SAMPLES, &maxSampleCount); - if (samples > maxSampleCount) { - samples = maxSampleCount; + return m_handle != 0; + } + + GLuint handle() { return m_handle; } + bool exists() { return m_handle != 0; } + }; + + struct Program { + GLuint m_handle = 0; + + bool create(std::initializer_list> shaders) { + m_handle = glCreateProgram(); + for (const auto& shader : shaders) { + glAttachShader(m_handle, shader.get().handle()); } - glTexStorage2DMultisample(GL_TEXTURE_2D_MULTISAMPLE, samples, internalFormat, width, height, GL_TRUE); - } else { - glTexStorage2D(binding, 1, internalFormat, width, height); + glLinkProgram(m_handle); + GLint success; + glGetProgramiv(m_handle, GL_LINK_STATUS, &success); + + if (!success) { + char buf[4096]; + glGetProgramInfoLog(m_handle, 4096, nullptr, buf); + fprintf(stderr, "Failed to link program\nError: %s\n", buf); + glDeleteProgram(m_handle); + + m_handle = 0; + } + + return m_handle != 0; } + + GLuint handle() { return m_handle; } + bool exists() { return m_handle != 0; } + void use() { glUseProgram(m_handle); } + }; + + static void dispatchCompute(GLuint groupsX = 1, GLuint groupsY = 1, GLuint groupsZ = 1) { + glDispatchCompute(groupsX, groupsY, groupsZ); } - void createMSAA(int width, int height, GLint internalFormat, int samples) { - create(width, height, internalFormat, GL_TEXTURE_2D_MULTISAMPLE, samples); - } + struct VertexBuffer { + GLuint m_handle = 0; - ~Texture() { glDeleteTextures(1, &m_handle); } - GLuint handle() { return m_handle; } - bool exists() { return m_handle != 0; } - void bind() { glBindTexture(m_binding, m_handle); } - int width() { return m_width; } - int height() { return m_height; } -}; - -struct Framebuffer { - GLuint m_handle = 0; - GLenum m_textureType; // GL_TEXTURE_2D or GL_TEXTURE_2D_MULTISAMPLE - - void create() { - if (m_handle == 0) { - glGenFramebuffers(1, &m_handle); + void create() { + if (m_handle == 0) { + glGenBuffers(1, &m_handle); + } } - } - Framebuffer(bool shouldCreate = false) { - if (shouldCreate) { + void createFixedSize(GLsizei size, GLenum usage = GL_DYNAMIC_DRAW) { create(); - } - } - - ~Framebuffer() { glDeleteFramebuffers(1, &m_handle); } - GLuint handle() { return m_handle; } - bool exists() { return m_handle != 0; } - void bind(GLenum target) { glBindFramebuffer(target, m_handle); } - void bind(FramebufferTypes target) { bind(static_cast(target)); } - - void createWithTexture(Texture& tex, GLenum mode = GL_FRAMEBUFFER, GLenum textureType = GL_TEXTURE_2D) { - m_textureType = textureType; - create(); - bind(mode); - glFramebufferTexture2D(mode, GL_COLOR_ATTACHMENT0, textureType, tex.handle(), 0); - } - - void createWithReadTexture(Texture& tex, GLenum textureType = GL_TEXTURE_2D) { - createWithTexture(tex, GL_READ_FRAMEBUFFER, textureType); - } - void createWithDrawTexture(Texture& tex, GLenum textureType = GL_TEXTURE_2D) { - createWithTexture(tex, GL_DRAW_FRAMEBUFFER, textureType); - } - - void createWithTextureMSAA(Texture& tex, GLenum mode = GL_FRAMEBUFFER) { - m_textureType = GL_TEXTURE_2D_MULTISAMPLE; - create(); - bind(mode); - glFramebufferTexture2D(mode, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D_MULTISAMPLE, tex.handle(), 0); - } -}; - -enum ShaderType { - Fragment = GL_FRAGMENT_SHADER, - Vertex = GL_VERTEX_SHADER, - Geometry = GL_GEOMETRY_SHADER, - Compute = GL_COMPUTE_SHADER, - TessControl = GL_TESS_CONTROL_SHADER, - TessEvaluation = GL_TESS_EVALUATION_SHADER -}; - -struct Shader { - GLuint m_handle = 0; - - Shader() {} - Shader(const std::string_view source, ShaderType type) { create(source, static_cast(type)); } - - // Returns whether compilation failed or not - bool create(const std::string_view source, GLenum type) { - m_handle = glCreateShader(type); - const GLchar* const sources[1] = {source.data()}; - - glShaderSource(m_handle, 1, sources, nullptr); - glCompileShader(m_handle); - - GLint success; - glGetShaderiv(m_handle, GL_COMPILE_STATUS, &success); - if (!success) { - char buf[4096]; - glGetShaderInfoLog(m_handle, 4096, nullptr, buf); - fprintf(stderr, "Failed to compile shader\nError: %s\n", buf); - glDeleteShader(m_handle); - - m_handle = 0; + bind(); + glBufferData(GL_ARRAY_BUFFER, size, nullptr, usage); } - return m_handle != 0; - } - - GLuint handle() { return m_handle; } - bool exists() { return m_handle != 0; } -}; - -struct Program { - GLuint m_handle = 0; - - bool create(std::initializer_list> shaders) { - m_handle = glCreateProgram(); - for (const auto& shader : shaders) { - glAttachShader(m_handle, shader.get().handle()); + VertexBuffer(bool shouldCreate = false) { + if (shouldCreate) { + create(); + } } - glLinkProgram(m_handle); - GLint success; - glGetProgramiv(m_handle, GL_LINK_STATUS, &success); + ~VertexBuffer() { glDeleteBuffers(1, &m_handle); } + GLuint handle() { return m_handle; } + bool exists() { return m_handle != 0; } + void bind() { glBindBuffer(GL_ARRAY_BUFFER, m_handle); } - if (!success) { - char buf[4096]; - glGetProgramInfoLog(m_handle, 4096, nullptr, buf); - fprintf(stderr, "Failed to link program\nError: %s\n", buf); - glDeleteProgram(m_handle); - - m_handle = 0; + // Reallocates the buffer on every call. Prefer the sub version if possible. + template + void bufferVerts(VertType* vertices, int vertCount, GLenum usage = GL_DYNAMIC_DRAW) { + glBufferData(GL_ARRAY_BUFFER, sizeof(VertType) * vertCount, vertices, usage); } - return m_handle != 0; - } - - GLuint handle() { return m_handle; } - bool exists() { return m_handle != 0; } - void use() { glUseProgram(m_handle); } -}; - -struct VertexBuffer { - GLuint m_handle = 0; - - void create() { - if (m_handle == 0) { - glGenBuffers(1, &m_handle); + // Only use if you used createFixedSize + template + void bufferVertsSub(VertType* vertices, int vertCount, GLintptr offset = 0) { + glBufferSubData(GL_ARRAY_BUFFER, offset, sizeof(VertType) * vertCount, vertices); } + }; + + static void setClearColor(float val) { glClearColor(val, val, val, val); } + static void setClearColor(float r, float g, float b, float a) { glClearColor(r, g, b, a); } + static void setClearDepth(float depth) { glClearDepthf(depth); } + static void setClearStencil(GLint stencil) { glClearStencil(stencil); } + static void clearColor() { glClear(GL_COLOR_BUFFER_BIT); } + static void clearDepth() { glClear(GL_DEPTH_BUFFER_BIT); } + static void clearStencil() { glClear(GL_STENCIL_BUFFER_BIT); } + static void clearColorAndDepth() { glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); } + static void clearColorAndStencil() { glClear(GL_COLOR_BUFFER_BIT | GL_STENCIL_BUFFER_BIT); } + static void clearDepthAndStencil() { glClear(GL_DEPTH_BUFFER_BIT | GL_STENCIL_BUFFER_BIT); } + static void clearAll() { glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT | GL_STENCIL_BUFFER_BIT); } + + static void setViewport(GLsizei width, GLsizei height) { glViewport(0, 0, width, height); } + static void setViewport(GLsizei x, GLsizei y, GLsizei width, GLsizei height) { glViewport(x, y, width, height); } + static void setScissor(GLsizei width, GLsizei height) { glScissor(0, 0, width, height); } + static void setScissor(GLsizei x, GLsizei y, GLsizei width, GLsizei height) { glScissor(x, y, width, height); } + static void setStencilMask(GLuint mask) { glStencilMask(mask); } + + static void bindScreenFramebuffer() { glBindFramebuffer(GL_FRAMEBUFFER, 0); } + static void enableScissor() { glEnable(GL_SCISSOR_TEST); } + static void disableScissor() { glDisable(GL_SCISSOR_TEST); } + static void enableBlend() { glEnable(GL_BLEND); } + static void disableBlend() { glDisable(GL_BLEND); } + static void enableDepth() { glEnable(GL_DEPTH_TEST); } + static void disableDepth() { glDisable(GL_DEPTH_TEST); } + static void enableStencil() { glEnable(GL_STENCIL_TEST); } + static void disableStencil() { glDisable(GL_STENCIL_TEST); } + + enum Primitives { + Triangle = GL_TRIANGLES, + Triangles = Triangle, + Tri = Triangle, + Tris = Triangle, + TriangleStrip = GL_TRIANGLE_STRIP, + TriangleFan = GL_TRIANGLE_FAN, + Line = GL_LINES, + Lines = Line, + LineStrip = GL_LINE_STRIP, + Point = GL_POINTS, + Points = Point + }; + + static void draw(Primitives prim, GLsizei vertexCount) { glDrawArrays(static_cast(prim), 0, vertexCount); } + static void draw(Primitives prim, GLint first, GLsizei vertexCount) { + glDrawArrays(static_cast(prim), first, vertexCount); } - void createFixedSize(GLsizei size, GLenum usage = GL_DYNAMIC_DRAW) { - create(); - bind(); - glBufferData(GL_ARRAY_BUFFER, size, nullptr, usage); - } + enum FillMode { DrawPoints = GL_POINT, DrawWire = GL_LINE, FillPoly = GL_FILL }; - VertexBuffer(bool shouldCreate = false) { - if (shouldCreate) { - create(); + static void setFillMode(GLenum mode) { glPolygonMode(GL_FRONT_AND_BACK, mode); } + static void setFillMode(FillMode mode) { glPolygonMode(GL_FRONT_AND_BACK, static_cast(mode)); } + static void drawWireframe() { setFillMode(DrawWire); } + + template + T get(GLenum query) { + T ret{}; + if constexpr (std::is_same()) { + glGetIntegerv(query, &ret); } + else if constexpr (std::is_same()) { + glGetFloatv(query, &ret); + } + else if constexpr (std::is_same()) { + glGetDoublev(query, &ret); + } + else if constexpr (std::is_same()) { + glGetBooleanv(query, &ret); + } + else { + static_assert(AlwaysFalse, "Invalid type for OpenGL::get"); + } + + return ret; } - ~VertexBuffer() { glDeleteBuffers(1, &m_handle); } - GLuint handle() { return m_handle; } - bool exists() { return m_handle != 0; } - void bind() { glBindBuffer(GL_ARRAY_BUFFER, m_handle); } + static bool isEnabled(GLenum query) { return glIsEnabled(query) != GL_FALSE; } - // Reallocates the buffer on every call. Prefer the sub version if possible. - template - void bufferVerts(VertType* vertices, int vertCount, GLenum usage = GL_DYNAMIC_DRAW) { - glBufferData(GL_ARRAY_BUFFER, sizeof(VertType) * vertCount, vertices, usage); + static GLint getDrawFramebuffer() { return get(GL_DRAW_FRAMEBUFFER_BINDING); } + static GLint maxSamples() { return get(GL_MAX_INTEGER_SAMPLES); } + static GLint getTex2D() { return get(GL_TEXTURE_BINDING_2D); } + static GLint getProgram() { return get(GL_CURRENT_PROGRAM); } + static bool scissorEnabled() { return isEnabled(GL_SCISSOR_TEST); } + + static bool versionSupported(int major, int minor) { return gl3wIsSupported(major, minor); } + + [[nodiscard]] static GLint uniformLocation(GLuint program, const char* name) { + return glGetUniformLocation(program, name); + } + [[nodiscard]] static GLint uniformLocation(Program& program, const char* name) { + return glGetUniformLocation(program.handle(), name); } - // Only use if you used createFixedSize - template - void bufferVertsSub(VertType* vertices, int vertCount, GLintptr offset = 0) { - glBufferSubData(GL_ARRAY_BUFFER, offset, sizeof(VertType) * vertCount, vertices); - } -}; + enum BlendEquation { + Add = GL_FUNC_ADD, + Sub = GL_FUNC_SUBTRACT, + ReverseSub = GL_FUNC_REVERSE_SUBTRACT, + Min = GL_MIN, + Max = GL_MAX + }; -static void setClearColor(float val) { glClearColor(val, val, val, val); } -static void setClearColor(float r, float g, float b, float a) { glClearColor(r, g, b, a); } -static void setClearDepth(float depth) { glClearDepthf(depth); } -static void clearColor() { glClear(GL_COLOR_BUFFER_BIT); } -static void clearDepth() { glClear(GL_DEPTH_BUFFER_BIT); } -static void clearColorAndDepth() { glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); } + static void setBlendColor(float r, float g, float b, float a = 1.0) { glBlendColor(r, g, b, a); } + static void setBlendEquation(BlendEquation eq) { glBlendEquation(eq); } + static void setBlendEquation(BlendEquation eq1, BlendEquation eq2) { glBlendEquationSeparate(eq1, eq2); } -static void setViewport(GLsizei width, GLsizei height) { glViewport(0, 0, width, height); } -static void setViewport(GLsizei x, GLsizei y, GLsizei width, GLsizei height) { glViewport(x, y, width, height); } -static void setScissor(GLsizei width, GLsizei height) { glScissor(0, 0, width, height); } -static void setScissor(GLsizei x, GLsizei y, GLsizei width, GLsizei height) { glScissor(x, y, width, height); } - -static void bindScreenFramebuffer() { glBindFramebuffer(GL_FRAMEBUFFER, 0); } -static void enableScissor() { glEnable(GL_SCISSOR_TEST); } -static void disableScissor() { glDisable(GL_SCISSOR_TEST); } -static void enableBlend() { glEnable(GL_BLEND); } -static void disableBlend() { glDisable(GL_BLEND); } - -enum Primitives { - Triangle = GL_TRIANGLES, - Triangles = Triangle, - Tri = Triangle, - Tris = Triangle, - TriangleStrip = GL_TRIANGLE_STRIP, - TriangleFan = GL_TRIANGLE_FAN, - Line = GL_LINES, - Lines = Line, - LineStrip = GL_LINE_STRIP, - Point = GL_POINTS, - Points = Point -}; - -static void draw(Primitives prim, GLsizei vertexCount) { glDrawArrays(static_cast(prim), 0, vertexCount); } -static void draw(Primitives prim, GLint first, GLsizei vertexCount) { - glDrawArrays(static_cast(prim), first, vertexCount); -} - -enum FillMode { DrawPoints = GL_POINT, DrawWire = GL_LINE, FillPoly = GL_FILL }; - -static void setFillMode(GLenum mode) { glPolygonMode(GL_FRONT_AND_BACK, mode); } -static void setFillMode(FillMode mode) { glPolygonMode(GL_FRONT_AND_BACK, static_cast(mode)); } -static void drawWireframe() { setFillMode(DrawWire); } - -template -T get(GLenum query) { - T ret{}; - if constexpr (std::is_same()) { - glGetIntegerv(query, &ret); - } else if constexpr (std::is_same()) { - glGetFloatv(query, &ret); - } else if constexpr (std::is_same()) { - glGetDoublev(query, &ret); - } else if constexpr (std::is_same()) { - glGetBooleanv(query, &ret); - } else { - static_assert(AlwaysFalse, "Invalid type for OpenGL::get"); + static void setBlendFactor(GLenum fac1, GLenum fac2) { glBlendFunc(fac1, fac2); } + static void setBlendFactor(GLenum fac1, GLenum fac2, GLenum fac3, GLenum fac4) { + glBlendFuncSeparate(fac1, fac2, fac3, fac4); } - return ret; -} + // Abstraction for GLSL vectors + template + class Vector { + // A GLSL vector can only have 2, 3 or 4 elements + static_assert(size == 2 || size == 3 || size == 4); + T m_storage[size]; -static bool isEnabled(GLenum query) { return glIsEnabled(query) != GL_FALSE; } + public: + T& r() { return m_storage[0]; } + T& g() { return m_storage[1]; } + T& b() { + static_assert(size >= 3, "Out of bounds OpenGL::Vector access"); + return m_storage[2]; + } + T& a() { + static_assert(size >= 4, "Out of bounds OpenGL::Vector access"); + return m_storage[3]; + } -static GLint getDrawFramebuffer() { return get(GL_DRAW_FRAMEBUFFER_BINDING); } -static GLint maxSamples() { return get(GL_MAX_INTEGER_SAMPLES); } -static GLint getTex2D() { return get(GL_TEXTURE_BINDING_2D); } -static GLint getProgram() { return get(GL_CURRENT_PROGRAM); } -static bool scissorEnabled() { return isEnabled(GL_SCISSOR_TEST); } + T& x() { return r(); } + T& y() { return g(); } + T& z() { return b(); } + T& w() { return a(); } + T& operator[](int index) { return m_storage[index]; } -static bool versionSupported(int major, int minor) { return gl3wIsSupported(major, minor); } + T& u() { return r(); } + T& v() { return g(); } -[[nodiscard]] static GLint uniformLocation(GLuint program, const char* name) { - return glGetUniformLocation(program, name); -} -[[nodiscard]] static GLint uniformLocation(Program& program, const char* name) { - return glGetUniformLocation(program.handle(), name); -} + T& s() { return r(); } + T& t() { return g(); } + T& p() { return b(); } + T& q() { return a(); } -enum BlendEquation { - Add = GL_FUNC_ADD, - Sub = GL_FUNC_SUBTRACT, - ReverseSub = GL_FUNC_REVERSE_SUBTRACT, - Min = GL_MIN, - Max = GL_MAX -}; + Vector(std::array list) { std::copy(list.begin(), list.end(), &m_storage[0]); } -static void setBlendColor(float r, float g, float b, float a = 1.0) { glBlendColor(r, g, b, a); } -static void setBlendEquation(BlendEquation eq) { glBlendEquation(eq); } -static void setBlendEquation(BlendEquation eq1, BlendEquation eq2) { glBlendEquationSeparate(eq1, eq2); } + Vector() {} + }; -static void setBlendFactor(GLenum fac1, GLenum fac2) { glBlendFunc(fac1, fac2); } -static void setBlendFactor(GLenum fac1, GLenum fac2, GLenum fac3, GLenum fac4) { - glBlendFuncSeparate(fac1, fac2, fac3, fac4); -} + using vec2 = Vector; + using vec3 = Vector; + using vec4 = Vector; -// Abstraction for GLSL vectors -template -class Vector { - // A GLSL vector can only have 2, 3 or 4 elements - static_assert(size == 2 || size == 3 || size == 4); - T m_storage[size]; + using dvec2 = Vector; + using dvec3 = Vector; + using dvec4 = Vector; - public: - T& r() { return m_storage[0]; } - T& g() { return m_storage[1]; } - T& b() { - static_assert(size >= 3, "Out of bounds OpenGL::Vector access"); - return m_storage[2]; - } - T& a() { - static_assert(size >= 4, "Out of bounds OpenGL::Vector access"); - return m_storage[3]; - } + using ivec2 = Vector; + using ivec3 = Vector; + using ivec4 = Vector; - T& x() { return r(); } - T& y() { return g(); } - T& z() { return b(); } - T& w() { return a(); } - T& operator[](int index) { return m_storage[index]; } + using uvec2 = Vector; + using uvec3 = Vector; + using uvec4 = Vector; - T& u() { return r(); } - T& v() { return g(); } + // A 2D rectangle, meant to be used for stuff like scissor rects or viewport rects + // We're never supporting 3D rectangles, because rectangles were never meant to be 3D in the first place + // x, y: Coords of the top left vertex + // width, height: Dimensions of the rectangle. Initialized to 0 if not specified. + template + struct Rectangle { + T x, y, width, height; - T& s() { return r(); } - T& t() { return g(); } - T& p() { return b(); } - T& q() { return a(); } + std::pair topLeft() { return std::make_pair(x, y); } + std::pair topRight() { return std::make_pair(x + width, y); } + std::pair bottomLeft() { return std::make_pair(x, y + height); } + std::pair bottomRight() { return std::make_pair(x + width, y + height); } - Vector(std::array list) { std::copy(list.begin(), list.end(), &m_storage[0]); } + Rectangle() : x(0), y(0), width(0), height(0) {} + Rectangle(T x, T y, T width, T height) : x(x), y(y), width(width), height(height) {} - Vector() {} -}; + bool isEmpty() { return width == 0 && height == 0; } + bool isLine() { return (width == 0 && height != 0) || (width != 0 && height == 0); } -using vec2 = Vector; -using vec3 = Vector; -using vec4 = Vector; + void setEmpty() { x = y = width = height = 0; } + }; -using dvec2 = Vector; -using dvec3 = Vector; -using dvec4 = Vector; - -using ivec2 = Vector; -using ivec3 = Vector; -using ivec4 = Vector; - -using uvec2 = Vector; -using uvec3 = Vector; -using uvec4 = Vector; - -// A 2D rectangle, meant to be used for stuff like scissor rects or viewport rects -// We're never supporting 3D rectangles, because rectangles were never meant to be 3D in the first place -// x, y: Coords of the top left vertex -// width, height: Dimensions of the rectangle. Initialized to 0 if not specified. -template -struct Rectangle { - T x, y, width, height; - - std::pair topLeft() { return std::make_pair(x, y); } - std::pair topRight() { return std::make_pair(x + width, y); } - std::pair bottomLeft() { return std::make_pair(x, y + height); } - std::pair bottomRight() { return std::make_pair(x + width, y + height); } - - Rectangle() : x(0), y(0), width(0), height(0) {} - Rectangle(T x, T y, T width, T height) : x(x), y(y), width(width), height(height) {} - - bool isEmpty() { return width == 0 && height == 0; } - bool isLine() { return (width == 0 && height != 0) || (width != 0 && height == 0); } - - void setEmpty() { x = y = width = height = 0; } -}; - -using Rect = Rectangle; + using Rect = Rectangle; } // end namespace OpenGL diff --git a/src/core/CPU/cpu_dynarmic.cpp b/src/core/CPU/cpu_dynarmic.cpp index ef751b22..b4d3e8e4 100644 --- a/src/core/CPU/cpu_dynarmic.cpp +++ b/src/core/CPU/cpu_dynarmic.cpp @@ -9,7 +9,7 @@ CPU::CPU(Memory& mem) : mem(mem) { // Write some code to memory. env.MemoryWrite16(0, 0x0088); // lsls r0, r1, #2 env.MemoryWrite16(2, 0x3045); // adds r0, #69 - env.MemoryWrite16(4, 0xdf45); // swi #69 + env.MemoryWrite16(4, 0x3845); // subs r0, #69 env.MemoryWrite16(6, 0xE7FE); // b +#0 (infinite loop) // Setup registers. diff --git a/src/core/elf.cpp b/src/core/elf.cpp new file mode 100644 index 00000000..4e65e54f --- /dev/null +++ b/src/core/elf.cpp @@ -0,0 +1,24 @@ +#include "memory.hpp" +#include "elfio/elfio.hpp" + +using namespace ELFIO; + +std::optional Memory::loadELF(std::filesystem::path& path) { + elfio reader; + if (!reader.load(path.string())) { + printf("Woops failed to load ELF\n"); + return std::nullopt; + } + + auto seg_num = reader.segments.size(); + printf("Number of segments: %d\n", seg_num); + for (int i = 0; i < seg_num; ++i) { + const auto pseg = reader.segments[i]; + std::cout << " [" << i << "] 0x" << std::hex << pseg->get_flags() + << "\t0x" << pseg->get_virtual_address() << "\t0x" + << pseg->get_file_size() << "\t0x" << pseg->get_memory_size() + << std::endl; + } + + return static_cast(reader.get_entry()); +} \ No newline at end of file diff --git a/src/emulator.cpp b/src/emulator.cpp index 6988c6ea..f5e4b4ad 100644 --- a/src/emulator.cpp +++ b/src/emulator.cpp @@ -34,4 +34,12 @@ void Emulator::runFrame() { for (u32 i = 0; i < freq; i += 2) { step(); } +} + +bool Emulator::loadELF(std::filesystem::path& path) { + std::optional entrypoint = memory.loadELF(path); + if (!entrypoint.has_value()) + return false; + + Helpers::panic("Entrypoint: %08X\n", entrypoint.value()); } \ No newline at end of file diff --git a/src/main.cpp b/src/main.cpp index a74a0915..b7c91866 100644 --- a/src/main.cpp +++ b/src/main.cpp @@ -7,5 +7,9 @@ int main (int argc, char *argv[]) { Helpers::panic("Failed to initialize OpenGL"); } + auto elfPath = std::filesystem::current_path() / (argc > 1 ? argv[1] : "simple_tri.elf"); + if (emu.loadELF(elfPath)) { + Helpers::panic("Failed to load ELF file: %s", elfPath.c_str()); + } emu.run(); } \ No newline at end of file