cxcm is a c++20 library that provides constexpr
versions of some of the functions in the <cmath>
include file.
Currently this is a single header library. All you need to do is include cxcm.hxx. The functions are in the cxcm
namespace.
Originally I was working on a project dealing with periodic intervals, and I wanted to make the project as constexpr as I could. The most important functions for me from <cmath>
were std::floor()
and std::ceil()
. I looked for constexpr <cmath>
projects, and discovered that none that I found matched my requirements.
GCEM seems to be very popular, but at the time I started cxcm, GCEM wasn't conforming to the standard for smaller numbers (subnormals). I needed more fidelity to the Standard Library, so I started creating my own small project.
cxcm is its own stand-alone project for extending <cmath>
to have more functions be constexpr, targeted for c++20
. c++23
and hopefully c++26
extend the amount of functions in <cmath>
to be constexpr, but this project aims to support c++20
.
Most of these constexpr functions have counterparts in <cmath>
. Many of these functions became constexpr in c++23
or c++26
; however, this is a c++20
library, so we are stuck with doing it ourselves if we want it.
template <std::floating_point T>
constexpr bool isnan(T value) noexcept;
template <std::floating_point T>
constexpr bool isinf(T value) noexcept;
template <std::floating_point T>
constexpr int fpclassify(T value) noexcept;
template <std::floating_point T>
constexpr bool isnormal(T value) noexcept;
template <std::floating_point T>
constexpr bool isfinite(T value) noexcept;
template <std::floating_point T>
constexpr bool signbit(T value) noexcept;
template <std::floating_point T>
constexpr T copysign(T value, T sgn) noexcept;
template <std::floating_point T>
constexpr T abs(T value) noexcept;
template <std::integral T>
constexpr T abs(T value) noexcept;
template <std::floating_point T>
constexpr T fabs(T value) noexcept;
template <std::integral T>
constexpr double fabs(T value) noexcept;
cxcm::fmod
- has efficient runtime use
template <std::floating_point T>
constexpr T fmod(T x, T y) noexcept;
cxcm::trunc
- has efficient runtime use
template <std::floating_point T>
constexpr T trunc(T value) noexcept;
template <std::integral T>
constexpr double trunc(T value) noexcept;
cxcm::floor
- has efficient runtime use
template <std::floating_point T>
constexpr T floor(T value) noexcept;
template <std::integral T>
constexpr double floor(T value) noexcept;
cxcm::ceil
- has efficient runtime use
template <std::floating_point T>
constexpr T ceil(T value) noexcept;
template <std::integral T>
constexpr double ceil(T value) noexcept;
cxcm::round
- has efficient runtime use
template <std::floating_point T>
constexpr T round(T value) noexcept;
template <std::integral T>
constexpr double round(T value) noexcept;
cxcm::round_even
- not in<cmath>
template <std::floating_point T>
constexpr T round_even(T value) noexcept;
template <std::integral T>
constexpr double round_even(T value) noexcept;
Returns a number equal to the nearest integer to value. A fractional part of 0.5 will round toward the nearest even integer. (Both 3.5 and 4.5 for value will return 4.0.)
cxcm::fract
- not in<cmath>
template <std::floating_point T>
constexpr T fract(T value) noexcept;
template <std::integral T>
constexpr double fract(T /* value */) noexcept;
Returns value - floor(value)
. The positive fractional part of a floating-point number. This is a number in the range [0.0, 1.0).
cxcm::is_negative_zero
- not in<cmath>
template <std::floating_point T>
constexpr bool is_negative_zero(T val) noexcept;
Negative zero compares as if were positive zero, but this function is for those times when we explicitly want to know if a floating-point number is negative zero.
cxcm::negative_zero
- variable template, not in<cmath>
template <std::floating_point T>
constexpr inline T negative_zero = T(-0);
There are specializations for double and float for this template variable. It returns the floating-point representation of negative zero.
cxcm::sqrt
- has efficient runtime use
template <std::floating_point T>
constexpr T sqrt(T value) noexcept;
cxcm::rsqrt
- reciprocal square root - not in<cmath>
template <std::floating_point T>
constexpr T rsqrt(T value) noexcept;
Returns Newton-Raphson version of 1.0 / sqrt(value)
.
cxcm::fast_rsqrt
- fast reciprocal square root - not in<cmath>
template <std::floating_point T>
constexpr T fast_rsqrt(T value) noexcept;
This is a fast approximation to cxcm::rsqrt()
, but it may or may not be faster. For all floats, fast_rsqrt(float)
is the same as rsqrt(float)
. When comparing with rsqrt(double)
, fast_rsqrt(double)
gives pretty good approximate results:
- 0 ulps: ~68.58%
- 1 ulps: ~31.00%
- 2 ulps: ~0.42%
Current version: v1.2.0
Not sure yet how much more to try and make constexpr
. This library is meant to support the needs of other libraries, so I suppose things will be added as needed.
The point of this library is to provide constexpr
versions of certain functions. This is helpful for compile time programming, but we don't usually want to run the constexpr
versions of the functions at runtime. If we discover that we are using these functions at runtime, we revert to the std::
versions for:
trunc(std::floating_point)
floor(std::floating_point)
ceil(std::floating_point)
round(std::floating_point)
sqrt()
fmod()
This project uses doctest for testing, and we are primarily testing the conformance of trunc
, floor
, ceil
, and round
with std::
. The tests have been run on:
- MSVC 2019 - v16.11.36
- MSVC 2022 - v17.12.3
- gcc 11.4.0
- clang 16.0.6
[doctest] doctest version is "2.4.11"
[doctest] run with "--help" for options
===============================================================================
[doctest] test cases: 33 | 33 passed | 0 failed | 0 skipped
[doctest] assertions: 595 | 595 passed | 0 failed |
[doctest] Status: SUCCESS!
It might work on earlier versions, and it certainly should work on later versions.
There are no specific tests for sqrt()
and rsqrt()
, but they have been thoroughly tested. They are in 100% agreement with std::sqrt(float)
. They also appear to be in 100% agreement with std::sqrt(double)
, but it is infeasible to test the entire double
range; however, there have been billions of comparisons run and they have all been in agreement.
We are also missing tests for fmod()
, fract()
, and round_even()
.
This project uses the Boost Software License 1.0.