Sane C++ Libraries
C++ Platform Abstraction Libraries
Plugin

🟨 Minimal dependency based plugin system with hot-reload

Plugin library allows extending application compiling C++ source to executable code at runtime.

Quick Sheet

namespace SC
{
//-----------------------------------------------------------------------------
// IPluginContract.h
//-----------------------------------------------------------------------------
// CLIENT - HOST Contract (Interface)
struct IPluginContract
{
static constexpr auto InterfaceHash = StringHashFNV("IPluginContract");
Function<void(void)> onDraw;
};
//-----------------------------------------------------------------------------
// PluginClient.cpp
//-----------------------------------------------------------------------------
// CLIENT Plugin (binds to contract functions)
struct PluginClient : public IPluginContract
{
PluginClient()
{
IPluginContract::onDraw = [this]()
{
// Draw stuff...
};
}
// Called when plugin is init
bool init() { return true; }
// Called when plugin is closed
bool close() { return true; }
};
SC_PLUGIN_DEFINE(PluginClient);
SC_PLUGIN_EXPORT_INTERFACES(PluginClient, IPluginContract);
//-----------------------------------------------------------------------------
// PluginHost.cpp
//-----------------------------------------------------------------------------
// Plugin HOST (loads plugin, obtains interface and calls functions)
struct PluginHost
{
// Fill directories before calling create
String executablePath; // Where executable lives
String libraryRootDirectory; // Where Sane C++ Libraries live
String someLibraryDirectory; // Where 3rd party-lib headers live
String pluginsPath; // Where Plugins live
PluginRegistry registry;
Result create(AsyncEventLoop& loop)
{
eventLoop = &loop;
// Setup Compiler
SC_TRY(PluginSysroot::findBestSysroot(compiler.type, sysroot));
// Add includes used by plugins...
SC_TRY(compiler.includePaths.push_back(libraryRootDirectory.view()));
SC_TRY(compiler.includePaths.push_back(someLibraryDirectory.view()));
// Setup File System Watcher
SC_TRY(fileSystemWatcher.init(fileSystemWatcherRunner, *eventLoop));
watcher.notifyCallback.bind<PluginHost, &PluginHost::onFileChanged>(*this);
SC_TRY(fileSystemWatcher.watch(watcher, pluginsPath.view()));
return Result(true);
}
Result close()
{
SC_TRY(fileSystemWatcher.close());
eventLoop = nullptr;
return Result(true);
}
Result syncRegistry()
{
Vector<PluginDefinition> definitions;
SC_TRY(PluginScanner::scanDirectory(pluginsPath.view(), definitions))
SC_TRY(registry.replaceDefinitions(move(definitions)));
return Result(true);
}
// Call this to load a plugin with a given identifier
Result load(StringView identifier)
{
// Force reload of plugin if already loaded
SC_TRY(registry.loadPlugin(identifier, compiler, sysroot,
executablePath.view(),
PluginRegistry::LoadMode::Reload));
// Obtain contract
SC_TRY(plugin->queryInterface(contract));
return Result(true);
}
void draw()
{
if (contract)
{
contract->draw()
}
}
private:
AsyncEventLoop* eventLoop;
PluginCompiler compiler;
PluginSysroot sysroot;
IPluginContract* contract = nullptr;
FileSystemWatcher fileSystemWatcher;
FileSystemWatcher::FolderWatcher watcher;
FileSystemWatcher::EventLoopRunner fileSystemWatcherRunner;
void onFileChanged(const FileSystemWatcher::Notification& notification)
{
auto reload = [this](const PluginIdentifier& plugin) { (void)load(plugin.view()); };
registry.getPluginsToReloadBecauseOf(notification.relativePath, Time::Milliseconds(500), reload);
}
};
}
constexpr T && move(T &value)
Converts an lvalue to an rvalue reference.
Definition: Compiler.h:269
#define SC_TRY(expression)
Checks the value of the given expression and if failed, returns this value to caller.
Definition: Result.h:48
constexpr unsigned int StringHashFNV(const char(&str)[N])
Compute compile time FNV hash for a char array.
Definition: StringHashFNV.h:33
static Result findBestCompiler(PluginCompiler &compiler)
Look for best compiler on current system.
static Result scanDirectory(const StringView directory, Vector< PluginDefinition > &definitions)
Scans a directory for PluginDefinition.
static Result findBestSysroot(PluginCompiler::Type compiler, PluginSysroot &sysroot)
Finds a reasonable sysroot for the given compiler.
Note
The above code is simplified. For a more complete implementation of an hot-reload system see SCExample code (HotReloadSystem.h).

