 
  M E T A C A L L
A library for providing inter-language foreign function interface calls
METACALL is a library that allows calling functions, methods or procedures between programming languages. With METACALL you can transparently execute code from / to any programming language, for example, call Python code from JavaScript code.
sum.py
def sum(a, b):
  return a + bmain.js
metacall_load_from_file('py', [ 'sum.py' ]);
metacall('sum', 3, 4); // 7- Abstract
- Table Of Contents
- 1. Motivation
- 2. Language Support (Backends)
- 3. Use Cases
- 4. Usage
- 4.1 Installation
- 5. Architecture
- 5. Application Programming Interface (API)
- 6. Build System
- 7. Platform Support
- 8. License
 
The METACALL project started time ago when I was coding a Game Engine for an MMORPG. My idea was to provide an interface to allow other programmers extend the Game Engine easily. By that time, I was finishing the university so I decide to do my Final Thesis and Presentation based on the plug-in system for my Game Engine. The Plugin Architecture designed for the Game Engine has similarities with METACALL although the architecture has been redefined and the code has been rewritten from scratch. After some refination of the system, I came up with METACALL and other use cases for the tool. Currently we are using METACALL to build a cutting edge FaaS (Function as a Service) https://metacall.io based on this technique to provide high scalability of the functions among multiple cores and Function Mesh pattern, a new technique I have developed to interconnect transparently functions in a distributed system based on this library.
This section describes all programming languages that METACALL supports, if you are interested in from what languages can be used METACALL you must go to ports section.
- Currently supported languages and run-times:
| Language | Runtime | Version | Tag | 
|---|---|---|---|
| Python | Python C API | >= 3.2 <= 3.6 | py | 
| NodeJS | N API | >= 8.11.1 <= 8.12.0 | node | 
| JavaScript | V8 | 5.1.117 | js | 
| C# | NetCore | 1.1.10 | cs | 
| Ruby | Ruby C API | >= 2.1 <= 2.3 | rb | 
| Mock | ∅ | 0.1.0 | mock | 
- Languages and run-times under construction:
| Language | Runtime | Tag | 
|---|---|---|
| Java | JNI | java | 
| C/C++ | Clang - LLVM - libffi | c | 
| File | ∅ | file | 
| Go | Go Runtime | go | 
| Haskell | Haskell FFI | hs | 
| JavaScript | SpiderMonkey | jsm | 
METACALL can be used in the following cases:
- 
Interconnect different technologies in the same project. It allows to have heterogeneous teams of developers working over same project in an isolated way and using different programming languages at the same time. 
- 
Embedding programming languages to existing softwares. Game Engines, 3D Editors like Blender, among others can take benefit of METACALL and extend the core functionality with higher level programming languages (aka scripting). 
- 
Function as a Service. METACALL can be used to implement efficient FaaS architectures. We are using it to implement our own FaaS (Function as a Service) https://metacall.io based on Function Mesh pattern and high performance function scalability thanks to this library. 
- 
Source code migrations. METACALL can wrap large and legacy code-bases, and provide an agnostic way to work with the codebase into a new programming language. Eventually the code can be migrated by parts, without need of creating a new project or stopping the production environment. Incremental changes can be done, solving the migration easily and with less time and effort. 
- 
Porting low level libraries to high level languages transparently. With METACALL you can get rid of extension APIs like Python C API or NodeJS N-API. You can call directly low level libraries from your high level languages without making a wrapper in C or C++ for it. 
As you can see, there are plenty of uses. METACALL introduces a new model of programming which allows a high interoperability between technologies. If you find any other use case just let us know about it with a Pull Request and we will add it to the list.
Prior to try any example, you must have METACALL installed in your system. To install METACALL you have the following options.
- Download a release.
- Install via package manager.
- Build and install it manually.
- Pull it from DockerHub.
This environment variables are optional, in case that you want to modify default paths of METACALL.
| Name | Description | Default Value | 
|---|---|---|
| DETOUR_LIBRARY_PATH | Directory where detour plugins to be loaded are located | detours | 
| SERIAL_LIBRARY_PATH | Directory where serial plugins to be loaded are located | serials | 
| CONFIGURATION_PATH | File path where the METACALL global configuration is located | configurations/global.json | 
| LOADER_LIBRARY_PATH | Directory where loader plugins to be loaded are located | loaders | 
| LOADER_SCRIPT_PATH | Directory where scripts to be loaded are located | ${execution_path}¹ | 
¹ ${execution_path} defines the path where the program is executed, . in Linux.
- 
MetaCall CLI. Example of a Command Language Interpreter based on MetaCall where you can load, unload scripts and call their functions. 
- 
MetaCall Rotulin. Example of a multi-language application built with METACALL. This application embeds a Django server with a Ruby DataBase and C# business layer based on ImageMagick. 
- 
To provide an high level API with a simple UX and to be easy to understand. 
- 
To work in high performance environments. 
- 
To be as cross-platform as possible. 
- 
To avoid to modify run-times directly or use the code inside METACALL in order to avoid maintaining them, or propagating security flaws or licenses into METACALL. 
- 
To provide support for any embeddable programming language and to provide support for METACALL to be used form any programming language. 
- 
All external code used into METACALL must be introduced by inversion of control in the plugin system, so that the core must not remain aware from what software is using. 
- 
All code developed in METACALL must be implemented in standalone libraries that can work by itself in an isolated way (aka modules). 
- 
adtprovides a base for Abstract Data Types and algorithms used in METACALL. Implementation must be done in an efficient and generic way. Some of the data structures implemented are vector, set, hash, comparable or trie.
- 
detourprovides an interface to hook into functions. Detours are used by the fork model to intercept fork calls.
- 
detoursimplement thedetourinterface by using a plugin architecture. The current list of available detour plugins is the following one.- funchook_detourimplemented by means of FuncHook library.
 
