A C++17 library for parsing and executing expressions. Allows easy exposure of variables and functions from C++ through built-in reflection functionality. JitCat is currently being expanded into a more fully featured scripting language.
- Supports floating point, integer, boolean, string and object (pointer or by value) typed expressions as well as void typed and any typed expressions. More types to come.
- Optional native code compilation of expressions using the LLVM core libraries.
- Expose variables for use in expressions through reflection of C++ classes.
- Create and expose custom data structures to expressions at runtime.
- Reflection supports easy reflection of C++ classes and structs including reflection of functions. Limited to the following member types (with more types to come):
- float, double, bool, int, std::string
- Reflected objects
- Reflected enums
- Pointers to reflected objects
- std::unique_ptr to reflected objects
- std::library container types that have implemented operator[]
- member functions that return one of the supported types or void and that have parameters that are among the supported types (or no parameters).
- static members and static functions
- constants
- Reflection can be done either internally to a class, using static member functions, or externally using an ExternalReflector.
- Built-in functions for use in expressions for common operations
- Basic optimizations of expressions such as const collapse.
- Graceful error handling with human-readable error messages, though this area can use improvement.
- Builds and is tested to work on Linux with clang (6.0) and gcc (8.0) and on Microsoft Windows with VS2019.
- Built-in functionality for code completion of expressions.
- Unit tests testing all aspects of the expression language and reflection.
- (Currently broken) Can export reflected types to a XML file for use with the JitCatValidator shared library to validate and code-complete expressions outside of the main C++ application.
#include <jitcat/Expression.h>
#include <jitcat/ExpressionAny.h>
using namespace jitcat;
//A simple floating point calculation
Expression<float> anExpression("2.0 * abs(21.0)");
anExpression.compile(nullptr);
float value = anExpression.getValue(nullptr);
//String addition
Expression<std::string> anotherExpression("\"Hello\" + \" World\"");
anotherExpression.compile(nullptr);
std::string message = anotherExpression.getValue(nullptr);
//An expression returning an object. It uses a context that contains variables that can be referenced inside the expression.
Expression<MyObject*> objectTypeExpression("anObject.member.list[42].getMyObject()");
objectTypeExpression.compile(myContext);
MyObject* objectResult = objectTypeExpression.getValue(myContext);
//An expression that accepts any return type, in this case it is a std::vector.
//Passing myContext to the constructor will call compile automatically.
ExpressionAny anyTypedExpression(myContext, "anObject.member.list");
std::any anyValue = anyTypedExpression.getValue(myContext);
if (anyTypedExpression.getType().isFloatType())
{
float floatValue = std::any_cast<float>(anyValue);
//Do something
}
Building JitCat
Building LLVM for use with JitCat
See the the examples directory for a basic example.
JitCat is under active development. These are some of the major features that can be expected in the future:
- Documentation and more example projects.
- Support for more types.
- Extending the language beyond simple expressions.
Uses the permissive MIT license, see the included LICENSE file.
Machiel van Hooren
Thanks to Ronimo Games, my ex-employer, for supporting and actively using this library. Many improvements have been made during office hours!