From e00cde609232af1cbc0971b37963e62e90843589 Mon Sep 17 00:00:00 2001 From: Robert Date: Thu, 16 Jul 2020 23:56:17 +0200 Subject: [PATCH] Initial commit --- .gitignore | 4 + .gitmodules | 3 + CMakeLists.txt | 32 ++++++ Mandelbrot/CMakeLists.txt | 29 ++++++ Mandelbrot/MainWindow.cu | 202 ++++++++++++++++++++++++++++++++++++++ Mandelbrot/MainWindow.cuh | 46 +++++++++ Mandelbrot/main.cu | 21 ++++ README.me | 1 + SDLFramework | 1 + 9 files changed, 339 insertions(+) create mode 100644 .gitignore create mode 100644 .gitmodules create mode 100644 CMakeLists.txt create mode 100644 Mandelbrot/CMakeLists.txt create mode 100644 Mandelbrot/MainWindow.cu create mode 100644 Mandelbrot/MainWindow.cuh create mode 100644 Mandelbrot/main.cu create mode 100644 README.me create mode 160000 SDLFramework diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..32dcdee --- /dev/null +++ b/.gitignore @@ -0,0 +1,4 @@ +.vs +out + +*.json \ No newline at end of file diff --git a/.gitmodules b/.gitmodules new file mode 100644 index 0000000..3a2e7b8 --- /dev/null +++ b/.gitmodules @@ -0,0 +1,3 @@ +[submodule "SDLFramework"] + path = SDLFramework + url = https://github.com/Lauchmelder23/SDLFramework diff --git a/CMakeLists.txt b/CMakeLists.txt new file mode 100644 index 0000000..cf0f04f --- /dev/null +++ b/CMakeLists.txt @@ -0,0 +1,32 @@ +# CMakeList.txt : Top-level CMake project file, do global configuration +# and include sub-projects here. +# +cmake_minimum_required (VERSION 3.8) + +project ("Mandelbrot") +enable_language(CUDA) + +set(CMAKE_CXX_STANDARD 20) +set(CMAKE_MODULE_PATH ${CMAKE_SOURCE_DIR}) + +set(THREADS_PREFER_PTHREAD_FLAG ON) + + +if(WIN32) + set(SDL2_INCLUDE_DIRS ${CMAKE_SOURCE_DIR}/SDLFramework/3rdparty/include/SDL) + set(SDL2_LIBRARIES + ${CMAKE_SOURCE_DIR}/SDLFramework/3rdparty/lib/SDL2.lib + ${CMAKE_SOURCE_DIR}/SDLFramework/3rdparty/lib/SDL2main.lib + ) + set(SDL2_DLL + ${CMAKE_SOURCE_DIR}/SDLFramework/3rdparty/lib/SDL2.dll + ) +else() + find_package(SDL2 REQUIRED) + + SET(CMAKE_CXX_FLAGS -pthread) +endif(WIN32) + +# Include sub-projects. +add_subdirectory ("SDLFramework") +add_subdirectory ("Mandelbrot") diff --git a/Mandelbrot/CMakeLists.txt b/Mandelbrot/CMakeLists.txt new file mode 100644 index 0000000..e9816f8 --- /dev/null +++ b/Mandelbrot/CMakeLists.txt @@ -0,0 +1,29 @@ +# CMakeList.txt : CMake project for Mandelbrot, include source and define +# project specific logic here. +# +cmake_minimum_required (VERSION 3.8) + +set(CMAKE_CUDA_FLAGS "${CMAKE_CUDA_FLAGS} --expt-relaxed-constexpr") + +# Add source to this project's executable. +add_executable (Mandelbrot "main.cu" + "MainWindow.cu" +) + +# TODO: Add tests and install targets if needed. +target_include_directories(Mandelbrot PRIVATE + ${SDL2_INCLUDE_DIRS} + ${CMAKE_CUDA_TOOLKIT_INCLUDE_DIRECTORIES} + ${CMAKE_SOURCE_DIR}/SDLFramework/src/sdlf +) + +target_link_libraries(Mandelbrot PRIVATE + ${SDL2_LIBRARIES} + sdlf +) + +if(WIN32) + add_custom_command(TARGET Mandelbrot POST_BUILD + COMMAND ${CMAKE_COMMAND} -E copy ${SDL2_DLL} $ + ) +endif(WIN32) \ No newline at end of file diff --git a/Mandelbrot/MainWindow.cu b/Mandelbrot/MainWindow.cu new file mode 100644 index 0000000..45084a8 --- /dev/null +++ b/Mandelbrot/MainWindow.cu @@ -0,0 +1,202 @@ +#include "MainWindow.cuh" +#include +#include + +#define WIN32_LEAN_AND_MEAN + +#include + +#include +#include + +#define CHECK_CUDA_ERROR(error, msg) { \ + if(error != cudaSuccess) { \ + std::cerr << msg << std::endl << cudaGetErrorString(error) << std::endl;\ + return false; \ + } \ +} + +__global__ void PopulateArrayWithIndexed(uint32_t* out, int n) +{ + int index = blockIdx.x * blockDim.x + threadIdx.x; + + if(index < n) + out[index] = ((uint32_t)(index % WIDTH) << 16 | (uint32_t)(index / WIDTH)); +} + +__global__ void Mandelbrot(uint32_t* in, uint32_t* out, + double centerX, double centerY, + double cmplxCenterX, double cmplxCenterY, + double pixelSize, + int maxIteration, int maxValue, + int n) +{ + int index = blockIdx.x * blockDim.x + threadIdx.x; + int stride = 1; + + double x, cx; + double y, cy; + double tempX; + + if(index < n) + { + x = 0; + y = 0; + + cx = (((in[index] & 0xFFFF0000) >> 16) - centerX) * pixelSize + cmplxCenterX; + cy = (((in[index] & 0x0000FFFF) >> 0) - centerY) * pixelSize - cmplxCenterY; + + for (uint32_t j = 0; j < maxIteration; j++) + { + tempX = x * x - y * y + cx; + y = 2 * x * y + cy; + x = tempX; + + if (x * x + y * y >= maxValue) + { + out[index] = 0xFFFFFFFF; + return; + } + } + + out[index] = 0x000000FF; + } +} + + +MainWindow::MainWindow() : + sf::IWindow(sf::Vector2u(1000, 800), + sf::UnitVector2i * SDL_WINDOWPOS_UNDEFINED, + "Mandelbrot"), + pRender(nullptr) +{ + +} + +void MainWindow::SetFunction(FractalSequence func) +{ + pFunction = func; +} + +bool MainWindow::OnCreate() +{ + pRender = SDL_CreateTexture(m_pRenderer, + SDL_PIXELFORMAT_RGBA8888, SDL_TEXTUREACCESS_STREAMING, + WIDTH, HEIGHT + ); + if (pRender == nullptr) + { + std::cerr << "Failed to initialize Render Texture: " << SDL_GetError() << std::endl; + return false; + } + + SDL_SetRenderDrawColor(m_pRenderer, 0, 0, 0, 255); + + return true; +} + +bool MainWindow::GetMandelbrotColors(Uint32** pixels) +{ + constexpr int ARR_SIZE = WIDTH * HEIGHT; + + // Calculate / set thread/block size + constexpr int threadsPerBlock = 256; + constexpr int blocksPerGrid = (ARR_SIZE + threadsPerBlock - 1) / threadsPerBlock; + + cudaError_t err; + + // Create device memory for screen indices + if (cuda_screen == NULL) + { + err = cudaMalloc((void**)&cuda_screen, ARR_SIZE * sizeof(uint32_t)); + CHECK_CUDA_ERROR(err, "Failed to create array on device"); + + // Call device kernel to fill array + PopulateArrayWithIndexed<<>>(cuda_screen, ARR_SIZE); + err = cudaGetLastError(); + CHECK_CUDA_ERROR(err, "Failed to launch kernel: "); + } + + // Create device memory for color data; + if (cuda_colors == NULL) + { + err = cudaMalloc((void**)&cuda_colors, ARR_SIZE * sizeof(uint32_t)); + CHECK_CUDA_ERROR(err, "Failed to create array on device"); + } + + // Call device kernel to calculate mandelbrot colors for each pixel + Mandelbrot<<>>(cuda_screen, cuda_colors, + WIDTH / 2, HEIGHT / 2, + -0.77568377f, 0.13646737, + xInterval / WIDTH, + 1000, 100000, + ARR_SIZE); + err = cudaGetLastError(); + CHECK_CUDA_ERROR(err, "Failed to launch kernel: "); + + // Free given memory to avoid memory leaks + if (*pixels == nullptr) + { + *pixels = (Uint32*)malloc(ARR_SIZE * sizeof(Uint32)); + } + + err = cudaMemcpy(*pixels, cuda_colors, ARR_SIZE * sizeof(Uint32), cudaMemcpyDeviceToHost); + CHECK_CUDA_ERROR(err, "Failed to memcpy from device to host"); + + return true; +} + +bool MainWindow::OnUpdate(double frametime) +{ + SDL_SetWindowTitle(m_pWindow, (std::to_string(1.f / frametime) + std::string(" FPS")).c_str()); + + static int pitch = 0; + SDL_LockTexture(pRender, NULL, (void**)&pPixels, &pitch); + + if (!GetMandelbrotColors(&pPixels)) + { + SDL_UnlockTexture(pRender); + return false; + } + + SDL_UnlockTexture(pRender); + + xInterval -= frametime * xInterval * 0.5; + + return true; +} + +void MainWindow::OnRender(SDL_Renderer* renderer) +{ + SDL_RenderClear(m_pRenderer); + SDL_RenderCopy(m_pRenderer, pRender, NULL, NULL); +} + +void MainWindow::OnClose() +{ + //SDL_DestroyTexture(pRender); + + //free(pPixels); + + //cudaError_t err; + + // Free device memory for color data + // err = cudaFree(cuda_colors); + + // Free device memory for screen indices + // err = cudaFree((void*)cuda_screen); +} + +fComplex64 MainWindow::MapComplex(const fComplex64& value, const SDL_Rect& from, const SDL_Rect& to) +{ + fComplex64 ret( + (value.real() - from.x) * (to.w - to.x) / (from.w - from.x) + to.x, + (value.imag() - from.y) * (to.h - to.y) / (from.h - from.y) + to.y + ); + return ret; +} + +fComplex64 MainWindow::MapComplex(const fComplex64& value, const fComplex64& centerPoint, double pixelSize) +{ + return ((value - centerPoint) * pixelSize); +} diff --git a/Mandelbrot/MainWindow.cuh b/Mandelbrot/MainWindow.cuh new file mode 100644 index 0000000..716f8b2 --- /dev/null +++ b/Mandelbrot/MainWindow.cuh @@ -0,0 +1,46 @@ +#pragma once + +#include +#include + +#include +#include "Window.hpp" + +typedef std::complex fComplex64; +typedef std::function FractalSequence; + +constexpr uint16_t WIDTH = 1000; +constexpr uint16_t HEIGHT = 800; + + +class MainWindow: public sf::IWindow +{ +public: + MainWindow(); + ~MainWindow() = default; + + void SetFunction(FractalSequence func); + +private: + bool OnCreate() override; + bool OnUpdate(double frametime) override; + void OnRender(SDL_Renderer* renderer) override; + void OnClose() override; + + fComplex64 MapComplex(const fComplex64& value, const SDL_Rect& from, const SDL_Rect& to); + fComplex64 MapComplex(const fComplex64& value, const fComplex64& centerPoint, double pixelSize); + + bool GetMandelbrotColors(Uint32** pixels); + +private: + FractalSequence pFunction; + + Uint32* cuda_screen = nullptr; + Uint32* cuda_colors = nullptr; + + double xInterval = 3.f; + + Uint32* pPixels = nullptr; + + SDL_Texture* pRender; +}; diff --git a/Mandelbrot/main.cu b/Mandelbrot/main.cu new file mode 100644 index 0000000..7a18001 --- /dev/null +++ b/Mandelbrot/main.cu @@ -0,0 +1,21 @@ +#include + +#include "MainWindow.cuh" + +fComplex64 func(fComplex64 z, fComplex64 c) +{ + return z * z + c; +} + +int main(int argc, char** argv) +{ + SDL_Init(SDL_INIT_VIDEO); + + MainWindow window; + window.SetFunction(func); + + window.Launch(false); + window.Stop(); + + return 0; +} \ No newline at end of file diff --git a/README.me b/README.me new file mode 100644 index 0000000..df064ad --- /dev/null +++ b/README.me @@ -0,0 +1 @@ +# Mandelbrot \ No newline at end of file diff --git a/SDLFramework b/SDLFramework new file mode 160000 index 0000000..f46ca19 --- /dev/null +++ b/SDLFramework @@ -0,0 +1 @@ +Subproject commit f46ca19d3294f2dd9480c6bccf59d734718240db