- 
distributabledefines the compilation of METACALL that generates an unique library with all core libraries bundled into it. As the METACALL architecture is divided by modules, in order to distribute METACALL is needed to build all of them into a single library. This module implements this compilation by means of CMake.
- 
dynlinkimplements a cross-platform method to dynamically load libraries. It is used to dynamically load plugins into METACALL.
- 
environmentimplements an standard way to deal with environment variables. METACALL uses environment variables to define custom paths for plugins and scripts.
- 
examples...
- 
filesystemprovides an abstraction for operative system file system.
- 
formatprovides an standard way for printing to standard input output for old C versions that does not support newest constructions.
- 
loader...
The module that holds the representation of types, values and functions is called reflect and it handles the abstraction of code loaded into METACALL.
METACALL uses reflection and introspection techniques to inspect the code loaded by the loaders in order to interpret it and provide an higher abstraction of it. With this higher abstraction METACALL can easily inter-operate between languages transparently.
METACALL implements an abstract type system which is a binary representation of the types supported by it. This means that METACALL can convert any type of a language to its own type system and back. Each loader is responsible of doing this conversions.
METACALL maintains most of the types of the languages but not all are supported. If new types are added they have to be implemented in the reflect module and also in the loaders and serials to fully support it.
| Type | Value | 
|---|---|
| Boolean | trueorfalse | 
| Char | -128to127 | 
| Short | -32,768to32,767 | 
| Int | -2,147,483,648to2,147,483,647 | 
| Long | –9,223,372,036,854,775,808to9,223,372,036,854,775,807 | 
| Float | 1.2E-38to3.4E+38 | 
| Double | 2.3E-308to1.7E+308 | 
| String | NULL terminated list of characters | 
| Buffer | Blob of memory representing a binary data | 
| Array | Arrangement of values of any type | 
| Map | List of elements formed by a key (String) value (Any) pair (Array) | 
| Pointer | Low level representation of a memory reference | 
| Null | Representation of NULL value type | 
- 
Boolean is mostly represented by an integer value. There are languages that does not support it so it gets converted to a integer value in the memory layout. 
- 
Integer and Floating Point values provide a complete abstraction to numerical types. Type sizes are preserved and the correct type is used when using any number. This depends on the internal implementation of the value by the run-time. Although there can be problems related to this. A bignumtype from Ruby may overflow if it is too big when trying to convert it to afloattype in C#.
- 
String is represented by ASCII encoding currently. Future versions will implement multiple encodings to be interoperable between other language encodings. 
- 
Buffer represents a blob of raw memory (i.e. an array of bytes). This can be used to represent files as images or any other resources into memory. 
- 
Array is implemented by means of array of values, which you can think it should be called list instead. But as the memory layout is stored into a contiguous memory block of references to values, it is considered an array. 
- 
Map implements an associative key value pair container. A map is implemented with an array of two sized elements array. Each element of the map is an array of size two, where the first element of it is always an String and the second element is a value of any type. 
- 
Pointer is an opaque value representing a raw reference to a memory block. Some languages allow to use references to memory and some others not. This type is opaque because METACALL does not know what kind of concrete value represents it. The representation may be a complex type handled by the developer source code inside the run-time. 
- 
Null type implements a null value. This type has only been implemented in order to support null value from multiple run-times. It represents a null value and it does not have data size on the value allocated. 
Values represent the instances of the METACALL type system.
The memory layout guarantees to fit at least the same size of the types into memory. This means if a boolean type can be represented with one bit inside a value of one byte size, maybe this value is stored in a bigger memory block and this fact is architecture and platform dependant.
When converting values between different types, if any potential number overflow or invalid conversion between types is done, METACALL will warn about it. If any conversion of types can be handled by METACALL, it will automatically cast or transform the values into the target type automatically in order to avoid errors in the call.
The value model is implemented by means of object pool. Each value is a reference to a memory block allocated from a memory pool (which can be injected into METACALL). The references can be passed by value, this means METACALL copies the reference value instead of the data which this reference is pointing to, like most run-times do when managing their own values.
Each created value must be destroyed manually. Otherwise it will lead to a memory leak. This fact only occurs when dealing with METACALL at C level. If METACALL is being used in an higher language through ports, the developer does not have to care about memory management.
The value memory layout is described in the following form.
| Memory Offset | 0tosizeof(data) - 1 | sizeof(data)tosizeof(data) + sizeof(type_id) - 1 | 
|---|---|---|
| Content | DATA | TYPE ID | 
This layout is used by the following reasons.
- 
Data is located at the first position of the memory block, so it can be used as a normal low level value. This allows to threat METACALL values as a normal C values. Therefore you can use METACALL with normal pointers to existing variables, literal values as shown in the previous examples or METACALL values. 
- 
Data can be accessed faster as it is located at first position of the memory block. There is not extra calculation of an offset when trying to access the pointer. 
- 
Data and type id are contiguously allocated in order to threat it as the same memory block so it can be freed with one operation. 
Functions are an abstract callable representation of functions, methods or procedures loaded by loaders. The functions are like a template who is linked to a loader run-time and allows to do a foreign function call.
A function is composed by a name and a signature. The signature defines the arguments name, type, and return type if any. When a function is loaded, METACALL tries to inspect the signature and records the types if any. It stores the arguments name and size and also a concrete type that will be used later by the loader to implement the call to the run-time.
The function interface must be implemented by the loaders and it has the following form.
typedef struct function_interface_type
{
  function_impl_interface_create create;
  function_impl_interface_invoke invoke;
  function_impl_interface_destroy destroy;
} * function_interface;- createinstantiates the function concrete data related to the run-time.
- invoketransforms arguments from- reflectabstract types to run-time concrete types, executes the call in the run-time, and converts the result of the call from run-time concrete type to- reflectabstract type.
- destroyclears all data previously instantiated in- create.
The type deduction can be done at different levels. For example, it is possible to guess function types from the loaded code.
def multiply_type(a: int, b: int) -> int:
  return a * bIf this code is loaded, METACALL will be able to inspect the types and define the signature. Signature includes the names of the arguments, the types of those arguments if any, and the return type if any.
