added more retentive stuff

This commit is contained in:
Lauchmelder 2021-12-16 16:35:37 +01:00
parent 75df9ad96d
commit b947b0b263
8 changed files with 117 additions and 107 deletions

View file

@ -1,7 +1,7 @@
add_library(nm_utils STATIC add_library(nm_utils STATIC
"Window.cpp" "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_include_directories(nm_utils PUBLIC ${SDL2_INCLUDE_DIRS} ${CMAKE_CURRENT_SOURCE_DIR})
target_link_libraries(nm_utils PUBLIC ${SDL2_LIBRARIES}) target_link_libraries(nm_utils PUBLIC ${SDL2_LIBRARIES})

View file

@ -3,6 +3,76 @@
#include <vector> #include <vector>
#include <array> #include <array>
#include <functional> #include <functional>
#include <memory>
template<
typename Type,
unsigned int AttentionSpan,
bool IsPointerType>
class _RetentiveEntityBase;
template<
typename Type,
unsigned int AttentionSpan>
class _RetentiveEntityBase<Type, AttentionSpan, false>
{
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<Type, AttentionSpan + 1> data;
};
template<
typename Type,
unsigned int AttentionSpan>
class _RetentiveEntityBase<Type, AttentionSpan, true>
{
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<std::shared_ptr<Type>, AttentionSpan + 1> data; // Shared for move semantics
};
/** /**
* @brief Abstract base type for all retentive things * @brief Abstract base type for all retentive things
@ -13,8 +83,8 @@
template< template<
typename Type, typename Type,
unsigned int AttentionSpan, unsigned int AttentionSpan,
typename std::enable_if_t<(AttentionSpan > 0), bool> = true> bool IsPointerType = false>
class RetentiveEntity class RetentiveEntity : public _RetentiveEntityBase<Type, AttentionSpan, IsPointerType>
{ {
public: public:
RetentiveEntity() {} RetentiveEntity() {}
@ -23,27 +93,6 @@ public:
RetentiveEntity<Type, AttentionSpan>& operator=(const RetentiveEntity<Type, AttentionSpan>& other) = delete; RetentiveEntity<Type, AttentionSpan>& operator=(const RetentiveEntity<Type, AttentionSpan>& other) = delete;
RetentiveEntity<Type, AttentionSpan>& operator=(RetentiveEntity<Type, AttentionSpan>&& other) = delete; RetentiveEntity<Type, AttentionSpan>& operator=(RetentiveEntity<Type, AttentionSpan>&& 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 * @brief Evolve the data in the entity
* *
@ -92,8 +141,4 @@ protected:
* @brief Swaps the objects in the array * @brief Swaps the objects in the array
*/ */
virtual void CycleGenerations() = 0; virtual void CycleGenerations() = 0;
protected:
std::array<Type, AttentionSpan + 1> data;
std::function<void(void)> rule = std::bind([]() {}); // Do nothing by default
}; };

View file

