#include <msp/core/maputils.h>
#include <msp/datafile/parser.h>
#include <msp/fs/dir.h>
+#include <msp/fs/stat.h>
#include <msp/fs/utils.h>
#include <msp/io/buffered.h>
#include <msp/io/file.h>
#include <msp/io/print.h>
#include <msp/strings/format.h>
+#include <msp/strings/utils.h>
#include <msp/time/timedelta.h>
#include <msp/time/utils.h>
#include "binarypackage.h"
using CreateFunc = Plugin *(Builder &);
FS::Path plugins_dir = FS::get_sys_lib_dir();
+ if(!FS::exists(plugins_dir))
+ return;
+
logger->log("files", "Traversing %s", plugins_dir);
+ vector<LoadedPlugin> unordered_plugins;
for(const string &f: list_filtered(plugins_dir, "\\.dlm$"))
{
LoadedPlugin plugin;
if(plugin.plugin)
{
logger->log("plugins", "Loaded plugin %s", f);
- plugins.emplace_back(move(plugin));
+ unordered_plugins.emplace_back(move(plugin));
continue;
}
else
logger->log("plugins", "Failed to initialize plugin %s: %s", f, exc.what());
}
}
+
+ add_plugins(unordered_plugins);
+}
+
+void Builder::add_plugins(vector<LoadedPlugin> &unordered_plugins)
+{
+ auto have_plugin = [this](const string &r){
+ return any_of(plugins.begin(), plugins.end(), [&r](const LoadedPlugin &p){ return FS::basepart(FS::basename(p.path))==r; });
+ };
+
+ while(!unordered_plugins.empty())
+ {
+ bool any_added = false;
+ for(auto i=unordered_plugins.begin(); i!=unordered_plugins.end(); )
+ {
+ const vector<string> &required = i->plugin->get_required_plugins();
+ if(all_of(required.begin(), required.end(), have_plugin))
+ {
+ plugins.push_back(move(*i));
+ i = unordered_plugins.erase(i);
+ any_added = true;
+ }
+ else
+ ++i;
+ }
+
+ if(!any_added)
+ break;
+ }
+
+ for(const LoadedPlugin &p: unordered_plugins)
+ {
+ vector<string> missing;
+ for(const string &r: p.plugin->get_required_plugins())
+ if(!have_plugin(r))
+ missing.push_back(r);
+ logger->log("plugins", "Missing required plugins for plugin %s: %s",
+ FS::basename(p.path), join(missing.begin(), missing.end()));
+ }
}
void Builder::set_architecture(const string &name)
int Builder::build(unsigned jobs, bool dry_run, bool show_progress)
{
unsigned total = build_graph.count_rebuild_targets();
+ Time::TimeStamp start_time = Time::now();
if(!total)
{
vector<Task *> tasks;
unsigned count = 0;
+ Time::TimeDelta sum_time;
bool fail = false;
bool finish = false;
{
++count;
+ const vector<const FileTarget *> &targets = tasks[i]->get_targets();
+ if(!targets.empty())
+ {
+ sum_time += tasks[i]->get_duration();
+ get_logger().log("timings", "%s built in %s", targets.front()->get_name(), tasks[i]->get_duration());
+ }
+
delete tasks[i];
tasks.erase(tasks.begin()+i);
if(status==Task::ERROR)
else if(show_progress)
get_logger().log("summary", "Build complete");
+ Time::TimeStamp end_time = Time::now();
+ get_logger().log("timings", "Build took %s, with a total %s spent on tasks", end_time-start_time, sum_time);
+
return fail;
}
other.plugin = 0;
}
+Builder::LoadedPlugin &Builder::LoadedPlugin::operator=(LoadedPlugin &&other)
+{
+ delete plugin;
+ delete module;
+ path = move(other.path);
+ module = other.module;
+ plugin = other.plugin;
+ other.module = 0;
+ other.plugin = 0;
+ return *this;
+}
+
Builder::LoadedPlugin::~LoadedPlugin()
{
delete plugin;