mirror of
https://github.com/The-Powder-Toy/The-Powder-Toy.git
synced 2025-01-17 06:18:22 +01:00
7e9d9686dd
Timing is now such that the input to the gravity process (the point masses and the mask derived from gravity walls) is passed in BeforeSim and the output of the gravity process (the 2D force field) is taken also in BeforeSim, at the same time. This means that input generated by particles in frame n is passed to the gravity process at the beginning of frame n+1, the corresponding output is available by the beginning of frame n+2, and so it is visible to particles in frame n+2. Thus, gravity is now properly and predictably one frame late, though, sadly, it's displayed two frames late. This is in contrast with the case of not being late at all, which would be the case if input generated in frame n produced visible output by frame n+1, and also in contrast with what we've had until now, which was that gravity was *at least* one frame late, but it could be more out of sync than that if the gravity process for some reason took more time to produce output. Further, both input and output now make it into snapshots and saves, which fixes one half of what causes the phenomenon wherein beams of PHOT under the effect of gravity wiggle for a few frames before stabilizing after a save is loaded. The other half is that velocities are saved at very a low precision, so overall, the effect of this change on this phenomenon is negligible. It is not crucial that a client understands this new piece of info, as clients have been fine not getting gravity data from saves for years. Thus, its presence does not restrict the save to client versions newer than the one that adds this feature. An interesting question this brings up is whether settings, crucially, the enabled state of Newtonian gravity, should be considered simulation state and be also included in saves. Gravity data is now also included in the output of sim.hash, so Newtonian gravity no longer has to be disabled in order to ensure local determinism. The diff is large because I gave up on being non-invasive and renamed important structures that lots of elements use. gravp, which was functionally just hypot(gravx, gravy), was removed, because it was only ever used to display a single value in the HUD. I've also taken a few detours and made changes that really should have been in separate commits, see below, but oh well. This commit also includes a complete rework of the gravity wall area of effect discovery algorithm. The old implementation was extremely broken even beyond the usual C99-isms. Also fix Simulation.NewtonianGravity being read as an integer from powder.pref, even though it's a boolean and is in fact saved as a boolean. It's pure luck that it's worked fine until now. Also introduce PlaneBase for use with PlaneAdapter. PlaneBase is a very budget version of std::span from C++20, so budget in fact that it doesn't even store span size; this is fine because PlaneAdapter knows the span size anyway. std::basic_string_view worked for char types but this new code deals with floats. Also include blockAir data in saves unconditionally. It used to be included only if ensureDeterminism was enabled, like rngState and frameCount, but those last two are conditionally included only because some assumptions are broken if they are included, such as that of expecting a saved simulation to take different paths of evolution after each reload. blockAir has no such effect. Also remove sim.resetGravityField aka tpt.reset_gravity_field because it was agreed that it's really weird that the output of the gravity process can be changed at all from Lua, not to mention that it can only be reset to zero. No scripts to my knowledge ever used these functions.
92 lines
1.7 KiB
C++
92 lines
1.7 KiB
C++
#pragma once
|
|
#include <cmath>
|
|
#include <cstdio>
|
|
#include <cstdlib>
|
|
#include <cstddef>
|
|
#include <vector>
|
|
|
|
template<class Signed>
|
|
inline std::pair<Signed, Signed> floorDiv(Signed a, Signed b)
|
|
{
|
|
auto quo = a / b;
|
|
auto rem = a % b;
|
|
if (a < Signed(0) && rem)
|
|
{
|
|
quo -= Signed(1);
|
|
rem += b;
|
|
}
|
|
return { quo, rem };
|
|
}
|
|
|
|
template<class Signed>
|
|
inline std::pair<Signed, Signed> ceilDiv(Signed a, Signed b)
|
|
{
|
|
return floorDiv(a + b - Signed(1), b);
|
|
}
|
|
|
|
//Linear interpolation
|
|
template <typename T> inline T LinearInterpolate(T val1, T val2, T lowerCoord, T upperCoord, T coord)
|
|
{
|
|
if(lowerCoord == upperCoord) return val1;
|
|
return (((val2 - val1) / (upperCoord - lowerCoord)) * (coord - lowerCoord)) + val1;
|
|
}
|
|
|
|
|
|
//Signum function
|
|
inline int isign(int i)
|
|
{
|
|
if (i<0)
|
|
return -1;
|
|
if (i>0)
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
inline int isign(float i)
|
|
{
|
|
if (i<0)
|
|
return -1;
|
|
if (i>0)
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
inline int iabs(int i)
|
|
{
|
|
return i * isign(i);
|
|
}
|
|
|
|
inline unsigned clamp_flt(float f, float min, float max)
|
|
{
|
|
if (f<min)
|
|
return 0;
|
|
if (f>max)
|
|
return 255;
|
|
return (int)(255.0f*(f-min)/(max-min));
|
|
}
|
|
|
|
inline float restrict_flt(float f, float min, float max)
|
|
{
|
|
// Fix crash in certain cases when f is nan
|
|
if (!std::isfinite(f))
|
|
return min;
|
|
if (f < min)
|
|
return min;
|
|
if (f > max)
|
|
return max;
|
|
return f;
|
|
}
|
|
|
|
void HSV_to_RGB(int h,int s,int v,int *r,int *g,int *b);
|
|
void RGB_to_HSV(int r,int g,int b,int *h,int *s,int *v);
|
|
|
|
class ByteString;
|
|
|
|
bool byteStringEqualsString(const ByteString &str, const char *data, size_t size);
|
|
template<size_t N>
|
|
// TODO: use std::literals::string_literals::operator""s if we get rid of ByteString
|
|
bool byteStringEqualsLiteral(const ByteString &str, const char (&lit)[N])
|
|
{
|
|
return byteStringEqualsString(str, lit, N - 1U);
|
|
}
|