glm::fastPow, glm::fastExp, glm::fastLog polynomial approximation

This commit is contained in:
ZvRzyan18 2025-02-20 14:01:44 +08:00 committed by GitHub
parent 2d4c4b4dd3
commit 1e94e6d17c
No known key found for this signature in database
GPG key ID: B5690EEEBB952194

View file

@ -2,12 +2,38 @@
namespace glm
{
// fastPow:
// fastPow
template<typename genType>
GLM_FUNC_QUALIFIER genType fastPow(genType x, genType y)
{
return exp(y * log(x));
}
template<>
GLM_FUNC_QUALIFIER float fastPow(float x, float y)
{
//negative y always reciprocal, pow(x, -y) = 1 / pow(x, abs(-y))
//when x is lower than 1.0, you can use this formula. pow(x, y) = 1 / glm::fastPow(1 / x, y)
// ex: when x is '0.3'... pow(0.3, y) = 1 / glm::fastPow(1 / 0.3, y)
uint32_t mantissa = 1065353216U | ((*(uint32_t*)&x) & 0x007FFFFF);
float mx = *(float*)&mantissa;
float lnx = y * float(1.44269504088896338700) * ((((*(uint32_t*)&x) >> 23)-127) * float(0.69314718055994528623) + ((((float(-0.056571767549593956) * mx + float(0.4471786103806078)) * mx - float(1.4699399880154467)) * mx + float(2.8211719569953488)) * mx - float(1.7417780977156199)));
uint32_t out = ((127U + uint32_t(lnx)) << 23U);
lnx -= uint32_t(lnx);
return (*(float*)(&out)) * ((float(0.34400110689651969) * lnx + float(0.65104678030290897)) * lnx + float(1.0024760564002857));
}
template<>
GLM_FUNC_QUALIFIER double fastPow(double x, double y)
{
//negative y always reciprocal, pow(x, -y) = 1 / pow(x, abs(-y))
uint64_t mantissa = 4607182418800017408U | ((*(uint64_t*)&x) & 0x000FFFFFFFFFFFFF);
double mx = *(double*)&mantissa;
double lnx = y * double(1.44269504088896338700) * ((((*(uint64_t*)&x) >> 52)-1023) * double(0.69314718055994528623) + ((((double(-0.056571767549593956) * mx + double(0.4471786103806078)) * mx - double(1.4699399880154467)) * mx + double(2.8211719569953488)) * mx - double(1.7417780977156199)));
uint64_t out = ((uint64_t)(1023U + uint64_t(lnx)) << 52U);
lnx -= uint64_t(lnx);
return (*(double*)(&out)) * ((double(0.34400110689651969) * lnx + double(0.65104678030290897)) * lnx + double(1.0024760564002857));
}
template<length_t L, typename T, qualifier Q>
GLM_FUNC_QUALIFIER vec<L, T, Q> fastPow(vec<L, T, Q> const& x, vec<L, T, Q> const& y)
@ -34,17 +60,30 @@ namespace glm
}
// fastExp
// Note: This function provides accurate results only for value between -1 and 1, else avoid it.
template<typename T>
GLM_FUNC_QUALIFIER T fastExp(T x)
template<>
GLM_FUNC_QUALIFIER float fastExp(float x)
{
// This has a better looking and same performance in release mode than the following code. However, in debug mode it's slower.
// return 1.0f + x * (1.0f + x * 0.5f * (1.0f + x * 0.3333333333f * (1.0f + x * 0.25 * (1.0f + x * 0.2f))));
T x2 = x * x;
T x3 = x2 * x;
T x4 = x3 * x;
T x5 = x4 * x;
return T(1) + x + (x2 * T(0.5)) + (x3 * T(0.1666666667)) + (x4 * T(0.041666667)) + (x5 * T(0.008333333333));
//negative x always reciprocal, exp(-x) = 1 / exp(abs(-x))
uint32_t out = ((127U + uint32_t(x *= 1.44269504088896338700f)) << 23U);
float mx = x - uint32_t(x);
/*
//(accurate) 2 degree polynomial exp2(), interval [0, 1]
return (*(float*)(&out)) * ((float(0.34400110689651969) * mx + float(0.65104678030290897)) * mx + float(1.0024760564002857));
*/
return (*(float*)(&out)) * (mx + 0.95696433397203284f);
}
template<>
GLM_FUNC_QUALIFIER double fastExp(double x)
{
//negative x always reciprocal, exp(-x) = 1 / exp(abs(-x))
uint64_t out = ((uint64_t)(1023U + uint64_t(x *= 1.44269504088896338700)) << 52U);
double mx = x - uint64_t(x);
/*
//(accurate) 2 degree polynomial exp2(), interval [0, 1]
return (*(double*)(&out)) * ((double(0.34400110689651969) * mx + double(0.65104678030290897)) * mx + double(1.0024760564002857));
*/
return (*(double*)(&out)) * (mx + 0.95696433397203284);
}
/* // Try to handle all values of float... but often shower than std::exp, glm::floor and the loop kill the performance
GLM_FUNC_QUALIFIER float fastExp(float x)
@ -93,14 +132,22 @@ namespace glm
return std::log(x);
}
/* Slower than the VC7.1 function...
// Slower than the VC7.1 function...
template<>
GLM_FUNC_QUALIFIER float fastLog(float x)
{
float y1 = (x - 1.0f) / (x + 1.0f);
float y2 = y1 * y1;
return 2.0f * y1 * (1.0f + y2 * (0.3333333333f + y2 * (0.2f + y2 * 0.1428571429f)));
//x less than 1 always negative log(0.2) = -log(1 / 0.2)
uint32_t mantissa = 1065353216U | ((*(uint32_t*)&x) & 0x007FFFFF);
return (((*(uint32_t*)&x) >> 23)-127) * float(0.69314718055994528623) + ((float(-0.2390307190544787) * (*(float*)&mantissa) + float(1.4033913763229589)) * (*(float*)&mantissa) - float(1.1609366765682689));
}
template<>
GLM_FUNC_QUALIFIER double fastLog(double x)
{
//x less than 1 always negative log(0.2) = -log(1 / 0.2)
uint64_t mantissa = 4607182418800017408U | ((*(uint64_t*)&x) & 0x000FFFFFFFFFFFFF);
return (((*(uint64_t*)&x) >> 52)-1023) * double(0.69314718055994528623) + ((double(-0.2390307190544787) * (*(double*)&mantissa) + double(1.4033913763229589)) * (*(double*)&mantissa) - double(1.1609366765682689));
}
*/
template<length_t L, typename T, qualifier Q>
GLM_FUNC_QUALIFIER vec<L, T, Q> fastLog(vec<L, T, Q> const& x)
@ -115,6 +162,24 @@ namespace glm
return fastExp(static_cast<genType>(0.69314718055994530941723212145818) * x);
}
template<>
GLM_FUNC_QUALIFIER float fastExp2(float x)
{
//negative x always reciprocal, exp2(-x) = 1 / exp2(abs(-x))
uint32_t out = ((127U + uint32_t(x)) << 23U);
x -= uint32_t(x);
return (*(float*)(&out)) * ((float(0.34400110689651969) * x + float(0.65104678030290897)) * x + float(1.0024760564002857));
}
template<>
GLM_FUNC_QUALIFIER double fastExp2(double x)
{
//negative x always reciprocal, exp2(-x) = 1 / exp2(abs(-x))
uint64_t out = ((uint64_t)(1023U + uint64_t(x)) << 52U);
x -= uint64_t(x);
return (*(double*)(&out)) * ((double(0.34400110689651969) * x + double(0.65104678030290897)) * x + double(1.0024760564002857));
}
template<length_t L, typename T, qualifier Q>
GLM_FUNC_QUALIFIER vec<L, T, Q> fastExp2(vec<L, T, Q> const& x)
{
@ -127,6 +192,20 @@ namespace glm
{
return fastLog(x) / static_cast<genType>(0.69314718055994530941723212145818);
}
template<>
GLM_FUNC_QUALIFIER float fastLog2(float x)
{
uint32_t mantissa = 1065353216U | ((*(uint32_t*)&x) & 0x007FFFFF);
return (((*(uint32_t*)&x) >> 23)-127) + ((float(-0.34484843300001944) * (*(float*)&mantissa) + float(2.0246657790474698)) * (*(float*)&mantissa) - float(1.674877586071156));
}
template<>
GLM_FUNC_QUALIFIER double fastLog2(double x)
{
uint64_t mantissa = 4607182418800017408U | ((*(uint64_t*)&x) & 0x000FFFFFFFFFFFFF);
return (((*(uint64_t*)&x) >> 52)-1023) + ((double(-0.34484843300001944) * (*(double*)&mantissa) + double(2.0246657790474698)) * (*(double*)&mantissa) - double(1.674877586071156));
}
template<length_t L, typename T, qualifier Q>
GLM_FUNC_QUALIFIER vec<L, T, Q> fastLog2(vec<L, T, Q> const& x)