It may be possible that the function loaded into METACALL is duck typed. This means it does not have information about what types it supports and therefore they cannot be inspected statically.
def multiply_duck(a, b):
  return a * bAt low level METACALL must always know the types to do the call. This types can be inferred statically or dynamically and this has implications over the call model.
In the first example, we can simply call the function without specifying the types.
metacall("multiply_type", 3, 4); // 12As the signature is already know the literal values 3 and 4 can be converted into METACALL values automatically. Note that in this case, as literal values are provided, if we pass a double floating point, the memory representation of the value will be corrupted as there is no possible way to detect input values and cast them to the correct target values.
In the second example, the values are not know. If we use the same API to call the function, METACALL will not be able to call correctly the function as its types are not know. To allow calls to duck typed functions the developer must specify the value types he is passing to the function.
const enum metacall_value_id multiply_types[] =
{
  METACALL_INT, METACALL_INT
};
metacallt("multiply_duck", multiply_types, 3, 4); // 12This method allows to pass different value types to the same function. The following call would be valid too.
const enum metacall_value_id multiply_types[] =
{
  METACALL_DOUBLE, METACALL_DOUBLE
};
metacallt("multiply_duck", multiply_types, 3.0, 4.0); // 12.0METACALL has a plugin architecture implemented at multiple levels.
- 
Loaders implement a layer of plugins related to the run-times. 
- 
Serials implement a layer of (de)serializers in order to transform input (arguments) or output (return value) of the calls into a generic format. 
- 
Detours is another layer of plugins focused on low level function interception (hooks). 
Each plugin is a piece of software that can be dynamically loaded into the METACALL core, used and unloaded when it is not needed anymore.
Loaders are responsible for embedding run-times into METACALL. Each loader has the following interface.
typedef struct loader_impl_interface_type
{
  loader_impl_interface_initialize initialize;
  loader_impl_interface_execution_path execution_path;
  loader_impl_interface_load_from_file load_from_file;
  loader_impl_interface_load_from_memory load_from_memory;
  loader_impl_interface_load_from_package load_from_package;
  loader_impl_interface_clear clear;
  loader_impl_interface_discover discover;
  loader_impl_interface_destroy destroy;
} * loader_impl_interface;A loader must implement it to be considered a valid loader.
- initializestarts up the run-time.
- execution_pathdefines a new import path to the run-time.
- load_from_fileloads a code from file into the run-time and returns a handle which represents it.
- load_from_memoryloads a code from memory into the run-time and returns a handle which represents it.
- load_from_packageloads a code from a compiled library or package into the run-time and returns a handle which represents it.
- clearunloads a handle from the run-time.
- discoverinspects a handle previously loaded.
- destroyshutdowns the run-time.
METACALL implements a fork safe model. This means if METACALL is running in any program instance, thr process where is running can be forked safely at any moment of the execution. This fact has many implications at design, implementation and use level. But the whole METACALL architecture tries to remove all responsibility from the developer and make this transparent.
To understand the METACALL fork model, first of all we have to understand the implications of the forking model in operative systems and the difference between fork-one and fork-all models.
The main difference between fork-one and fork-all is that in fork-one only the thread which called the fork is preserved after the fork (i.e. gets cloned). In fork-all model, all threads are preserved after cloning. POSIX uses fork-one model, meanwhile Oracle Solaris use the fork-all model.
Because of fork-one model, forking a running run-time like NodeJS (which has a thread pool) implies that in the child process the thread pool will be almost dead except the thread which did the fork call. So NodeJS run-time cannot continue the execution anymore and the event-loop enters into a deadlock state.
When a fork is done, the status of the execution is lost by the moment. METACALL is not able to preserve the state when a fork is done. Some run-times do not allow to preserve the internal state. For example, the bad design of NodeJS does not allow to manage the thread pool from outside, so it cannot be preserved after a fork.
Because of these restrictions, METACALL cannot preserve the status of the run-times. In the future this model will be improved to maintain consistency and preserve the execution state of the run-times making METACALL more robust.
Although the state is not preserved, fork safety is. The mechanism METACALL uses to allow fork safety is described in the following enumeration.
- 
Intercept fork call done by the program where METACALL is running. 
- 
Shutdown all run-times by means of unloading all loaders. 
- 
Execute the real fork function. 
- 
Restore all run-times by means of reloading all loaders. 
- 
Execute user defined fork callback if any. 
To achieve this, METACALL hooks fork primitives depending on the platform.
- forkon POSIX systems.
- RtlCloneUserProcesson Windows systems.
If you use clone instead of fork to spawn a new process in a POSIX system, METACALL won't catch it.
Whenever you call a to a cloning primitive METACALL intercepts it by means of detour. Detours is a way to intercept functions at low level by editing the memory and introducing a jump over your own function preserving the address of the old one. METACALL uses this method instead of POSIX pthread_atfork for three main reasons.
- 
The first one is that pthread_atforkis only supported by POSIX systems. So it is not a good solution because of the philosophy of METACALL is to be as cross-platform as possible.
- 
The second is that pthread_atforkhas a bug in the design of the standard. It was designed to solve a problem which cannot be solved withpthread_atforkitself. This means that even having the control of NodeJS thread pool, it will not be possible to restore the mutexes in the child process. The only possibility is to re-implement the thread pool of NodeJS with async safe primitives like a semaphore. Async safe primitives will be able to work in the child process handler. But this is not possible as it enters in conflict with the design decision of to not modify the run-times.
- 
The third one is that the mechanism of pthread_atforkalso will be deprecated because of second reason.The pthread_atfork()function may be formally deprecated (for example, by shading it OB) in a future version of this standard.
Detours model is not safe. It is platform dependant and implies that the program modifies the memory of itself during the execution which is not safe at all and can induce bugs or security flaws if it is not done correctly. But because of limitations of run-times, there is not another alternative to solve the problem of fork safety.
Usually the developer is the same who does the fork, but it may be possible that METACALL is embedded into a larger application and the developer is in the middle between the application code and METACALL so it is impossible to control when a fork is done. Because of this the developer can register a callback by means of metacall_fork to know when a fork is executed to do the actions needed after the fork, for example, re-loading all previous code and restore the state of the run-times. This gives a partial solution to the problem of losing the state when doing a fork.
Follow these steps to build and install METACALL manually.
git clone --recursive https://github.com/metacall/core.git
mkdir core/build && cd core/build
cmake ..
make
make test
sudo make installThese options can be set using -D prefix when configuring CMake. For example, the following configuration enables the build of Python and Ruby loaders.
cmake -DOPTION_BUILD_LOADERS_PY=On -DOPTION_BUILD_LOADERS_RB=On ..Available build options are the following ones.
| Build Option | Description | Default Value | 
|---|---|---|
| BUILD_SHARED_LIBS | Build shared instead of static libraries. | ON | 
| OPTION_BUILD_DIST_LIBS | Build all libraries into a single compilation unit. | ON | 
| OPTION_SELF_CONTAINED | Create a self-contained install with all dependencies. | OFF | 
| OPTION_BUILD_TESTS | Build tests. | ON | 
| OPTION_BUILD_BENCHMARKS | Build benchmarks. | OFF | 
| OPTION_BUILD_DOCS | Build documentation. | OFF | 
| OPTION_BUILD_EXAMPLES | Build examples. | ON | 
| OPTION_BUILD_LOADERS | Build loaders. | ON | 
| OPTION_BUILD_SCRIPTS | Build scripts. | ON | 
| OPTION_BUILD_SERIALS | Build serials. | ON | 
| OPTION_BUILD_DETOURS | Build detours. | ON | 
| OPTION_BUILD_PORTS | Build ports. | OFF | 
| OPTION_FORK_SAFE | Enable fork safety. | OFF | 
| OPTION_THREAD_SAFE | Enable thread safety. | OFF | 
| CMAKE_BUILD_TYPE | Define the type of build. | Release | 
It is possible to enable or disable concrete loaders, script, ports, serials or detours. For building use the following options.
| Build Option Prefix | Build Option Suffix | 
|---|---|
| OPTION_BUILD_LOADERS_ | CJSCSMOCKPYJSMNODERBJSMFILE | 
| OPTION_BUILD_SCRIPTS_ | CCSJSNODEPYRBJAVA | 
| OPTION_BUILD_SERIALS_ | METACALLRAPID_JSON | 
| OPTION_BUILD_DETOURS_ | FUNCHOOK | 
| OPTION_BUILD_PORTS_ | CSCXXDGOJAVAJSLUANODEPHPPLPYRRB | 
The following platforms and architectures have been tested an work correctly with all plugins of METACALL.
| Operative System | Architecture | Compiler | Build Status | 
|---|---|---|---|
| ubuntu:xenial | amd64 | gcc | |
| debian:stretch-slim | amd64 | gcc:6.3.0 | |
| debian:buster-slim | amd64 | gcc:8.2.0 | |
| windows | x86x64 | msvc | 
To provide a reproducible environment METACALL is also distributed under Docker on DockerHub. Current images are based on debian:stretch-slim for amd64 architecture.
For pulling the METACALL latest image containing the runtime, use:
docker pull metacall/coreFor pulling a specific image depending on the tag, use:
- METACALL depsimage. Includes all dependencies for development:
docker pull metacall/core:deps- METACALL devimage. Includes all dependencies, headers and libraries for development:
docker pull metacall/core:dev- METACALL runtimeimage. Includes all dependencies and libraries for runtime (equivalent tolatest):
docker pull metacall/core:runtimeMETACALL is licensed under Apache License Version 2.0.
Copyright (C) 2016 - 2019 Vicente Eduardo Ferrer Garcia <[email protected]>
Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.