Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/adelks/zecalculator

C++20 math expression parsing and evaluation library
https://github.com/adelks/zecalculator

Last synced: about 1 month ago
JSON representation

C++20 math expression parsing and evaluation library

Awesome Lists containing this project

README

        

### ZeCalculator

`ZeCalculator` is a `C++20` library for parsing and computing mathematical expressions and objects.

#### Features

- Parse and compute math objects defined through a simple equation of the type ` = `
- Multi-variable functions e.g. `f(x, y) = cos(x) * sin(y)`
- Global Variables: functions without arguments, e.g. `my_var = f(1, 1)`
- Global constants: simple valued e.g. `my_constant = 1.2`
- Handle elegantly wrong math expressions
- No exceptions (only used when the user does something wrong)
- Give meaningful error messages: what went wrong, on what part of the equation
- Has no dependencies for easy packaging
- Fast repetitive evaluation of math objects.
- Know dependencies between objects (which object calls which other objects)

#### Example code

```c++
#include
#include

using namespace zc;
using namespace tl;
using namespace std;

double square(double x) { return x * x; }

int main()
{
rpn::MathWorld world;

// Notes about adding a math object to a math world:
// - Each added object exists only within the math world that creates it
// - Adding a math object returns a DynMathObject reference that is essentially an expected
// with some helper functions.
// - the variant contains all the possible objects: function, sequence, global constant, cpp function
// - the error expresses what went wrong in adding the object / parsing the equation
rpn::DynMathObject& obj1 = world.new_object();

// Assign a one parameter function named "f"
// Note that 'my_constant' is only defined later
// - this function's state will be updated once 'my_constant' is defined
// - (re)defining objects within a math world can potentially modify every other objects
obj1 = "f(x) = x + my_constant + cos(math::pi)";

// We can query the direct dependencies of any object: name, type, and where in the equation
// Note: only Function and Sequence instances with a valid equation return a non-empty set
assert(bool(obj1.direct_dependencies()
== deps::Deps{{"my_constant", {deps::Dep::VARIABLE, {11}}},
{"cos", {deps::Dep::FUNCTION, {25}}},
{"math::pi", {deps::Dep::VARIABLE, {29}}}}));

// the expected should hold an error since 'my_constant' is undefined at this point
assert(not obj1.has_value()
and obj1.error()
== Error::undefined_variable(parsing::tokens::Text{"my_constant", 11},
"f(x) = x + my_constant + cos(math::pi)"));

rpn::DynMathObject& obj2 = world.new_object();

// Assign a global constant called "my_constant" with an initial value of 3.0
obj2 = "my_constant = 3.0";

// now that 'my_constant' is defined, 'obj1' gets modified to properly hold a function
// Note that assigning to an object in the MathWorld may affect any other object
// -> Assigning to objects is NOT thread-safe
assert(obj1.holds());

// We can evaluate 'obj1' with an initializer_list
// note: we could also do it when 'my_constant' was undefined,
// in that case the result would be the same error as above
expected eval = obj1({1.0});

// Notes:
// - We know the expression is correct, otherwise the call `.value()` will throw
// - The error can be recovered with '.error()`
// - To know if the result is correct
// - call `.has_value()`
// - use the `bool()` operator on 'eval'
assert(eval.value() == 3);

// add a single argument function 'g' to the world
world.new_object() = "g(z) = 2*z + my_constant";

// assign a new equation to 'obj1'
// - Now it's the Fibonacci sequence called 'u'
// - we can force the parser to interpret it as a sequence
// - unneeded here, just for demo
// - the object will contain an error if the equation doesn't fit with the type asked for
// - even if the equation is a valid e.g. a GlobalConstant expression
obj1 = As{"u(n) = 0 ; 1 ; u(n-1) + u(n-2)"};

// should hold a Sequence now
assert(obj1.holds());

// evaluate function again and get the new value
assert(obj1({10}).value() == 55);

// C++ double(double...) functions can also be registered in a world
auto& obj3 = world.new_object();
obj3 = CppFunction{"square", square};

assert(world.evaluate("square(2)").value() == 4.);

// ======================================================================================

// the underlying objects can be retrieved either by using the fact that
// DynMathObject publicly inherits expected or the 'value_as' helper function:
// - "value" prefix just like expected::value, i.e. can throw
// - "value_as" as a wrapper to std::get<>(expected::value), can throw for two different reasons
// - the expected has an error
// - the alternative asked is not the actual one held by the variant
[[maybe_unused]] rpn::Sequence& u = obj1.value_as();
[[maybe_unused]] GlobalConstant& my_constant = obj2.value_as();

// each specific math object has extra public methods that may prove useful

