/* Crafter® Build Copyright (C) 2026 Catcrafts® Catcrafts.net This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License version 3.0 as published by the Free Software Foundation; This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ module Crafter.Build:Interface_impl; import std; import :Interface; import :Platform; namespace fs = std::filesystem; namespace Crafter { ModulePartition::ModulePartition(std::string&& name, fs::path&& path) : name(std::move(name)), path(std::move(path)), compiled(false), checked(false) {} bool ModulePartition::Check(const fs::path& pcmDir, fs::file_time_type sourceFloor) { if(!checked) { checked = true; std::string pcmPath = (pcmDir/path.filename()).generic_string()+".pcm"; std::string cppmPath = path.generic_string()+".cppm"; if(fs::exists(pcmPath) && std::max(fs::last_write_time(cppmPath), sourceFloor) < fs::last_write_time(pcmPath)) { fs::file_time_type pcmTime = fs::last_write_time(pcmPath); for(ModulePartition* dependency : partitionDependencies) { if(dependency->Check(pcmDir, sourceFloor)) { needsRecompiling = true; return true; } } for(Module* dependency : moduleDependencies) { if(dependency->Check(pcmDir, sourceFloor)) { needsRecompiling = true; return true; } } for(const auto& [externalMod, externalPcmPath] : externalModuleDependencies) { std::error_code ec; fs::file_time_type t = fs::last_write_time(externalPcmPath, ec); if (!ec && t >= pcmTime) { needsRecompiling = true; return true; } } needsRecompiling = false; compiled.store(true); return false; } else { needsRecompiling = true; return true; } } else { return needsRecompiling; } } void ModulePartition::Compile(const std::string_view clang, const fs::path& pcmDir, const fs::path& buildDir, std::atomic& buildCancelled, std::string& buildError) { for(ModulePartition* dependency : partitionDependencies) { if(!dependency->compiled.load()) { dependency->compiled.wait(false); } } for(Module* dependency : moduleDependencies) { if(!dependency->compiled.load()) { dependency->compiled.wait(false); } } if (buildCancelled.load(std::memory_order_relaxed)) { compiled.store(true); compiled.notify_all(); return; } std::string result = RunCommand(std::format("{} {}.cppm --precompile -o {}.pcm", clang, path.string(), (pcmDir/path.filename()).string())); if (!result.empty()) { bool expected = false; if (buildCancelled.compare_exchange_strong(expected, true)) { buildError = std::move(result); compiled.store(true); compiled.notify_all(); return; } } compiled.store(true); compiled.notify_all(); result = RunCommand(std::format("{} -Wno-unused-command-line-argument {}.pcm -c -o {}.o", clang, (pcmDir/path.filename()).string(), (buildDir/path.filename()).string())); if (!result.empty()) { bool expected = false; if (buildCancelled.compare_exchange_strong(expected, true)) { buildError = std::move(result); } } } Module::Module(std::string&& name, fs::path&& path) : name(std::move(name)), path(std::move(path)), compiled(false), checked(false) {} bool Module::Check(const fs::path& pcmDir, fs::file_time_type sourceFloor) { if(!checked) { checked = true; std::string pcmPath = (pcmDir/path.filename()).generic_string()+".pcm"; std::string cppmPath = path.generic_string()+".cppm"; if(fs::exists(pcmPath) && std::max(fs::last_write_time(cppmPath), sourceFloor) < fs::last_write_time(pcmPath)) { bool depCheck = false; for(std::unique_ptr& partition : partitions) { if(partition->Check(pcmDir, sourceFloor)) { depCheck = true; } } if(depCheck) { needsRecompiling = true; return true; } else { needsRecompiling = false; compiled.store(true); return false; } } else { for(std::unique_ptr& partition : partitions) { partition->Check(pcmDir, sourceFloor); } needsRecompiling = true; return true; } } else { return needsRecompiling; } } void Module::Compile(const std::string_view clang, const fs::path& pcmDir, const fs::path& buildDir, std::atomic& buildCancelled, std::string& buildError) { std::vector threads; threads.reserve(partitions.size()); for(std::unique_ptr& part : partitions) { if(part->needsRecompiling) { threads.emplace_back(&ModulePartition::Compile, part.get(), clang, pcmDir, buildDir, std::ref(buildCancelled), std::ref(buildError)); } } for(std::thread& thread : threads){ thread.join(); } if (buildCancelled.load(std::memory_order_relaxed)) { compiled.store(true); compiled.notify_all(); return; } std::string result = RunCommand(std::format("{} {}.cppm --precompile -o {}.pcm", clang, path.string(), (pcmDir/path.filename()).string())); if (!result.empty()) { bool expected = false; if (buildCancelled.compare_exchange_strong(expected, true)) { buildError = std::move(result); compiled.store(true); compiled.notify_all(); return; } } compiled.store(true); compiled.notify_all(); result = RunCommand(std::format("{} -Wno-unused-command-line-argument {}.pcm -c -o {}.o", clang, (pcmDir/path.filename()).string(), (buildDir/path.filename()).string())); if (!result.empty()) { bool expected = false; if (buildCancelled.compare_exchange_strong(expected, true)) { buildError = std::move(result); } } } }