Features

  • Compile and link cpp files to dynamic library
  • Unload / recompile / reload Plugin
  • Reload all dependant plugins of a given one
  • PluginDynamicLibrary::queryInterface allows creating contracts between Plugins or Plugins and Host
  • Support creating libc++ and libc free plugins that only use Sane C++ Libraries
  • Allows toolchain customization through SC::PluginSysroot and SC::PluginCompiler
  • Forcefully unlock and delete dll being debugged from Visual Studio Debugger

Status

🟨 MVP
This library is expected to work correctly on macOS, Windows, and Linux using MSVC, clang and GCC compiler toolchains.

Description

The main use case is for now splitting applications in smaller pieces that can be hot-reloaded at runtime.
A secondary use case could be allowing customization on a delivered application (mainly on Desktop systems). Plugins are always meant to be delivered in source code form (.cpp) and they're compiled on the fly. A plugin is made of a single .cpp file and it declares itself through a special comment in the source code. Such comment can declare the name, the version, a description / category and a list of dependencies.

A plugin can be modified, unloaded, re-compiled and re-loaded to provide additional functionality.

The list of dependencies makes it possible to find recursive dependencies and unload them before unload a plugin.

The library doesn't use a build system, but it compiles the .cpp files directly, linking it with symbols exported from the loading executable (using bundle_loader on macOS and linking library exported from loading executable on windows). Plugin Dynamic Libraries are compiled with nostdlib and nostdlib++ and they include a stub the allows defining some symbols needed due to not linking the C++ CRT.
Some special build flags however allow using libc, libc++ or other sysroot / compiler supplied windows.
The idea is that plugins only use functionality provided by the calling executable or by other plugins.

On Windows, some extra care has been taken to force-unlock the .pdb file from visual studio debugger, that happens if the dll is being loaded on a program being debugged.

As of today this is all implemented using native dynamic library mechanisms that are being loaded directly in the process.
Doing the wrong thing with memory or forgetting to clean everything during shutdown can quickly crash the main executable.

Videos

This is the list of videos that have been recorded showing some usages of the library:

Blog

Some relevant blog posts are:

Examples

  • SCExample uses Plugin library for a simple hot-reload system
  • Unit test inside PluginTest.cpp show how the API is meant to be used

Roadmap

There are plans to experiment with out of process plugins using some sort of RPC system (like Audio Unit plugins on macOS) and / or experimenting using WASM as a plugin host to eliminate such instability / security issues. Other ideas include redistribute a minimal C++ toolchain (probably a customized clang) that can compile the plugins without needing a system compiler or a sysroot, as all public headers of libraries in this project do not need any system or compiler header.

🟩 Usable Features:

  • Specify directory for compiled intermediate and output files
  • Parallel / Async compile and link
  • Improve error handling and reporting
  • Further customization of some build flags and features:
    • Custom libraries to link (declared in the plugin)
    • Custom include paths (declared in the plugin)

🟦 Complete Features:

  • Create minimal clang toolchain to compile scripts on non-developer machines
  • Integrate with Build library (once it will gain capability to build standalone without needing Xcode or Visual Studio)
  • Evaluate possibility to achieve some minimal error recovery
  • Easily integration with some RPC mechanism

💡 Unplanned Features:

  • Compile plugins to WASM ?
  • Deploy closed-source (already compiled) binary plugins
  • Allow plugin to be compiled with different compiler from the one used in the Host