// -*- c++ -*- // Distributed under the BSD 2-Clause License. // See accompanying file LICENSE for details. #include "task_ld.h" #include #include #include "ctor.h" #include "execute.h" #include "util.h" #include "tools.h" TaskLD::TaskLD(const ctor::build_configuration& config_, const ctor::settings& settings_, const std::string& target, const std::vector& objects, const std::string& sourceDir_, bool is_self_) : Task(config_, settings_, sourceDir_) , config(config_) , settings(settings_) , sourceDir(sourceDir_) , is_self(is_self_) { target_type = config.type; output_system = config.system; if(target_type == ctor::target_type::automatic) { target_type = ctor::target_type::executable; } _targetFile = target; auto toolchain = getToolChain(config.system); _targetFile = extension(toolchain, target_type, config.system, _targetFile); for(const auto& object : objects) { std::filesystem::path objectFile = object; objectFiles.push_back(objectFile); dependsStr.push_back(objectFile.string()); } for(const auto& dep : config.depends) { depFiles.emplace_back(dep); } flagsFile = std::filesystem::path(settings.builddir) / cleanUp(sourceDir) / targetFile().stem(); flagsFile += ".flags"; source_language = ctor::language::c; for(const auto& source : config.sources) { std::filesystem::path sourceFile(source.file); if(sourceFile.extension().string() != ".c") { source_language = ctor::language::cpp; } } std::filesystem::create_directories(targetFile().parent_path()); } bool TaskLD::dirtyInner() { if(!std::filesystem::exists(targetFile())) { return true; } if(!std::filesystem::exists(flagsFile)) { return true; } { auto lastFlags = readFile(flagsFile.string()); if(flagsString() != lastFlags) { //std::cout << "The compiler flags changed\n"; return true; } } return false; } int TaskLD::runInner() { auto toolchain = getToolChain(config.system); std::vector args; for(const auto& dep : getDependsTasks()) { auto depFile = dep->targetFile(); auto dep_type = target_type_from_extension(toolchain, depFile); if(dep_type == ctor::target_type::dynamic_library) { append(args, ld_option(toolchain, ctor::ld_opt::library_path, targetFile().parent_path().string())); auto lib = depFile.stem().string().substr(3); // strip 'lib' prefix append(args, ld_option(toolchain, ctor::ld_opt::link, lib)); } else if(dep_type == ctor::target_type::static_library || dep_type == ctor::target_type::object) { args.push_back(depFile.string()); } } for(const auto& flag : config.flags.ldflags) { append(args, to_strings(toolchain, flag)); } append(args, ld_option(toolchain, ctor::ld_opt::output, targetFile().string())); append(args, to_strings(toolchain, {ctor::toolchain::msvc, ctor::ld_opt::custom, "/nologo"})); { // Write flags to file. std::ofstream flagsStream(flagsFile); flagsStream << flagsString(); } if(settings.verbose == 0) { std::cout << "LD => " << targetFile().string() << std::endl; } std::string tool; const auto& c = ctor::get_configuration(); if(toolchain == ctor::toolchain::gcc || toolchain == ctor::toolchain::clang) { tool = compiler(); } else // msvc { switch(outputSystem()) { case ctor::output_system::host: tool = c.get(ctor::cfg::host_ld, "/usr/bin/ld"); break; case ctor::output_system::build: tool = c.get(ctor::cfg::build_ld, "/usr/bin/ld"); break; } } return execute(settings, tool, args, c.env, is_self); } int TaskLD::clean() { if(std::filesystem::exists(targetFile())) { std::cout << "Removing " << targetFile().string() << "\n"; std::filesystem::remove(targetFile()); } if(std::filesystem::exists(flagsFile)) { std::cout << "Removing " << flagsFile.string() << "\n"; std::filesystem::remove(flagsFile); } return 0; } std::vector TaskLD::depends() const { std::vector deps; for(const auto& objectFile : objectFiles) { deps.push_back(objectFile.string()); } for(const auto& depFile : depFiles) { deps.push_back(depFile.string()); } return deps; } std::string TaskLD::target() const { return _targetFile.string(); } std::filesystem::path TaskLD::targetFile() const { return std::filesystem::path(settings.builddir) / sourceDir / _targetFile; } bool TaskLD::derived() const { return false; } std::string TaskLD::flagsString() const { auto toolchain = getToolChain(config.system); std::string flagsStr; bool first{true}; for(const auto& flag : config.flags.ldflags) { for(const auto& str : to_strings(toolchain, flag)) { if(first) { flagsStr += " "; first = false; } flagsStr += str; } } for(const auto& dep : config.depends) { if(dep != config.depends[0]) { flagsStr += " "; } flagsStr += dep; } auto deps = depends(); for(const auto& dep : deps) { flagsStr += " "; flagsStr += dep; } return flagsStr; }