From b947b0b263ec386ab7ceddbbef2c5b710d5cbb9f Mon Sep 17 00:00:00 2001 From: Lauchmelder Date: Thu, 16 Dec 2021 16:35:37 +0100 Subject: [PATCH] added more retentive stuff --- lib/CMakeLists.txt | 2 +- lib/RetentiveEntity.hpp | 99 ++++++++++++++++++------- lib/RetentiveObject.hpp | 42 ++--------- {src/EulerFluid => lib}/VectorField.cpp | 6 +- {src/EulerFluid => lib}/VectorField.hpp | 0 src/EulerFluid/CMakeLists.txt | 2 +- src/EulerFluid/FluidField.cpp | 67 ++++++++--------- src/EulerFluid/FluidField.hpp | 6 +- 8 files changed, 117 insertions(+), 107 deletions(-) rename {src/EulerFluid => lib}/VectorField.cpp (93%) rename {src/EulerFluid => lib}/VectorField.hpp (100%) diff --git a/lib/CMakeLists.txt b/lib/CMakeLists.txt index 9b2560d..6b17d91 100644 --- a/lib/CMakeLists.txt +++ b/lib/CMakeLists.txt @@ -1,7 +1,7 @@ add_library(nm_utils STATIC "Window.cpp" - "RetentiveArray.hpp" "RetentiveObject.hpp" "RetentiveEntity.hpp") + "RetentiveArray.hpp" "RetentiveObject.hpp" "RetentiveEntity.hpp" "VectorField.hpp" "VectorField.cpp") target_include_directories(nm_utils PUBLIC ${SDL2_INCLUDE_DIRS} ${CMAKE_CURRENT_SOURCE_DIR}) target_link_libraries(nm_utils PUBLIC ${SDL2_LIBRARIES}) diff --git a/lib/RetentiveEntity.hpp b/lib/RetentiveEntity.hpp index 79203f0..cf79a7b 100644 --- a/lib/RetentiveEntity.hpp +++ b/lib/RetentiveEntity.hpp @@ -3,6 +3,76 @@ #include #include #include +#include + +template< + typename Type, + unsigned int AttentionSpan, + bool IsPointerType> +class _RetentiveEntityBase; + +template< + typename Type, + unsigned int AttentionSpan> +class _RetentiveEntityBase +{ +public: + /** + * @brief Get the entity from `index` generations ago + * + * @param index Amount of generations to go backwards in time + * @return The entity from before `index` generations + */ + Type& operator[](size_t index) + { + return data[index]; + } + + /** + * @brief Get the most up-to-date entity + * + * @return The entity with generation 0 + */ + Type& Current() + { + return data[0]; + } + +protected: + std::array data; +}; + + +template< + typename Type, + unsigned int AttentionSpan> +class _RetentiveEntityBase +{ +public: + /** + * @brief Get the entity from `index` generations ago + * + * @param index Amount of generations to go backwards in time + * @return The entity from before `index` generations + */ + Type& operator[](size_t index) + { + return *(data[index]); + } + + /** + * @brief Get the most up-to-date entity + * + * @return The entity with generation 0 + */ + Type& Current() + { + return *(data[0]); + } + +protected: + std::array, AttentionSpan + 1> data; // Shared for move semantics +}; /** * @brief Abstract base type for all retentive things @@ -13,8 +83,8 @@ template< typename Type, unsigned int AttentionSpan, - typename std::enable_if_t<(AttentionSpan > 0), bool> = true> -class RetentiveEntity + bool IsPointerType = false> +class RetentiveEntity : public _RetentiveEntityBase { public: RetentiveEntity() {} @@ -23,27 +93,6 @@ public: RetentiveEntity& operator=(const RetentiveEntity& other) = delete; RetentiveEntity& operator=(RetentiveEntity&& other) = delete; - /** - * @brief Get the entity from `index` generations ago - * - * @param index Amount of generations to go backwards in time - * @return The entity from before `index` generations - */ - virtual Type& operator[](size_t index) - { - return data[index]; - } - - /** - * @brief Get the most up-to-date entity - * - * @return The entity with generation 0 - */ - virtual Type& Current() - { - return data[0]; - } - /** * @brief Evolve the data in the entity * @@ -92,8 +141,4 @@ protected: * @brief Swaps the objects in the array */ virtual void CycleGenerations() = 0; - -protected: - std::array data; - std::function rule = std::bind([]() {}); // Do nothing by default }; diff --git a/lib/RetentiveObject.hpp b/lib/RetentiveObject.hpp index 397ebf9..1e9fbce 100644 --- a/lib/RetentiveObject.hpp +++ b/lib/RetentiveObject.hpp @@ -21,9 +21,8 @@ */ template< typename Type, - unsigned int AttentionSpan, - typename std::enable_if_t<(AttentionSpan > 0), bool> = true> -class RetentiveObject : public RetentiveEntity + unsigned int AttentionSpan> +class RetentiveObject : public RetentiveEntity { public: /** @@ -34,7 +33,7 @@ public: // Create new vectors for every generation that needs to be remembered for (int n = 0; n <= AttentionSpan; n++) { - data[n] = new Type(); + data[n] = std::make_shared(); } } @@ -45,10 +44,10 @@ public: */ RetentiveObject(const Type& initVal) { - // Create new vectors for every generation that needs to be remembered + // Create new objects for every generation that needs to be remembered for (int n = 0; n <= AttentionSpan; n++) { - data[n] = new Type(initVal); + data[n] = std::make_shared(initVal); } } @@ -77,7 +76,6 @@ public: for (int n = 0; n <= AttentionSpan; n++) { data[n] = other.data[n]; - other.data[n] = nullptr; } return *this; @@ -103,36 +101,6 @@ public: *this = other; } - ~RetentiveObject() - { - for (int n = 0; n <= AttentionSpan; n++) - { - if (data[n] != nullptr) - delete data[n]; - } - } - - /** - * @brief Get the object from `index` generations ago - * - * @param index Amount of generations to go backwards in time - * @return The object from before `index` generations - */ - Type& operator[](size_t index) override - { - return *(data[index]); - } - - /** - * @brief Get the most up-to-date object - * - * @return The object with generation 0 - */ - Type& Current() override - { - return *(data[0]); - } - private: /** * @brief Swaps the objects in the array diff --git a/src/EulerFluid/VectorField.cpp b/lib/VectorField.cpp similarity index 93% rename from src/EulerFluid/VectorField.cpp rename to lib/VectorField.cpp index 66337fd..7118a45 100644 --- a/src/EulerFluid/VectorField.cpp +++ b/lib/VectorField.cpp @@ -12,8 +12,8 @@ VectorField::VectorField() : VectorField::VectorField(int width, int height) : width(width), height(height) { - horizontal = std::vector(width * height, 0.0); - vertical = std::vector(width * height, 0.0); + horizontal = std::vector(width * height, 0.0); + vertical = std::vector(width * height, 0.0); biggestMagnitude = 1.0f; } @@ -62,7 +62,7 @@ void VectorField::RecalculateMagnitude() { double u = horizontal[y * this->width + x]; double v = vertical[y * this->width + x]; - double magnitude = u*u + v*v; + double magnitude = u * u + v * v; biggestMagnitude = std::max(biggestMagnitude, magnitude); } diff --git a/src/EulerFluid/VectorField.hpp b/lib/VectorField.hpp similarity index 100% rename from src/EulerFluid/VectorField.hpp rename to lib/VectorField.hpp diff --git a/src/EulerFluid/CMakeLists.txt b/src/EulerFluid/CMakeLists.txt index dc62eb0..28789b9 100644 --- a/src/EulerFluid/CMakeLists.txt +++ b/src/EulerFluid/CMakeLists.txt @@ -4,7 +4,7 @@ cmake_minimum_required (VERSION 3.8) # Add source to this project's executable. -add_executable (EulerFluid "main.cpp" "EulerFluid.hpp" "EulerFluid.cpp" "VectorField.hpp" "VectorField.cpp" "FluidField.hpp" "FluidField.cpp") +add_executable (EulerFluid "main.cpp" "EulerFluid.hpp" "EulerFluid.cpp" "FluidField.hpp" "FluidField.cpp") target_include_directories(EulerFluid PUBLIC nm_utils) target_link_libraries(EulerFluid PRIVATE nm_utils) diff --git a/src/EulerFluid/FluidField.cpp b/src/EulerFluid/FluidField.cpp index c24e368..d988374 100644 --- a/src/EulerFluid/FluidField.cpp +++ b/src/EulerFluid/FluidField.cpp @@ -2,6 +2,7 @@ #include #include + #include "VectorField.hpp" #define VALUE(arr, x, y) ((arr)[(y) * this->size + (x)]) @@ -33,14 +34,12 @@ FluidField::FluidField(int size) : } } - vel = new VectorField(this->size, this->size, hori, vert); // gonna do the same inefficient calculations twice, why not - prevVel = new VectorField(this->size, this->size, hori, vert); + velocity = RetentiveObject(VectorField(this->size, this->size, hori, vert)); } FluidField::~FluidField() { - delete prevVel; - delete vel; + // Do nothing } void FluidField::AddSource(int x, int y, double dens, double dt) @@ -51,8 +50,8 @@ void FluidField::AddSource(int x, int y, double dens, double dt) void FluidField::AddFlow(int x, int y, double dx, double dy, double dt) { - VALUE(vel->horizontal, x, y) += dt * dx; - VALUE(vel->vertical, x, y) += dt * dy; + velocity.Current().horizontal[IDX(x, y, size)] += dt * dx; + velocity.Current().vertical[IDX(x, y, size)] += dt * dy; } void FluidField::ApplyBoundaryConditions(BoundaryCondition condition, std::vector& field) @@ -101,8 +100,8 @@ void FluidField::Advect(double dt) { for (int j = 1; j <= N; j++) { - double x = i - dt0 * VALUE(vel->horizontal, i, j); - double y = j - dt0 * VALUE(vel->vertical, i, j); + double x = i - dt0 * velocity.Current().horizontal[IDX(i, j, size)]; + double y = j - dt0 * velocity.Current().vertical[IDX(i, j, size)]; if (x < 0.5) x = 0.5; if (x > N + 0.5) x = N + 0.5; @@ -138,13 +137,13 @@ void FluidField::DiffuseVelocity(double visc, double dt) { for (int j = 1; j <= N; j++) { - VALUE(vel->horizontal, i, j) = (VALUE(prevVel->horizontal, i, j) + a * (VALUE(vel->horizontal, i - 1, j) + VALUE(vel->horizontal, i + 1, j) + VALUE(vel->horizontal, i, j - 1) + VALUE(vel->horizontal, i, j + 1))) / (1 + 4 * a); - VALUE(vel->vertical, i, j) = (VALUE(prevVel->vertical, i, j) + a * (VALUE(vel->vertical, i - 1, j) + VALUE(vel->vertical, i + 1, j) + VALUE(vel->vertical, i, j - 1) + VALUE(vel->vertical, i, j + 1))) / (1 + 4 * a); + velocity.Current().horizontal[IDX(i, j, size)] = (velocity[1].horizontal[IDX(i, j, size)] + a * (velocity[1].horizontal[IDX(i - 1, j, size)] + velocity[1].horizontal[IDX(i + 1, j, size)] + velocity[1].horizontal[IDX(i, j - 1, size)] + velocity[1].horizontal[IDX(i, j + 1, size)])) / (1 + 4 * a); + velocity.Current().vertical[IDX(i, j, size)] = (velocity[1].vertical[IDX(i, j, size)] + a * (velocity[1].vertical[IDX(i - 1, j, size)] + velocity[1].vertical[IDX(i + 1, j, size)] + velocity[1].vertical[IDX(i, j - 1, size)] + velocity[1].vertical[IDX(i, j + 1, size)])) / (1 + 4 * a); } } - ApplyBoundaryConditions(BoundaryCondition::Continuous, vel->horizontal); - ApplyBoundaryConditions(BoundaryCondition::Continuous, vel->vertical); + ApplyBoundaryConditions(BoundaryCondition::Continuous, velocity.Current().horizontal); + ApplyBoundaryConditions(BoundaryCondition::Continuous, velocity.Current().vertical); } } @@ -157,8 +156,8 @@ void FluidField::AdvectVelocity(double dt) { for (int j = 1; j <= N; j++) { - double x = i - dt0 * VALUE(prevVel->horizontal, i, j); - double y = j - dt0 * VALUE(prevVel->vertical, i, j); + double x = i - dt0 * velocity[1].horizontal[IDX(i, j, size)]; + double y = j - dt0 * velocity[1].vertical[IDX(i, j, size)]; if (x < 0.5) x = 0.5; if (x > N + 0.5) x = N + 0.5; @@ -175,16 +174,16 @@ void FluidField::AdvectVelocity(double dt) double t1 = y - j0; double t0 = 1 - t1; - VALUE(vel->horizontal, i, j) = s0 * (t0 * VALUE(prevVel->horizontal, i0, j0) + t1 * VALUE(prevVel->horizontal, i0, j1)) + - s1 * (t0 * VALUE(prevVel->horizontal, i1, j0) + t1 * VALUE(prevVel->horizontal, i1, j1)); + velocity.Current().horizontal[IDX(i, j, size)] = s0 * (t0 * velocity[1].horizontal[IDX(i0, j0, size)] + t1 * velocity[1].horizontal[IDX(i0, j1, size)]) + + s1 * (t0 * velocity[1].horizontal[IDX(i1, j0, size)] + t1 * velocity[1].horizontal[IDX(i1, j1, size)]); - VALUE(vel->vertical, i, j) = s0 * (t0 * VALUE(prevVel->vertical, i0, j0) + t1 * VALUE(prevVel->vertical, i0, j1)) + - s1 * (t0 * VALUE(prevVel->vertical, i1, j0) + t1 * VALUE(prevVel->vertical, i1, j1)); + velocity.Current().vertical[IDX(i, j, size)] = s0 * (t0 * velocity[1].vertical[IDX(i0, j0, size)] + t1 * velocity[1].vertical[IDX(i0, j1, size)]) + + s1 * (t0 * velocity[1].vertical[IDX(i1, j0, size)] + t1 * velocity[1].vertical[IDX(i1, j1, size)]); } } - ApplyBoundaryConditions(BoundaryCondition::InvertHorizontal, vel->horizontal); - ApplyBoundaryConditions(BoundaryCondition::InvertVertical, vel->vertical); + ApplyBoundaryConditions(BoundaryCondition::InvertHorizontal, velocity.Current().horizontal); + ApplyBoundaryConditions(BoundaryCondition::InvertVertical, velocity.Current().vertical); } void FluidField::VelocityStep(double visc, double dt) @@ -206,11 +205,9 @@ void FluidField::VelocityStep(double visc, double dt) lastMouseX = dx; lastMouseY = dy; - std::swap(vel, prevVel); - DiffuseVelocity(visc, dt); + velocity.Evolve(std::bind(&FluidField::DiffuseVelocity, this, visc, dt)); Project(); - std::swap(vel, prevVel); - AdvectVelocity(dt); + velocity.Evolve(std::bind(&FluidField::AdvectVelocity, this, dt)); Project(); // vel->RecalculateMagnitude(); @@ -225,13 +222,13 @@ void FluidField::Project() { for (int j = 1; j <= N; j++) { - VALUE(prevVel->vertical, i, j) = -0.5 * h * (VALUE(vel->horizontal, i + 1, j) - VALUE(vel->horizontal, i - 1, j) + VALUE(vel->vertical, i, j + 1) - VALUE(vel->vertical, i, j - 1)); - VALUE(prevVel->horizontal, i, j) = 0; + velocity[1].vertical[IDX(i, j, size)] = -0.5 * h * (velocity.Current().vertical[IDX(i + 1, j, size)] - velocity.Current().vertical[IDX(i - 1, j, size)] + velocity.Current().vertical[IDX(i, j + 1, size)] - velocity.Current().vertical[IDX(i, j - 1, size)]); + velocity[1].horizontal[IDX(i, j, size)] = 0; } } - ApplyBoundaryConditions(BoundaryCondition::Continuous, prevVel->horizontal); - ApplyBoundaryConditions(BoundaryCondition::Continuous, prevVel->vertical); + ApplyBoundaryConditions(BoundaryCondition::Continuous, velocity[1].horizontal); + ApplyBoundaryConditions(BoundaryCondition::Continuous, velocity[1].vertical); for (int k = 0; k < 20; k++) { @@ -239,24 +236,24 @@ void FluidField::Project() { for (int j = 1; j <= N; j++) { - VALUE(prevVel->horizontal, i, j) = (VALUE(prevVel->vertical, i, j) + VALUE(prevVel->horizontal, i - 1, j) + VALUE(prevVel->horizontal, i + 1, j) + VALUE(prevVel->vertical, i, j - 1) + VALUE(prevVel->vertical, i, j + 1)) / 4.0; + velocity[1].horizontal[IDX(i, j, size)] = (velocity[1].vertical[IDX(i, j, size)] + velocity[1].vertical[IDX(i - 1, j, size)] + velocity[1].vertical[IDX(i + 1, j, size)] + velocity[1].vertical[IDX(i, j - 1, size)] + velocity[1].vertical[IDX(i, j + 1, size)]) / 4.0; } } - ApplyBoundaryConditions(BoundaryCondition::Continuous, prevVel->horizontal); + ApplyBoundaryConditions(BoundaryCondition::Continuous, velocity[1].horizontal); } for (int i = 1; i <= N; i++) { for (int j = 1; j <= N; j++) { - VALUE(vel->horizontal, i, j) -= 0.5 * (VALUE(prevVel->horizontal, i + 1, j) - VALUE(prevVel->horizontal, i - 1, j)) / h; - VALUE(vel->vertical, i, j) -= 0.5 * (VALUE(prevVel->horizontal, i, j + 1) - VALUE(prevVel->horizontal, i, j - 1)) / h; + velocity.Current().horizontal[IDX(i, j, size)] -= 0.5 * (velocity[1].horizontal[IDX(i + 1, j, size)] - velocity[1].horizontal[IDX(i - 1, j, size)]) / h; + velocity.Current().vertical[IDX(i, j, size)] -= 0.5 * (velocity[1].horizontal[IDX(i, j + 1, size)] - velocity[1].horizontal[IDX(i, j - 1, size)]) / h; } } - ApplyBoundaryConditions(BoundaryCondition::InvertHorizontal, prevVel->horizontal); - ApplyBoundaryConditions(BoundaryCondition::InvertVertical, prevVel->vertical); + ApplyBoundaryConditions(BoundaryCondition::InvertHorizontal, velocity[1].horizontal); + ApplyBoundaryConditions(BoundaryCondition::InvertVertical, velocity[1].vertical); } void FluidField::DensityStep(double diff, double dt) @@ -300,5 +297,5 @@ void FluidField::Draw(SDL_Renderer* renderer, const SDL_Rect& target) } } - vel->Draw(renderer, target); + velocity.Current().Draw(renderer, target); } diff --git a/src/EulerFluid/FluidField.hpp b/src/EulerFluid/FluidField.hpp index 0e40600..b3c6a16 100644 --- a/src/EulerFluid/FluidField.hpp +++ b/src/EulerFluid/FluidField.hpp @@ -1,9 +1,10 @@ #pragma once #include +#include "VectorField.hpp" #include "RetentiveArray.hpp" +#include "RetentiveObject.hpp" -class VectorField; struct SDL_Renderer; struct SDL_Rect; @@ -38,8 +39,7 @@ public: private: int size; - VectorField* vel; - VectorField* prevVel; + RetentiveObject velocity; RetentiveArray density; int lastMouseX, lastMouseY;