Skip to content

Latest commit

 

History

History
125 lines (93 loc) · 3.48 KB

README.md

File metadata and controls

125 lines (93 loc) · 3.48 KB

Automatic C++ Language Bindings With Clang

Introduction

CPPBind is a libtooling based program that uses the Clang AST API to automatically generate language bindings to C++. In its approach it is similar to SWIG but has two distinct advantages:

  • No reliance on a custom C++ parser, increasing maintainability
  • Backends can be implemented in Python, allowing for rapid prototyping

Currently CPPBindg supports C, Rust and Lua as target languages.

Building

I recommend installing the necessary dependencies via conan. From the root of this repository, simply run:

conan install . --output-folder=build --build=missing

And then run cmake to prepare to build CPPBind:

cmake -B build -DCMAKE_BUILD_TYPE=Release -DCMAKE_C_COMPILER=clang -DCMAKE_CXX_COMPILER=clang++ -DCMAKE_TOOLCHAIN_FILE=conan_toolchain.cmake

Note that you must compile CPPBind with Clang.

Afterwards, simply build with make. The generated binary will be placed under build/source/cppbind_tool. You can also run make test to check that everything is okay (set CTEST_OUTPUT_ON_FAILURE=1 to display any errors). In order to run the tests you will additionally need the lua5.4 interpreter and rustc installed on your system.

Usage

Run cppbind_tool -h for a full list of options. Basic usage is as follows, using the Lua backend as an example:

We'll start from the following C++ header, named any_stack.h:

#pragma once

#include <any>
#include <stack>

// A stack which supports pushing and popping different types

class AnyStack
{
public:
  // push value
  template<typename T>
  void push(T val)
  { _stack.push(val); }

  // pop value, result is undefined if type does not match corresponding push
  template<typename T>
  T pop()
  {
    std::any ret(_stack.top());
    _stack.pop();
    return std::any_cast<T>(ret);
  }

  bool empty() const
  { return _stack.empty(); }

private:
  std::stack<std::any> _stack;
};

We can run cppbind_tool over this header, using the Lua backend as such:

cppbind_tool any_stack.h
             --template-instantiations any_stack.tcc
             --backend=lua
             --wrap-rule 'record:hasName("AnyStack")'
             --extra-arg '-std=c++17'
             --

The --wrap-rule option can be used one or multiple times to refine what should be wrapped using the syntax obj:rule where obj is one of enum, variable, function, or record and rule is a Clang AST matcher rule. Furthermore, any_stack.tcc is an extra file containing the explicit template instantiations of AnyStack for which wrapper code should be generated. This extra step is necessary because Lua has no conception of compile time templates:

template void AnyStack::push<int>(int);
template int AnyStack::pop<int>();
template void AnyStack::push<double>(double);
template double AnyStack::pop<double>();

This will generate a file any_stack_lua.cc in the current working directory (file location and name can be controlled via options). We can compile this file into a Lua module as follows:

clang++ -o any_stack.so -std=c++17 -shared -fPIC any_stack_lua.cc -I $CPPBIND_ROOT_DIR/generate -llua

Where CPPBIND_ROOT_DIR is set to the root directory of this repository. We can then use anys_stack.so from Lua:

local any_stack = require 'any_stack'

local s = any_stack.AnyStack.new()

s:push_int(1)
s:push_double(3.14)

print(s:pop_double())
print(s:pop_int())