Ecosyste.ms: Awesome

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

Awesome Lists | Featured Topics | Projects

https://github.com/nelarius/wrenpp

Minimal, zero dependency C++ binding generator for the Wren programming language
https://github.com/nelarius/wrenpp

bindings c-plus-plus cpp language-bindings scripting wren

Last synced: 2 months ago
JSON representation

Minimal, zero dependency C++ binding generator for the Wren programming language

Awesome Lists containing this project

README

        

# Wren++

[![Build Status](https://travis-ci.org/Nelarius/wrenpp.svg?branch=master)](https://travis-ci.org/Nelarius/wrenpp)

A C++ wrapper for the [Wren programming language](http://munificent.github.io/wren/), in the spirit of LuaBridge and Luabind. Both Wren and this library are still under development, so breaking changes will occasionally be introduced.

Wren++ currently provides:
- A RAII wrapper for the Wren virtual machine
- Automatic binding code generation for any C++ function and class
- Convenient access for calling Wren class methods from C++
- Uses C++14 template metaprogramming -- no macros!

Known issues:
- Not type-safe. It's undefined what happens when you try to bind code that returns a type which hasn't itself been bound (most likely a crash is going to happen)
- Wren access from C++ is rather minimal

## Table of contents
* [Build](#build)
* [At a glance](#at-a-glance)
* [Accessing Wren from Cpp](#accessing-wren-from-cpp)
* [Methods](#methods)
* [Accessing Cpp from Wren](#accessing-cpp-from-wren)
* [Foreign methods](#foreign-methods)
* [Foreign classes](#foreign-classes)
* [Properties](#properties)
* [Methods](#methods)
* [CFunctions](#cfunctions)
* [Cpp and Wren lifetimes](#cpp-and-wren-lifetimes)
* [Customize VM behavior](#customize-vm-behavior)
* [Customize printing](#customize-printing)
* [Customize error printing](#customize-error-printing)
* [Customize module loading](#customize-module-loading)
* [Customize heap allocation and garbage collection](#customize-heap-allocation-and-garbage-collection)

## Build

Clone the repository using `git clone https://github.com/nelarius/wrenpp.git`. The entire library consists just of `Wren++.h` and `Wren++.cpp`, and so the easiest way to use the library is just to include them in your project directly. Just remember to compile with C++14 features turned on!

Alternatively, you can use premake to generate a build file for the static library:

```sh
premake5 vs2015 --include=
```

If you want to build the tests as well, then you need to include the location of wren's `lib/` folder.

```sh
premake5 vs2015 --include= --link=
```

## At a glance

Let's fire up an instance of the Wren VM and execute some code:

```cpp
#include "Wren++.h"

int main() {
wrenpp::VM vm{};
vm.executeString( "System.print(\"Hello, world!\")" );

return 0;
}
```

The virtual machine is held internally in a pointer. `Wren` uniquely owns the virtual machine, which means that the `Wren` instance can't be copied, but can be moved when needed - just like a unique pointer.

Module names work the same way by default as the module names of the Wren command line module. You specify the location of the module without the `.wren` postfix.

```cpp
vm.executeModule( "script" ); // refers to script.wren
```

Both `executeString` and `executeModule` return a code indicating if the interpretation encountered any errors.

```cpp
wrenpp::Result res = vm.executeString(
"// calling nonexistent variable"
"foobar.call()"
);
if ( res == wrenpp::Result::CompileError ) {
std::cout << "foobar doesn't exist and a compilation error occurs.\n";
}
```

There are two other codes: `wrenpp::Result::RuntimeError` and `wrenpp::Result::Success`.

## Accessing Wren from Cpp

### Methods

You can use the `Wren` instance to get a callable handle for a method implemented in Wren. Given the following class, defined in `bar.wren`,

```dart
class Foo {
static say( text ) {
System.print( text )
}
}
```

you can call the static method `say` from C++ by using the variadic template method `operator()`,

```cpp
wrenpp::VM vm{};
vm.executeModule( "bar" );

wrenpp::Method say = vm.method( "main", "Foo", "say(_)" );
say( "Hello from C++!" );
```

`VM::method` has the following signature:

```cpp
Method VM::method(
const std::string& module,
const std::string& variable,
const std::string& signature
);
```

`module` will be `"main"`, if you're not in an imported module. `variable` should contain the variable name of the object that you want to call the method on. Note that you use the class name when the method is static. The signature of the method has to be specified, because Wren supports function overloading by arity (overloading by the number of arguments).

The return value of a Wren method can be accessed by doing

```cpp
wrenpp::Method returnsThree = vm.method("main", "returnsThree", "call()");
wrenpp::Value val = returnsThree();
double val = val.as();
```

The `operator()` method on `wrenpp::Method` returns a `wrenpp::Value` object, which can be cast to the wanted return type by calling `as()`.

Number, boolean, and string values are stored within `wrenpp::Value` itself. Note that do to this, you *don't* want to write

```cpp
const char* greeting = returnsGreeting().as();
```

because the string value is storesd in the `wrenpp::Value` instance itself. This would result in trying to dereference a string value which no longer exists. Store it locally before using the string value, like we did above:

```cpp
wrenpp::Value greeting = returnsGreeting();
printf("%s\n", greeting.as());
```

## Accessing Cpp from Wren

Wren++ allows you to bind C++ functions and methods to Wren classes. You provide the VM instance with the name of the foreign method and the corresponding C++ function pointer. These are then looked up by the VM when it encounters a foreign method in source code.

### Foreign methods

You can implement a Wren foreign method as a stateless free function in C++. Wren++ offers an easy to use wrapper over the functions. Here's how you could implement a simple math library in Wren by binding the C++ standard math library functions.

math.wren:
```dart
class Math {
foreign static cos( x )
foreign static sin( x )
foreign static tan( x )
foreign static exp( x )
}
```
main.cpp:
```cpp
#include "Wren++.h"
#include

int main() {

wrenpp::VM vm;
vm.beginModule( "math" )
.beginClass( "Math" )
.bindFunction< decltype(&cos), &cos >( true, "cos(_)" )
.bindFunction< decltype(&sin), &sin >( true, "sin(_)" )
.bindFunction< decltype(&tan), &tan >( true, "tan(_)" )
.bindFunction< decltype(&exp), &exp >( true, "exp(_)" )
.endClass()
.endModule();

vm.executeString( "import \"math\" for Math\nSystem.print( Math.cos(0.12345) )" );

return 0;
}
```

Both the type of the function (in the case of `cos` the type is `double(double)`, for instance, and could be used instead of `decltype(&cos)`) and the reference to the function have to be provided to `bindFunction` as template arguments. As arguments, `bindFunction` needs to be provided with a boolean which is true, when the foreign method is static, false otherwise. Finally, the method signature is passed.

### Foreign classes

Free functions don't get us very far if we want there to be some state on a per-object basis. Foreign classes can be registered by using `bindClass` on a module context. Let's look at an example. Say we have the following Wren class representing a 3-vector:

```dart
foreign class Vec3 {
construct new( x, y, z ) {}

foreign norm()
foreign dot( rhs )
foreign cross( rhs ) // returns the result as a new vector

// accessors
foreign x
foreign x=( rhs )
foreign y
foreign y=( rhs )
foreign z
foreign z=( rhs )
}
```

We would like to implement it using the following C++ struct.

```cpp
struct Vec3 {
union {
float v[3];
struct { float x, y, z; };
};

Vec3( float x, float y, float z )
: v{ x, y, z }
{}

float norm() const {
return sqrt( x*x + y*y + z*z );
}

float dot( const Vec3& rhs ) const {
return x*rhs.x + y*rhs.y + z*rhs.z;
}

Vec3 cross( const Vec3& rhs ) const {
return Vec3 {
y*rhs.z - z*rhs.y,
z*rhs.x - x*rhs.z,
x*rhs.y - y*rhs.x
};
}
};
```

Let's start by binding the class to Wren and adding properties.

A class is bound by writing `bindClass` instead of `beginClass`. This binds the specified constructor and destructor to Wren, and allocates a new instance of your class within Wren.

```cpp
#include "Wren++.h"

int main() {
wrenpp::VM vm;
vm.beginModule( "main" )
.bindClass< Vec3, float, float, float >( "Vec3" );
// you can now construct Vec3 in Wren

return 0;
}
```

Pass the class type, and constructor argument types to `bindClass`. Even though a C++ class may have many constructors, only one constructor can be registered with Wren.

#### Properties

If your class or struct has public fields you wish to expose, you can do so by using `bindGetter` and `bindSetter`. This will automatically generate a function which returns the value of the field to Wren.

```cpp
vm.beginModule( "main" )
.bindClass< Vec3, float, float, float >( "Vec3" )
.bindGetter< decltype(Vec3::x), &Vec3::x >( "x" )
.bindSetter< decltype(Vec3::x), &Vec3::x >( "x=(_)" )
.bindGetter< decltype(Vec3::y), &Vec3::y >( "y" )
.bindSetter< decltype(Vec3::y), &Vec3::y >( "y=(_)" )
.bindGetter< decltype(Vec3::z), &Vec3::z >( "z" )
.bindSetter< decltype(Vec3::z), &Vec3::z >( "z=(_)" );
```

Getters and setters are implicitly assumed to be non-static methods.

#### Methods

Using `registerMethod` allows you to bind a class method to a Wren foreign method. Just do:

```cpp
#include "Wren++.h"

int main() {
wrenpp::VM vm;
vm.beginModule( "main" )
.bindClass< Vec3, float, float, float >( "Vec3" )
// properties as before
.bindGetter< decltype(Vec3::x), &Vec3::x >( "x" )
.bindSetter< decltype(Vec3::x), &Vec3::x >( "x=(_)" )
.bindGetter< decltype(Vec3::y), &Vec3::y >( "y" )
.bindSetter< decltype(Vec3::y), &Vec3::y >( "y=(_)" )
.bindGetter< decltype(Vec3::z), &Vec3::z >( "z" )
.bindSetter< decltype(Vec3::z), &Vec3::z >( "z=(_)" )
// methods
.bindMethod< decltype(&Vec3::norm), &Vec3::norm >( false, "norm()" )
.bindMethod< decltype(&Vec3::dot), &Vec3::dot >( false, "dot(_)" )
.endClass()
.endModule();

return 0;
}
```

The arguments are the same as what you pass `bindFunction`, but as the template parameters pass the method type and pointer instead of a function.

We've now implemented two of `Vec3`'s three foreign functions -- what about the last foreign method, `cross(_)` ?

### CFunctions

Wren++ let's you bind functions of the type `WrenForeignMethodFn`, typedefed in `wren.h`, directly. They're called CFunctions for brevity (and because of Lua). Sometimes it's convenient to wrap a collection of C++ code manually. This happens when the C++ library interface doesn't match Wren classes that well. Let's take a look at binding the excellent [dear imgui](https://github.com/ocornut/imgui) library to Wren.

Many functions to ImGui are very overloaded and have lengthy signatures. We would have to fully qualify the function pointers, which would make the automatic bindings a mess. Additionally, many ImGui functions (such as SliderFloat) take in pointers to primitive types, like float, bool. Wren doesn't have any concept of `out` parameters, so we will make our Wren ImGui API take in a number by value, and return the new value.

```dart
class Imgui {

// windows
foreign static begin( name ) // begin a window scope
foreign static end() // end a window scope

foreign static sliderFloat( label, value, min, max ) // RETURNS the new value
}
```

First, let's implement wrappers for `ImGui::Begin` and `ImGui::SliderFloat` with reasonable default values.

```cpp
void begin(WrenVM* vm) {
ImGui::Begin((const char*)wrenGetSlotString(vm, 1), NULL, 0);
}

void sliderFloat(WrenVM* vm) {
const char* label = wrenGetSlotString(vm, 1);
float value = float(wrenGetSlotDouble(vm, 2));
float min = float(wrenGetSlotDouble(vm, 3));
float max = float(wrenGetSlotDouble(vm, 4));
ImGui::SliderFloat(label, &value, min, max);
wrenSetSlotDouble(vm, 0, value);
}
```

Here's what the binding code looks like. Note that ImGui::End is trivial to bind as it takes no arguments.

```cpp
vm.beginModule( "builtin/imgui" )
.beginClass( "Imgui" )
// windows & their formatting
.bindCFunction( true, "begin(_)", &VM::begin )
.bindFunction< decltype(&ImGui::End), &ImGui::End>( true, "end()" )
.bindCFunction( true, "sliderFloat(_,_,_,_)", &VM::sliderFloat)
.endClass();
```

If you need to access and return foreign object instances within your CFunction, you can use the following two helper functions.

Use `wrenpp::getSlotForeign(WrenVM*, int)` to get a bound type from the slot API:

```cpp
void setWindowSize(WrenVM* vm) {
ImGui::SetNextWindowSize(*(const ImVec2*)wrenpp::getSlotForeign(vm, 1));
}
```

Use `wrenpp::setSlotForeignValue(WrenVM*, int, const T&)` and `wrenpp::setSlotForeignPtr(WrenVM*, int, T* obj)` to place an object with foreign bytes in a slot, by value and by reference, respectively. `wrenpp::setSlotForeignValue` uses the type's copy constructor to copy the object into the new value.

### Cpp and Wren lifetimes

If the return type of a bound method or function is a reference or pointer to an object, then the returned wren object will have C++ lifetime, and Wren will not garbage collect the object pointed to. If an object is returned by value, then a new instance of the object is also constructed withing the returned Wren object. In this situation, the returned Wren object has Wren lifetime and is garbage collected.

## Customize VM behavior

The following customizations are affect all VMs.

### Customize printing

You can provide your own implementation for `System.print` by assigning a callable object with the signature `void(const char*)` to `wrenpp::VM::writeFn`. By default, `writeFn` is implemented simply as:

```cpp
WriteFn VM::writeFn = []( const char* text ) -> void { std::cout << text; };
```

### Customize error printing

You can provide your own function to route error messages. Assign a callable object with the signature `void(WrenErrorType, const char*, int, const char*)` (for the error type, module name, line number, and message, respectively) to `wrenpp::VM::errorFn`. By default, Wren++ styles the errors to `stdout` as

```
WREN_ERROR_COMPILE in main:15 > Error at 'Vec3': Variable is used but not defined.
```

### Customize module loading

When the virtual machine encounters an import statement, it executes a callback function which returns the module source for a given module name. If you want to change the way modules are named, or want some kind of custom file interface, you can change the callback function. Just set give `VM::loadModuleFn` a new value, which can be a free standing function, or callable object of type `char*( const char* )`.

By default, `VM::loadModuleFn` has the following value.

```cpp
VM::loadModuleFn = []( const char* mod ) -> char* {
std::string path( mod );
path += ".wren";
auto source = wrenpp::fileToString( path );
char* buffer = (char*) malloc( source.size() );
memcpy( buffer, source.c_str(), source.size() );
return buffer;
};
```

### Customize heap allocation and garbage collection

You can bind your own allocator to Wren by providing the following generic allocation function (which is set to `std::realloc` by default):

`wrenpp::VM::reallocateFn = std::realloc;`

`reallocateFn` needs to be a callable of type `void*(void* memory, std::size_t newSize)`. To allocate memory, `memory` is null and `newSize` is the desired size. To free memory, `memory` is the allocated pointer, and `newSize` is zero. To grow an existing allocation, `memory` is the already allocated memory, and `newSize` is the desired size. The function should return the same pointer if it was able to grow the allocation in place. The new pointer is returned if the allocation was moved. To shrink an allocation, `memory` is the already allocated pointer, and `newSize` is the desired size. The same pointer is returned.

The initial heap size is the number of bytes Wren will have allocated before triggering the first garbage collection. By default, it's 10 MiB.

`wrenpp::VM::initialHeapSize = 0xA00000u;`

After a collection occurs the heap will have shrunk. Wren will allow the heap to grow to (100 + heapGrowthPercent) % of the current heap size before the next collection occurs. By default, the heap growth percentage is 50 %.

`wrenpp::VM::heapGrowthPercent = 50;`

The minimum heap size is the heap size, in bytes, below which collections will not be carried out. The idea of the minimum heap size is to avoid miniscule heap growth (calculated based on the percentage mentioned previously) and thus very frequent collections. By default, the minimum heap size is 1 MiB.

`wrenpp::VM::minHeapSize = 0x100000u;`

## TODO:

* A compile-time method must be devised to assert that a type is registered with Wren. Use static assert, so incorrect code isn't even compiled!
* For instance, two separate `Type`s. One is used for registration, which iterates `Type` as well. This doesn't work in the case that the user registers different types for multiple `Wren` instances.
* There needs to be better error handling for not finding a method.
* Is Wren actually responsible for crashing the program when a method is not found?
* Does Wren actually crash the program when an invalid operator is used on a class instance?