@ -21,9 +21,8 @@
*/ */
template< template<
typename Type, typename Type,
unsigned int AttentionSpan, unsigned int AttentionSpan>
typename std::enable_if_t<(AttentionSpan > 0), bool> = true> class RetentiveObject : public RetentiveEntity<Type, AttentionSpan, true>
class RetentiveObject : public RetentiveEntity<Type*, AttentionSpan>
{ {
public: public:
/** /**
@ -34,7 +33,7 @@ public:
// Create new vectors for every generation that needs to be remembered // Create new vectors for every generation that needs to be remembered
for (int n = 0; n <= AttentionSpan; n++) for (int n = 0; n <= AttentionSpan; n++)
{ {
data[n] = new Type(); data[n] = std::make_shared<Type>();
} }
} }
@ -45,10 +44,10 @@ public:
*/ */
RetentiveObject(const Type& initVal) 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++) for (int n = 0; n <= AttentionSpan; n++)
{ {
data[n] = new Type(initVal); data[n] = std::make_shared<Type>(initVal);
} }
} }
@ -77,7 +76,6 @@ public:
for (int n = 0; n <= AttentionSpan; n++) for (int n = 0; n <= AttentionSpan; n++)
{ {
data[n] = other.data[n]; data[n] = other.data[n];
other.data[n] = nullptr;
} }
return *this; return *this;
@ -103,36 +101,6 @@ public:
*this = other; *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: private:
/** /**
* @brief Swaps the objects in the array * @brief Swaps the objects in the array

View file

@ -12,8 +12,8 @@ VectorField::VectorField() :
VectorField::VectorField(int width, int height) : VectorField::VectorField(int width, int height) :
width(width), height(height) width(width), height(height)
{ {
horizontal = std::vector<double>(width * height, 0.0); horizontal = std::vector<double>(width * height, 0.0);
vertical = std::vector<double>(width * height, 0.0); vertical = std::vector<double>(width * height, 0.0);
biggestMagnitude = 1.0f; biggestMagnitude = 1.0f;
} }
@ -62,7 +62,7 @@ void VectorField::RecalculateMagnitude()
{ {
double u = horizontal[y * this->width + x]; double u = horizontal[y * this->width + x];
double v = vertical[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); biggestMagnitude = std::max(biggestMagnitude, magnitude);
} }

View file

@ -4,7 +4,7 @@
cmake_minimum_required (VERSION 3.8) cmake_minimum_required (VERSION 3.8)
# Add source to this project's executable. # 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_include_directories(EulerFluid PUBLIC nm_utils)
target_link_libraries(EulerFluid PRIVATE nm_utils) target_link_libraries(EulerFluid PRIVATE nm_utils)

View file

@ -2,6 +2,7 @@
#include <iostream> #include <iostream>
#include <SDL.h> #include <SDL.h>
#include "VectorField.hpp" #include "VectorField.hpp"
#define VALUE(arr, x, y) ((arr)[(y) * this->size + (x)]) #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 velocity = RetentiveObject<VectorField, 1>(VectorField(this->size, this->size, hori, vert));
prevVel = new VectorField(this->size, this->size, hori, vert);
} }
FluidField::~FluidField() FluidField::~FluidField()
{ {
delete prevVel; // Do nothing
delete vel;
} }
void FluidField::AddSource(int x, int y, double dens, double dt) 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) void FluidField::AddFlow(int x, int y, double dx, double dy, double dt)
{ {
VALUE(vel->horizontal, x, y) += dt * dx; velocity.Current().horizontal[IDX(x, y, size)] += dt * dx;
VALUE(vel->vertical, x, y) += dt * dy; velocity.Current().vertical[IDX(x, y, size)] += dt * dy;
} }
void FluidField::ApplyBoundaryConditions(BoundaryCondition condition, std::vector<double>& field) void FluidField::ApplyBoundaryConditions(BoundaryCondition condition, std::vector<double>& field)
@ -101,8 +100,8 @@ void FluidField::Advect(double dt)
{ {
for (int j = 1; j <= N; j++) for (int j = 1; j <= N; j++)
{ {
double x = i - dt0 * VALUE(vel->horizontal, i, j); double x = i - dt0 * velocity.Current().horizontal[IDX(i, j, size)];
double y = j - dt0 * VALUE(vel->vertical, i, j); double y = j - dt0 * velocity.Current().vertical[IDX(i, j, size)];
if (x < 0.5) x = 0.5; if (x < 0.5) x = 0.5;
if (x > N + 0.5) x = N + 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++) 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); 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);
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().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, velocity.Current().horizontal);
ApplyBoundaryConditions(BoundaryCondition::Continuous, vel->vertical); ApplyBoundaryConditions(BoundaryCondition::Continuous, velocity.Current().vertical);
} }
} }
@ -157,8 +156,8 @@ void FluidField::AdvectVelocity(double dt)
{ {
for (int j = 1; j <= N; j++) for (int j = 1; j <= N; j++)
{ {
double x = i - dt0 * VALUE(prevVel->horizontal, i, j); double x = i - dt0 * velocity[1].horizontal[IDX(i, j, size)];
double y = j - dt0 * VALUE(prevVel->vertical, i, j); double y = j - dt0 * velocity[1].vertical[IDX(i, j, size)];
if (x < 0.5) x = 0.5; if (x < 0.5) x = 0.5;
if (x > N + 0.5) x = N + 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 t1 = y - j0;
double t0 = 1 - t1; double t0 = 1 - t1;
VALUE(vel->horizontal, i, j) = s0 * (t0 * VALUE(prevVel->horizontal, i0, j0) + t1 * VALUE(prevVel->horizontal, i0, 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 * VALUE(prevVel->horizontal, i1, j0) + t1 * VALUE(prevVel->horizontal, i1, j1)); 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)) + 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 * VALUE(prevVel->vertical, i1, j0) + t1 * VALUE(prevVel->vertical, i1, j1)); s1 * (t0 * velocity[1].vertical[IDX(i1, j0, size)] + t1 * velocity[1].vertical[IDX(i1, j1, size)]);
} }
} }
ApplyBoundaryConditions(BoundaryCondition::InvertHorizontal, vel->horizontal); ApplyBoundaryConditions(BoundaryCondition::InvertHorizontal, velocity.Current().horizontal);
ApplyBoundaryConditions(BoundaryCondition::InvertVertical, vel->vertical); ApplyBoundaryConditions(BoundaryCondition::InvertVertical, velocity.Current().vertical);
} }
void FluidField::VelocityStep(double visc, double dt) void FluidField::VelocityStep(double visc, double dt)
@ -206,11 +205,9 @@ void FluidField::VelocityStep(double visc, double dt)
lastMouseX = dx; lastMouseX = dx;
lastMouseY = dy; lastMouseY = dy;
std::swap(vel, prevVel); velocity.Evolve(std::bind(&FluidField::DiffuseVelocity, this, visc, dt));
DiffuseVelocity(visc, dt);
Project(); Project();
std::swap(vel, prevVel); velocity.Evolve(std::bind(&FluidField::AdvectVelocity, this, dt));
AdvectVelocity(dt);
Project(); Project();
// vel->RecalculateMagnitude(); // vel->RecalculateMagnitude();
@ -225,13 +222,13 @@ void FluidField::Project()
{ {
for (int j = 1; j <= N; j++) 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)); 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)]);
VALUE(prevVel->horizontal, i, j) = 0; velocity[1].horizontal[IDX(i, j, size)] = 0;
} }
} }
ApplyBoundaryConditions(BoundaryCondition::Continuous, prevVel->horizontal); ApplyBoundaryConditions(BoundaryCondition::Continuous, velocity[1].horizontal);
ApplyBoundaryConditions(BoundaryCondition::Continuous, prevVel->vertical); ApplyBoundaryConditions(BoundaryCondition::Continuous, velocity[1].vertical);
for (int k = 0; k < 20; k++) for (int k = 0; k < 20; k++)
{ {
@ -239,24 +236,24 @@ void FluidField::Project()
{ {
for (int j = 1; j <= N; j++) 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 i = 1; i <= N; i++)
{ {
for (int j = 1; j <= N; j++) 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; 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;
VALUE(vel->vertical, i, j) -= 0.5 * (VALUE(prevVel->horizontal, i, j + 1) - VALUE(prevVel->horizontal, i, j - 1)) / 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::InvertHorizontal, velocity[1].horizontal);
ApplyBoundaryConditions(BoundaryCondition::InvertVertical, prevVel->vertical); ApplyBoundaryConditions(BoundaryCondition::InvertVertical, velocity[1].vertical);
} }
void FluidField::DensityStep(double diff, double dt) 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);
} }

View file

@ -1,9 +1,10 @@
#pragma once #pragma once
#include <vector> #include <vector>
#include "VectorField.hpp"
#include "RetentiveArray.hpp" #include "RetentiveArray.hpp"
#include "RetentiveObject.hpp"
class VectorField;
struct SDL_Renderer; struct SDL_Renderer;
struct SDL_Rect; struct SDL_Rect;
@ -38,8 +39,7 @@ public:
private: private:
int size; int size;
VectorField* vel; RetentiveObject<VectorField, 1> velocity;
VectorField* prevVel;
RetentiveArray<double, 1> density; RetentiveArray<double, 1> density;
int lastMouseX, lastMouseY; int lastMouseX, lastMouseY;