// -*- c++ -*- // Distributed under the BSD 2-Clause License. // See accompanying file LICENSE for details. #include "rebuild.h" #include #include #include #include "configure.h" #include "settings.h" #include "libctor.h" #include "tasks.h" #include "build.h" std::array configFiles; std::size_t numConfigFiles{0}; // TODO: Use c++20 when ready, somehing like this: //int reg(const std::source_location location = std::source_location::current()) int reg(const char* location, std::vector (*cb)()) { // NOTE: std::cout cannot be used here if(numConfigFiles >= configFiles.size()) { fprintf(stderr, "Max %d build configurations currently supported.\n", (int)configFiles.size()); exit(1); } configFiles[numConfigFiles].file = location; configFiles[numConfigFiles].cb = cb; ++numConfigFiles; return 0; } int unreg(const char* location) { std::size_t found{0}; for(std::size_t i = 0; i < numConfigFiles;) { if(std::string(location) == configFiles[i].file) { ++found; for(std::size_t j = i; j < numConfigFiles; ++j) { configFiles[j] = configFiles[j + 1]; } --numConfigFiles; } else { ++i; } } for(std::size_t i = 0; i < numExternalConfigFiles;) { if(std::string(location) == externalConfigFiles[i].file) { ++found; for(std::size_t j = i; j < numExternalConfigFiles; ++j) { externalConfigFiles[j] = externalConfigFiles[j + 1]; } --numExternalConfigFiles; } else { ++i; } } return found; } std::array externalConfigFiles; std::size_t numExternalConfigFiles{0}; // TODO: Use c++20 when ready, somehing like this: //int reg(const std::source_location location = std::source_location::current()) int reg(const char* location, std::vector (*cb)()) { // NOTE: std::cout cannot be used here if(numExternalConfigFiles >= externalConfigFiles.size()) { fprintf(stderr, "Max %d external configurations currently supported.\n", (int)externalConfigFiles.size()); exit(1); } externalConfigFiles[numExternalConfigFiles].file = location; externalConfigFiles[numExternalConfigFiles].cb = cb; ++numExternalConfigFiles; return 0; } namespace { bool contains(const std::vector& sources, const std::string& file) { for(const auto& source : sources) { if(source.file == file) { return true; } } return false; } } void recompileCheck(const Settings& global_settings, int argc, char* argv[], bool force, bool relaunch_allowed) { using namespace std::string_literals; if(global_settings.verbose > 1) { std::cout << "Recompile check (" << numConfigFiles << "):\n"; } BuildConfiguration config; config.name = "ctor"; config.cxxflags = std::vector({ "-s", "-O3", "-std=c++17" }); if(hasConfiguration(cfg::ctor_includedir)) { config.cxxflags.push_back("-I"s + getConfiguration(cfg::ctor_includedir)); } if(hasConfiguration(cfg::ctor_libdir)) { config.ldflags.push_back("-L"s + getConfiguration(cfg::ctor_libdir)); } config.ldflags.push_back("-lctor"); config.ldflags.push_back("-pthread"); Settings settings{global_settings}; settings.verbose = -1; // Make check completely silent. settings.builddir += "/ctor"; // override builddir to use ctor subdir { std::filesystem::path buildfile = settings.builddir; std::filesystem::path currentfile = argv[0]; config.target = std::filesystem::relative(currentfile, buildfile).string(); } if(std::filesystem::exists(configurationFile)) { config.sources.push_back(configurationFile.string()); } for(std::size_t i = 0; i < numConfigFiles; ++i) { std::string location = configFiles[i].file; if(global_settings.verbose > 1) { std::cout << " - " << location << "\n"; } // Ensure that files containing multiple configurations are only added once. if(!contains(config.sources, location)) { config.sources.push_back(location); } } for(std::size_t i = 0; i < numExternalConfigFiles; ++i) { std::string location = externalConfigFiles[i].file; if(global_settings.verbose > 1) { std::cout << " - " << location << "\n"; } // Ensure that files containing multiple configurations are only added once. if(!contains(config.sources, location)) { config.sources.push_back(location); } } auto tasks = taskFactory({config}, settings, {}); for(auto task : tasks) { if(task->registerDepTasks(tasks)) { return; } } auto dirty_tasks = build(settings, "ctor", tasks, true); // dryrun if(dirty_tasks) { std::cout << "Rebuilding config.\n"; build(settings, "ctor", tasks); // run for real } }