// -*- c++ -*- // Distributed under the BSD 2-Clause License. // See accompanying file LICENSE for details. #include "util.h" #include #include #include #include #include #include std::string to_lower(std::string str) { std::transform(str.begin(), str.end(), str.begin(), [](unsigned char c) { return std::tolower(c); }); return str; } std::string readFile(const std::string& fileName) { std::ifstream ifs(fileName.c_str(), std::ios::in | std::ios::binary | std::ios::ate); auto size = ifs.tellg(); if(size < 0) { return {}; } ifs.seekg(0, std::ios::beg); std::string bytes(static_cast(size), '\0'); ifs.read(bytes.data(), static_cast(bytes.size())); return bytes; } ctor::language languageFromExtension(const std::filesystem::path& file) { auto ext = file.extension().string(); // First a few case sensitive comparisons if(ext == ".c") { return ctor::language::c; } if(ext == ".C") { return ctor::language::cpp; } // The rest are compared in lowercase ext = to_lower(ext); if(ext == ".cc" || ext == ".cpp" || ext == ".c++" || ext == ".cp" || ext == ".cxx") { return ctor::language::cpp; } if(ext == ".s" || ext == ".asm") { return ctor::language::assembler; } std::cerr << "Could not deduce language from " << file.string() << "\n"; exit(1); return {}; } namespace { bool isClean(char c) { return c != '.' && c != '/'; } } std::string cleanUp(const std::string& path) { std::string cleaned; for(const auto& c : path) { if(isClean(c)) { cleaned += c; } else { cleaned += '_'; } } return cleaned; } std::string esc(const std::string& in) { std::string out; for(auto c : in) { switch(c) { case '\\': out += "\\\\"; break; case '"': out += "\\\""; break; default: out += c; break; } } return out; } std::vector get_paths(const std::string& path_env_) { std::string path_env; if(!path_env_.empty()) { path_env = path_env_; } else { get_env("PATH", path_env); } std::vector paths; #ifdef _WIN32 const char sep{';'}; #else const char sep{':'}; #endif std::stringstream ss(path_env); std::string path; while (std::getline(ss, path, sep)) { paths.push_back(path); } return paths; } bool check_executable(const std::filesystem::path& prog) { auto perms = std::filesystem::status(prog).permissions(); if((perms & std::filesystem::perms::owner_exec) != std::filesystem::perms::none) { return true; } if((perms & std::filesystem::perms::group_exec) != std::filesystem::perms::none) { return true; } if((perms & std::filesystem::perms::others_exec) != std::filesystem::perms::none) { return true; } return false; } std::string locate(const std::string& prog, const std::vector& paths, const std::string& arch) { std::string program = prog; if(!arch.empty()) { program = arch + "-" + prog; } // first check if arch contains an absolute path to prog if(std::filesystem::exists(program)) { if(check_executable(program)) { return program; } } for(const auto& path_str : paths) { std::filesystem::path path(path_str); auto prog_path = path / program; if(std::filesystem::exists(prog_path)) { if(check_executable(prog_path)) { return prog_path.string(); } } } return {}; } std::vector argsplit(const std::string& str) { enum class state { normal, in_quot, in_apotrophe, } state{state::normal}; bool esc{false}; std::string token; std::vector tokens; for(auto c : str) { switch(state) { case state::normal: if(esc) { esc = false; } else { if(c == ' ') { tokens.push_back(token); token.clear(); continue; } if(c == '\\') { esc = true; } if(c == '"') { state = state::in_quot; } if(c == '\'') { state = state::in_apotrophe; } } token += c; break; case state::in_quot: if(esc) { esc = false; } else { if(c == '\\') { esc = true; } if(c == '"') { state = state::normal; } } token += c; break; case state::in_apotrophe: if(esc) { esc = false; } else { if(c == '\\') { esc = true; } if(c == '\'') { state = state::normal; } } token += c; break; } } if(!token.empty()) { tokens.push_back(token); } return tokens; } bool get_env(std::string_view name, std::string& value) { auto var = getenv(name.data()); if(var) { value = var; return true; } return false; }