return 0;
}
```

More examples of what the library can do are in the [test](./test/) folder.

#### Documentation

Classes within header files are fully documented.

Overview:
1. The entry-point class is [MathWorld](./include/zecalculator/mathworld/decl/mathworld.h)
```c++
rpn::MathWorld mathworld;
```
- Within the same `MathWorld` instance, objects can "see" and "talk" to each other.
- Every instance of `MathWorld` is filled with the usual functions and constants, see [builtin.h](./include/zecalculator/math_objects/builtin.h).
- Stores its objects in a container of [zc::DynMathObject](./include/zecalculator/math_objects/decl/dyn_math_object.h)
- Does not invalidate references to unaffected `zc::DynMathObject` instances it contains when growing/shrinking
- When adding an object, the math world returns a [zc::DynMathObject&](./include/zecalculator/math_objects/decl/dyn_math_object.h), and that can be used as a "permanent handle"
```c++
rpn::DynMathObject& obj = mathworld.new_object();
```
2. [DynMathObject](./include/zecalculator/math_objects/decl/dyn_math_object.h) is essentially (inherits) an `std::expected, zc::Error>`
- Has the `std::expected` public methods
- `bool has_value()` to know if it's currently holding an `std::variant`
- `zc::Error error()` to retrieve the error, when it's in an error state
- `std::variant& value()` to retrieve the variant (with check)
- `std::variant& operator * ()` to retrieve the variant (without check)
- Some extra helper methods
- `bool holds` to know if it's in a good state, and the variant holds the alternative `T`
- `T& value_as` to retrieve a specific alternative of the variant
- Can be assigned, using `operator =`, equations or math objects.
- [CppFunction](./include/zecalculator/math_objects/decl/cpp_function.h)
```c++
double square(double x) { return x * x; }
// ...
obj = zc::CppFunction{"square", square};
```
- [Function](./include/zecalculator/math_objects/decl/function.h)
```c++
// automatic type deduction
obj = "f(x) = cos(x)";
// or force type
obj = As{"f(x) = cos(x)"};
```
- [Sequence](./include/zecalculator/math_objects/decl/sequence.h)
```c++
// needs forcing the type, automatic type deduction would otherwise deduce a zc::Function
obj = As{"u(n) = n"};
// or sequence with first values, the last expression is the generic expression
obj = "fibonacci(n) = 0 ; 1 ; fibonacci(n-1) + fibonacci(n-2)"
```
- [GlobalConstant](./include/zecalculator/math_objects/decl/global_constant.h)
```c++
// defined through an equation of type "name = number"
obj = "pi = 3.14";
// or assigned directly without the need of parsing
obj = zc::GlobalConstant{"pi", 3.14};
```
- Can be evaluated
```c++
tl::expected res1 = obj(1.0);
tl::expected res2 = obj.evaluate(12.0);
```
3. Error messages when expressions have faulty syntax or semantics are expressed through the [zc::Error](include/zecalculator/error.h) class:
- If it is known, gives what part of the equation raised the error with the `token` member, of the type [zc::tokens::Text](./include/zecalculator/parsing/data_structures/token.h)
- If it is known, gives the type of error.
4. Two namespaces are offered, that express the underlying representation of the parsed math objects
- `zc::fast::`: using the abstract syntax tree representation (AST)
- `zc::rpn::`: using reverse polish notation (RPN) / postfix notation in a flat representation in memory.
- Generated from the `fast` representation, but the time taken by the extra step is negligible (see results of the test "AST/FAST/RPN creation speed")
- Has faster evaluation

#### Benchmarks
There is for now one benchmark defined in the tests, called "parametric function benchmark" in the file [test/function_test.cpp](test/function_test.cpp), that computes the average evaluation time of the function `f(x) = 3*cos(t*x) + 2*sin(x/t) + 4`, where `t` is a global constant, in `ast` vs `rpn` vs `c++`.

The current results are (AMD Ryzen 5950X, `-march=native -O3` compile flags)
- `g++ 13.2.1` + `libstdc++` + `ld.bfd 2.41.0`
- `ast`: 270ns ± 5ns
- `rpn`: 135ns ± 5ns
- `c++`: 75ns ± 5ns
- `clang++ 17.0.6` + `libc++` + `ld.lld 17.0.6`
- `ast`: 245ns ± 5ns
- `rpn`: 140ns ± 5ns
- `c++`: 75ns ± 5ns

Benchmark code snippet

```c++
"parametric function benchmark"_test = []()
{
constexpr auto duration = nanoseconds(500ms);
{
constexpr parsing::Type type = std::is_same_v ? parsing::Type::FAST : parsing::Type::RPN;
constexpr std::string_view data_type_str_v = std::is_same_v ? "FAST" : "RPN";

MathWorld world;
auto& t = world.add("t = 1").template value_as();
auto& f = world.add("f(x) =3*cos(t*x) + 2*sin(x/t) + 4").template value_as>();

double x = 0;
double res = 0;
size_t iterations =
loop_call_for(duration, [&]{
res += f(x).value();
x++;
t += 1;
});
std::cout << "Avg zc::Function<" << data_type_str_v << "> eval time: "
<< duration_cast(duration / iterations).count() << "ns"
<< std::endl;
std::cout << "dummy val: " << res << std::endl;
}
{
double cpp_t = 1;
auto cpp_f = [&](double x) {
return 3*cos(cpp_t*x) + 2*sin(x/cpp_t) + 4;
};

double x = 0;
double res = 0;
size_t iterations =
loop_call_for(duration, [&]{
res += cpp_f(x);
iterations++;
x++;
cpp_t++;
});
std::cout << "Avg C++ function eval time: " << duration_cast(duration/iterations).count() << "ns" << std::endl;
std::cout << "dummy val: " << res << std::endl;

}

} | std::tuple{};
```

#### How to build

The project uses the [meson](mesonbuild.com/) build system. Being header-only,
it does not have a shared library to build: downstream projects only need the headers

#### How to run tests

To build tests
```shell
git clone https://github.com/AdelKS/ZeCalculator
cd ZeCalculator
meson setup build -D test=true
cd build
meson compile
```
Once the library is built, all tests can simply be run with
```
meson test
```
in the `build` folder.

#### How to install
Once the library is built (see above), you can install it by running
```
meson install
```
in the `build` folder.