-
-
Notifications
You must be signed in to change notification settings - Fork 1.6k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
They are not actually part of the store layer, but instead part of the Nix executable infra (libraries don't need plugins, executables do). This is part of a larger project of moving all of our legacy settings infra to libmain, and having the underlying libraries just have plain configuration structs detached from any settings infra / UI layer. Progress on #5638
- Loading branch information
1 parent
f8372cb
commit 679ed2b
Showing
8 changed files
with
131 additions
and
126 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -5,6 +5,7 @@ | |
#include "logging.hh" | ||
#include "loggers.hh" | ||
#include "util.hh" | ||
#include "plugin.hh" | ||
|
||
namespace nix { | ||
|
||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,117 @@ | ||
#ifndef _WIN32 | ||
# include <dlfcn.h> | ||
#endif | ||
|
||
#include "config-global.hh" | ||
#include "signals.hh" | ||
|
||
namespace nix { | ||
|
||
struct PluginFilesSetting : public BaseSetting<Paths> | ||
{ | ||
bool pluginsLoaded = false; | ||
|
||
PluginFilesSetting( | ||
Config * options, | ||
const Paths & def, | ||
const std::string & name, | ||
const std::string & description, | ||
const std::set<std::string> & aliases = {}) | ||
: BaseSetting<Paths>(def, true, name, description, aliases) | ||
{ | ||
options->addSetting(this); | ||
} | ||
|
||
Paths parse(const std::string & str) const override; | ||
}; | ||
|
||
Paths PluginFilesSetting::parse(const std::string & str) const | ||
{ | ||
if (pluginsLoaded) | ||
throw UsageError( | ||
"plugin-files set after plugins were loaded, you may need to move the flag before the subcommand"); | ||
return BaseSetting<Paths>::parse(str); | ||
} | ||
|
||
struct PluginSettings : Config | ||
{ | ||
PluginFilesSetting pluginFiles{ | ||
this, | ||
{}, | ||
"plugin-files", | ||
R"( | ||
A list of plugin files to be loaded by Nix. Each of these files will | ||
be dlopened by Nix. If they contain the symbol `nix_plugin_entry()`, | ||
this symbol will be called. Alternatively, they can affect execution | ||
through static initialization. In particular, these plugins may construct | ||
static instances of RegisterPrimOp to add new primops or constants to the | ||
expression language, RegisterStoreImplementation to add new store | ||
implementations, RegisterCommand to add new subcommands to the `nix` | ||
command, and RegisterSetting to add new nix config settings. See the | ||
constructors for those types for more details. | ||
Warning! These APIs are inherently unstable and may change from | ||
release to release. | ||
Since these files are loaded into the same address space as Nix | ||
itself, they must be DSOs compatible with the instance of Nix | ||
running at the time (i.e. compiled against the same headers, not | ||
linked to any incompatible libraries). They should not be linked to | ||
any Nix libs directly, as those will be available already at load | ||
time. | ||
If an entry in the list is a directory, all files in the directory | ||
are loaded as plugins (non-recursively). | ||
)"}; | ||
}; | ||
|
||
static PluginSettings pluginSettings; | ||
|
||
static GlobalConfig::Register rPluginSettings(&pluginSettings); | ||
|
||
void initPlugins() | ||
{ | ||
assert(!pluginSettings.pluginFiles.pluginsLoaded); | ||
for (const auto & pluginFile : pluginSettings.pluginFiles.get()) { | ||
std::vector<std::filesystem::path> pluginFiles; | ||
try { | ||
auto ents = std::filesystem::directory_iterator{pluginFile}; | ||
for (const auto & ent : ents) { | ||
checkInterrupt(); | ||
pluginFiles.emplace_back(ent.path()); | ||
} | ||
} catch (std::filesystem::filesystem_error & e) { | ||
if (e.code() != std::errc::not_a_directory) | ||
throw; | ||
pluginFiles.emplace_back(pluginFile); | ||
} | ||
for (const auto & file : pluginFiles) { | ||
checkInterrupt(); | ||
/* handle is purposefully leaked as there may be state in the | ||
DSO needed by the action of the plugin. */ | ||
#ifndef _WIN32 // TODO implement via DLL loading on Windows | ||
void * handle = dlopen(file.c_str(), RTLD_LAZY | RTLD_LOCAL); | ||
if (!handle) | ||
throw Error("could not dynamically open plugin file '%s': %s", file, dlerror()); | ||
|
||
/* Older plugins use a statically initialized object to run their code. | ||
Newer plugins can also export nix_plugin_entry() */ | ||
void (*nix_plugin_entry)() = (void (*)()) dlsym(handle, "nix_plugin_entry"); | ||
if (nix_plugin_entry) | ||
nix_plugin_entry(); | ||
#else | ||
throw Error("could not dynamically open plugin file '%s'", file); | ||
#endif | ||
} | ||
} | ||
|
||
/* Since plugins can add settings, try to re-apply previously | ||
unknown settings. */ | ||
globalConfig.reapplyUnknownSettings(); | ||
globalConfig.warnUnknownSettings(); | ||
|
||
/* Tell the user if they try to set plugin-files after we've already loaded */ | ||
pluginSettings.pluginFiles.pluginsLoaded = true; | ||
} | ||
|
||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
#pragma once | ||
///@file | ||
|
||
namespace nix { | ||
|
||
/** | ||
* This should be called after settings are initialized, but before | ||
* anything else | ||
*/ | ||
void initPlugins(); | ||
|
||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters