Sane C++ Libraries
C++ Platform Abstraction Libraries
Plugin.h
1// Copyright (c) Stefano Cristiano
2// SPDX-License-Identifier: MIT
3#pragma once
4#include "../Containers/Vector.h"
5#include "../Containers/VectorMap.h"
6#include "../Foundation/Function.h"
7#include "../Strings/String.h"
8#include "../Time/Time.h"
9#include "Internal/DynamicLibrary.h"
10
11namespace SC
12{
13struct PluginDefinition;
14struct PluginScanner;
15struct PluginFile;
16struct PluginIdentity;
17struct PluginDynamicLibrary;
18struct PluginCompiler;
19struct PluginCompilerEnvironment;
20struct PluginSysroot;
21struct PluginRegistry;
22using PluginIdentifier = SmallString<30>;
23} // namespace SC
24
29
32
35{
37};
38
41{
45
49 bool operator==(const PluginIdentity& other) const { return identifier == other.identifier; }
50};
51
54{
59
63
66 PluginFile& getMainPluginFile() { return files[pluginFileIndex]; }
67
70 const PluginFile& getMainPluginFile() const { return files[pluginFileIndex]; }
71
76 [[nodiscard]] static bool find(const StringView text, StringView& extracted);
77
82 [[nodiscard]] static bool parse(StringView text, PluginDefinition& pluginDefinition);
83
87 [[nodiscard]] Result getDynamicLibraryAbsolutePath(String& fullDynamicPath) const;
88
92 [[nodiscard]] Result getDynamicLibraryPDBAbsolutePath(String& fullDynamicPath) const;
93
94 private:
95 [[nodiscard]] static bool parseLine(StringIteratorASCII& iterator, StringView& key, StringView& value);
96
97 size_t pluginFileIndex = 0;
98 friend struct PluginScanner;
99};
100
103{
108 [[nodiscard]] static Result scanDirectory(const StringView directory, Vector<PluginDefinition>& definitions);
109};
110
113{
120 Result compile(const PluginDefinition& definition, const PluginSysroot& sysroot,
121 const PluginCompilerEnvironment& environment, String& compilerLog) const;
122
130 Result link(const PluginDefinition& definition, const PluginSysroot& sysroot,
131 const PluginCompilerEnvironment& environment, StringView executablePath, String& linkerLog) const;
132
134 enum class Type
135 {
136 ClangCompiler,
137 GnuCompiler,
138 MicrosoftCompiler
139 };
140 Type type = Type::ClangCompiler;
143
145
148
152 [[nodiscard]] static Result findBestCompiler(PluginCompiler& compiler);
153
154 private:
155 [[nodiscard]] Result compileFile(const PluginDefinition& definition, const PluginSysroot& sysroot,
156 const PluginCompilerEnvironment& compilerEnvironment, StringView sourceFile,
157 StringView objectFile, String& compilerLog) const;
158 struct Internal;
159};
160
163{
166
168
173 [[nodiscard]] static Result findBestSysroot(PluginCompiler::Type compiler, PluginSysroot& sysroot);
174};
175
178{
179 StringView cFlags;
180 StringView ldFlags;
181
182 private:
183 struct Internal;
184 friend struct PluginCompiler;
185};
188{
190 SystemDynamicLibrary dynamicLibrary;
194
198 template <typename T>
199 [[nodiscard]] bool queryInterface(T*& outInterface) const
200 {
201 if (pluginQueryInterface and instance != nullptr)
202 {
203 return pluginQueryInterface(instance, T::InterfaceHash, reinterpret_cast<void**>(&outInterface));
204 }
205 return false;
206 }
207
208 PluginDynamicLibrary() : lastLoadTime(Time::Realtime::now()) { numReloads = 0; }
209
210 private:
211 void* instance = nullptr;
212 bool (*pluginInit)(void*& instance) = nullptr;
213 bool (*pluginClose)(void* instance) = nullptr;
214
215 bool (*pluginQueryInterface)(void* instance, uint32_t hash, void** instanceInterface) = nullptr;
216
217 friend struct PluginRegistry;
218 [[nodiscard]] Result load(const PluginCompiler& compiler, const PluginSysroot& sysroot, StringView executablePath);
219 [[nodiscard]] Result unload();
220};
221
224{
227
232
234 enum class LoadMode
235 {
236 Load = 0,
237 Reload = 1,
238 };
239
247 [[nodiscard]] Result loadPlugin(const StringView identifier, const PluginCompiler& compiler,
248 const PluginSysroot& sysroot, StringView executablePath,
249 LoadMode loadMode = LoadMode::Load);
250
254 [[nodiscard]] Result unloadPlugin(const StringView identifier);
255
259 [[nodiscard]] Result removeAllBuildProducts(const StringView identifier);
260
264 [[nodiscard]] const PluginDynamicLibrary* findPlugin(const StringView identifier);
265
267 [[nodiscard]] size_t getNumberOfEntries() const { return libraries.size(); }
268
270 [[nodiscard]] const PluginIdentifier& getIdentifierAt(size_t index) const { return libraries.items[index].key; }
271
273 [[nodiscard]] const PluginDynamicLibrary& getPluginDynamicLibraryAt(size_t index)
274 {
275 return libraries.items[index].value;
276 }
277
283 Function<void(const PluginIdentifier&)> onPlugin);
284
285 private:
287};
288
unsigned int uint32_t
Platform independent (4) bytes unsigned int.
Definition: PrimitiveTypes.h:38
@ Native
Encoding is UTF8.
Wraps function pointers, member functions and lambdas without ever allocating.
Definition: Function.h:19
Reads and holds CFLAGS and LDFLAGS environment variables, mainly to pass down sysroot location.
Definition: Plugin.h:178
Compiles a plugin to a dynamic library.
Definition: Plugin.h:113
SmallVector< StringNative< 256 >, 8 > includePaths
Path to include directories used to compile plugin.
Definition: Plugin.h:144
SmallVector< StringNative< 256 >, 8 > compilerIncludePaths
Path to compiler include directories.
Definition: Plugin.h:146
SmallVector< StringNative< 256 >, 8 > compilerLibraryPaths
Path to compiler library directories.
Definition: Plugin.h:147
StringNative< 256 > compilerPath
Path to the compiler.
Definition: Plugin.h:141
Type type
Compile Type.
Definition: Plugin.h:140
Result compile(const PluginDefinition &definition, const PluginSysroot &sysroot, const PluginCompilerEnvironment &environment, String &compilerLog) const
Compiles a Definition to an object file.
Type
Compiler type (clang/gcc/msvc)
Definition: Plugin.h:135
static Result findBestCompiler(PluginCompiler &compiler)
Look for best compiler on current system.
StringNative< 256 > linkerPath
Path to the linker.
Definition: Plugin.h:142
Result link(const PluginDefinition &definition, const PluginSysroot &sysroot, const PluginCompilerEnvironment &environment, StringView executablePath, String &linkerLog) const
Links a Definition into a dynamic library, with symbols from executablePath
Plugin description, category, dependencies, files and directory location.
Definition: Plugin.h:54
Result getDynamicLibraryAbsolutePath(String &fullDynamicPath) const
Gets absolute path of where compiled dynamic library will exist after plugin is compiled.
SmallString< 255 > description
Long description of plugin.
Definition: Plugin.h:56
SmallVector< PluginIdentifier, 8 > dependencies
Dependencies necessary to load this plugin.
Definition: Plugin.h:60
PluginIdentity identity
Uniquely identifier a plugin.
Definition: Plugin.h:55
static bool find(const StringView text, StringView &extracted)
Extracts the plugin definition (SC_BEGIN_PLUGIN / SC_END_PLUGIN) comment from a .cpp file.
const PluginFile & getMainPluginFile() const
Get main plugin file, holding plugin definition.
Definition: Plugin.h:70
SmallString< 255 > directory
Path to the directory holding the plugin.
Definition: Plugin.h:58
Result getDynamicLibraryPDBAbsolutePath(String &fullDynamicPath) const
Gets absolute path of where compiled Program Database File will exist after plugin is compiled.
SmallVector< PluginFile, 10 > files
Source files that compose this plugin.
Definition: Plugin.h:62
static bool parse(StringView text, PluginDefinition &pluginDefinition)
Parses an extracted plugin definition text.
SmallVector< SmallString< 10 >, 8 > build
Build options.
Definition: Plugin.h:61
SmallString< 10 > category
Category where plugin belongs to.
Definition: Plugin.h:57
PluginFile & getMainPluginFile()
Get main plugin file, holding plugin definition.
Definition: Plugin.h:66
A plugin dynamic library loaded from a SC::PluginRegistry.
Definition: Plugin.h:188
String lastErrorLog
Last error log of compiler / linker (if any)
Definition: Plugin.h:193
Time::Absolute lastLoadTime
Last time when this plugin was last loaded.
Definition: Plugin.h:191
SystemDynamicLibrary dynamicLibrary
System handle of plugin's dynamic library.
Definition: Plugin.h:190
PluginDefinition definition
Definition of the loaded plugin.
Definition: Plugin.h:189
uint32_t numReloads
Number of times that the plugin has been hot-reloaded.
Definition: Plugin.h:192
bool queryInterface(T *&outInterface) const
Try to obtain a given interface as exported by a plugin through SC_PLUGIN_EXPORT_INTERFACES macro.
Definition: Plugin.h:199
Holds path to a given plugin source file.
Definition: Plugin.h:35
SmallString< 255 > absolutePath
Absolute path to a plugin source file.
Definition: Plugin.h:36
Represents the unique signature / identity of a Plugin.
Definition: Plugin.h:41
bool operator==(const PluginIdentity &other) const
Compares two plugins on Identity::identifier.
Definition: Plugin.h:49
SmallString< 10 > version
Plugin version (x.y.z)
Definition: Plugin.h:44
PluginIdentifier identifier
Unique string identifying the plugin.
Definition: Plugin.h:42
SmallString< 30 > name
Plugin name.
Definition: Plugin.h:43
Holds a registry of plugins, loading and compiling them on the fly.
Definition: Plugin.h:224
LoadMode
Instructs loadPlugin to Load or Reload the plugin.
Definition: Plugin.h:235
Result unloadPlugin(const StringView identifier)
Unloads an already loaded plugin by its identifier.
Result loadPlugin(const StringView identifier, const PluginCompiler &compiler, const PluginSysroot &sysroot, StringView executablePath, LoadMode loadMode=LoadMode::Load)
Loads a plugin with given identifier, compiling it with given PluginCompiler.
size_t getNumberOfEntries() const
Returns the total number of registry entries (counting both loaded and unloaded plugins)
Definition: Plugin.h:267
Result removeAllBuildProducts(const StringView identifier)
Removes all temporary build products of the Plugin with given identifier.
const PluginDynamicLibrary * findPlugin(const StringView identifier)
Find a PluginDynamicLibrary in the registry with a given identifier.
const PluginDynamicLibrary & getPluginDynamicLibraryAt(size_t index)
Returns the PluginIdentifier corresponding to the index entry of the registry.
Definition: Plugin.h:273
Result close()
Unregisters all plugins.
const PluginIdentifier & getIdentifierAt(size_t index) const
Returns the PluginIdentifier corresponding to the index entry of the registry.
Definition: Plugin.h:270
Result replaceDefinitions(Vector< PluginDefinition > &&definitions)
Appends the definitions to registry.
void getPluginsToReloadBecauseOf(StringView relativePath, Time::Milliseconds tolerance, Function< void(const PluginIdentifier &)> onPlugin)
Enumerates all plugins that must be reloaded when relativePath is modified.
Scans a directory for PluginDefinition.
Definition: Plugin.h:103
static Result scanDirectory(const StringView directory, Vector< PluginDefinition > &definitions)
Scans a directory for PluginDefinition.
Holds include and library paths for a system toolchain, used to let plugins link to libc and libc++.
Definition: Plugin.h:163
SmallVector< StringNative< 256 >, 8 > includePaths
Path to system include directories.
Definition: Plugin.h:164
static Result findBestSysroot(PluginCompiler::Type compiler, PluginSysroot &sysroot)
Finds a reasonable sysroot for the given compiler.
SmallVector< StringNative< 256 >, 8 > libraryPaths
Path to system library directories.
Definition: Plugin.h:165
StringNative< 256 > isysroot
Path to sysroot include (optional)
Definition: Plugin.h:167
An ascii string used as boolean result. SC_TRY macro forwards errors to caller.
Definition: Result.h:12
String with compile time configurable inline storage (small string optimization)
Definition: String.h:168
A Vector that can hold up to N elements inline and > N on heap.
Definition: Vector.h:260
A non-modifiable owning string with associated encoding.
Definition: String.h:29
A string iterator for ASCII strings.
Definition: StringIterator.h:244
Non-owning view over a range of characters with UTF Encoding.
Definition: StringView.h:47
Absolute time as realtime or monotonically increasing clock.
Definition: Time.h:114
Type-safe wrapper of uint64 used to represent milliseconds.
Definition: Time.h:44
A contiguous sequence of heap allocated elements.
Definition: Vector.h:189
A map holding VectorMapItem key-value pairs in an unsorted Vector.
Definition: VectorMap.h:35