/* -*- mode: C++; c-basic-offset: 2; indent-tabs-mode: nil -*- */ /* * Main authors: * Guido Tack */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include #include #include #include #include #include #include #include #include #include #include using namespace std; namespace MiniZinc { bool SolverConfig::ExtraFlag::validate(const std::string& v) const { try { switch (flagType) { case FlagType::T_BOOL: case FlagType::T_STRING: return range.empty() || std::find(range.begin(), range.end(), v) != range.end(); case FlagType::T_INT: { long long i = stoll(v); return range.empty() || (i >= stoll(range[0]) && i <= stoll(range[1])); } case FlagType::T_FLOAT: { double i = stod(v); return range.empty() || (i >= stod(range[0]) && i <= stod(range[1])); } } } catch (const invalid_argument&) { return false; } catch (const out_of_range&) { return false; } return false; } namespace { std::string get_string(AssignI* ai) { if (auto* sl = ai->e()->dynamicCast()) { return std::string(sl->v().c_str(), sl->v().size()); } throw ConfigException("invalid configuration item (right hand side must be string)"); } bool get_bool(AssignI* ai) { if (auto* bl = ai->e()->dynamicCast()) { return bl->v(); } throw ConfigException("invalid configuration item (right hand side must be bool)"); } int get_int(AssignI* ai) { if (auto* il = ai->e()->dynamicCast()) { return static_cast(il->v().toInt()); } throw ConfigException("invalid configuration item (right hand side must be int)"); } std::vector get_string_list(AssignI* ai) { if (auto* al = ai->e()->dynamicCast()) { std::vector ret; for (unsigned int i = 0; i < al->size(); i++) { if (auto* sl = (*al)[i]->dynamicCast()) { ret.emplace_back(sl->v().c_str(), sl->v().size()); } else { throw ConfigException( "invalid configuration item (right hand side must be a list of strings)"); } } return ret; } throw ConfigException("invalid configuration item (right hand side must be a list of strings)"); } std::vector > get_string_pair_list(AssignI* ai) { if (auto* al = ai->e()->dynamicCast()) { std::vector > ret; if (al->dims() != 2 || al->min(1) != 1 || al->max(1) != 2) { throw ConfigException( "invalid configuration item (right hand side must be a 2d array of strings)"); } for (unsigned int i = 0; i < al->size(); i += 2) { auto* sl1 = (*al)[i]->dynamicCast(); auto* sl2 = (*al)[i + 1]->dynamicCast(); if ((sl1 != nullptr) && (sl2 != nullptr)) { ret.emplace_back(std::string(sl1->v().c_str(), sl1->v().size()), std::string(sl2->v().c_str(), sl2->v().size())); } else { throw ConfigException( "invalid configuration item (right hand side must be a 2d array of strings)"); } } return ret; } throw ConfigException( "invalid configuration item (right hand side must be a 2d array of strings)"); } std::vector > get_default_option_list(AssignI* ai) { if (auto* al = ai->e()->dynamicCast()) { std::vector > ret; if (al->size() == 0) { return ret; } if (al->dims() != 2) { throw ConfigException( "invalid configuration item (right hand side must be a 2d array of strings)"); } int nCols = al->max(1) - al->min(1) + 1; if (nCols != 3) { throw ConfigException( "invalid configuration item (right hand side must be a 2d array of strings with 3 " "columns)"); } for (unsigned int i = 0; i < al->size(); i += nCols) { auto* sl0 = (*al)[i]->dynamicCast(); auto* sl1 = (*al)[i + 1]->dynamicCast(); auto* sl2 = (*al)[i + 2]->dynamicCast(); if ((sl0 != nullptr) && (sl1 != nullptr) && (sl2 != nullptr)) { ret.push_back(std::vector({std::string(sl0->v().c_str(), sl0->v().size()), std::string(sl1->v().c_str(), sl1->v().size()), std::string(sl2->v().c_str(), sl2->v().size())})); } else { throw ConfigException( "invalid configuration item (right hand side must be a list of strings)"); } } return ret; } throw ConfigException( "invalid configuration item (right hand side must be a 2d array of strings)"); } std::vector get_extra_flag_list(AssignI* ai) { if (auto* al = ai->e()->dynamicCast()) { std::vector ret; if (al->size() == 0) { return ret; } if (al->dims() != 2) { throw ConfigException( "invalid configuration item (right hand side must be a 2d array of strings)"); } int nCols = al->max(1) - al->min(1) + 1; if (nCols < 2 || nCols > 4) { throw ConfigException( "invalid configuration item (right hand side must be a 2d array of strings)"); } bool haveType = (nCols >= 3); bool haveDefault = (nCols >= 4); for (unsigned int i = 0; i < al->size(); i += nCols) { auto* sl1 = (*al)[i]->dynamicCast(); auto* sl2 = (*al)[i + 1]->dynamicCast(); StringLit* sl3 = haveType ? (*al)[i + 2]->dynamicCast() : nullptr; StringLit* sl4 = haveDefault ? (*al)[i + 3]->dynamicCast() : nullptr; std::string opt_type_full = sl3 != nullptr ? std::string(sl3->v().c_str(), sl3->v().size()) : "bool"; std::vector opt_range; std::string opt_def; if (sl4 != nullptr) { opt_def = std::string(sl4->v().c_str(), sl4->v().size()); } else if (opt_type_full == "bool") { opt_def = "false"; } else if (opt_type_full == "int") { opt_def = "0"; } else if (opt_type_full == "float") { opt_def = "0.0"; } size_t split = opt_type_full.find(":"); std::string opt_type = opt_type_full.substr(0, split); SolverConfig::ExtraFlag::FlagType flag_type; if (opt_type == "bool") { flag_type = SolverConfig::ExtraFlag::FlagType::T_BOOL; } else if (opt_type == "int") { flag_type = SolverConfig::ExtraFlag::FlagType::T_INT; } else if (opt_type == "float") { flag_type = SolverConfig::ExtraFlag::FlagType::T_FLOAT; } else if (opt_type == "string" || opt_type == "opt") { flag_type = SolverConfig::ExtraFlag::FlagType::T_STRING; } if (split != std::string::npos) { opt_type_full = opt_type_full.substr(split + 1); while (!opt_type_full.empty()) { split = opt_type_full.find(":"); opt_range.push_back(opt_type_full.substr(0, split)); opt_type_full = split == std::string::npos ? "" : opt_type_full.substr(split + 1); } } if ((sl1 != nullptr) && (sl2 != nullptr)) { ret.emplace_back(std::string(sl1->v().c_str(), sl1->v().size()), std::string(sl2->v().c_str(), sl2->v().size()), flag_type, opt_range, opt_def); } else { throw ConfigException( "invalid configuration item (right hand side must be a 2d array of strings)"); } } return ret; } throw ConfigException( "invalid configuration item (right hand side must be a 2d array of strings)"); } std::string get_env(const char* v) { std::string ret; #ifdef _MSC_VER size_t len; getenv_s(&len, nullptr, 0, v); if (len > 0) { char* p = static_cast(malloc(len * sizeof(char))); getenv_s(&len, p, len, v); if (len > 0) { ret = p; } free(p); } #else char* p = getenv(v); if (p != nullptr) { ret = p; } #endif return ret; } char char_to_lower(char c) { return static_cast(std::tolower(static_cast(c))); } std::string string_to_lower(std::string s) { std::transform(s.begin(), s.end(), s.begin(), char_to_lower); return s; } struct SortByLowercase { bool operator()(const std::string& n1, const std::string& n2) { for (size_t i = 0; i < n1.size() && i < n2.size(); i++) { if (std::tolower(n1[i]) != std::tolower(n2[i])) { return std::tolower(n1[i]) < std::tolower(n2[i]); } } return n1.size() < n2.size(); } }; struct SortByName { const std::vector& solvers; SortByLowercase sortByLowercase; SortByName(const std::vector& solvers0) : solvers(solvers0) {} bool operator()(int idx1, int idx2) { return sortByLowercase(solvers[idx1].name(), solvers[idx2].name()); } }; } // namespace SolverConfig SolverConfig::load(const string& filename) { SolverConfig sc; sc._configFile = FileUtils::file_path(filename); ostringstream errstream; try { Env confenv; Model* m = nullptr; if (JSONParser::fileIsJSON(filename)) { JSONParser jp(confenv.envi()); try { m = new Model; GCLock lock; jp.parse(m, filename, false); } catch (JSONError& e) { delete m; m = nullptr; throw ConfigException(e.msg()); } } else { vector filenames; filenames.push_back(filename); m = parse(confenv, filenames, vector(), "", "", vector(), false, true, false, false, errstream); } if (m != nullptr) { bool hadId = false; bool hadVersion = false; bool hadName = false; string basePath = FileUtils::dir_name(sc._configFile); for (auto& i : *m) { if (auto* ai = i->dynamicCast()) { if (ai->id() == "id") { sc._id = get_string(ai); hadId = true; } else if (ai->id() == "name") { sc._name = get_string(ai); hadName = true; } else if (ai->id() == "executable") { std::string exePath = get_string(ai); sc._executable = exePath; std::string exe = FileUtils::find_executable(FileUtils::file_path(exePath, basePath)); int nr_found = (int)(!exe.empty()); std::string tmp = FileUtils::file_path(FileUtils::find_executable(exePath)); nr_found += (int)((!tmp.empty()) && tmp != exe); exe = exe.empty() ? tmp : exe; if (nr_found > 0) { sc._executableResolved = exe; if (nr_found > 1) { std::cerr << "Warning: multiple executables '" << exePath << "' found on the system, using '" << exe << "'" << std::endl; } } } else if (ai->id() == "mznlib") { std::string libPath = get_string(ai); sc._mznlib = libPath; if (!libPath.empty()) { if (libPath[0] == '-') { sc._mznlibResolved = libPath; } else { sc._mznlibResolved = FileUtils::file_path(libPath, basePath); } } } else if (ai->id() == "version") { sc._version = get_string(ai); hadVersion = true; } else if (ai->id() == "mznlibVersion") { sc._mznlibVersion = get_int(ai); } else if (ai->id() == "description") { sc._description = get_string(ai); } else if (ai->id() == "contact") { sc._contact = get_string(ai); } else if (ai->id() == "website") { sc._website = get_string(ai); } else if (ai->id() == "supportsMzn") { sc._supportsMzn = get_bool(ai); } else if (ai->id() == "supportsFzn") { sc._supportsFzn = get_bool(ai); } else if (ai->id() == "supportsNL") { sc._supportsNL = get_bool(ai); } else if (ai->id() == "needsSolns2Out") { sc._needsSolns2Out = get_bool(ai); } else if (ai->id() == "isGUIApplication") { sc._isGUIApplication = get_bool(ai); } else if (ai->id() == "needsMznExecutable") { sc._needsMznExecutable = get_bool(ai); } else if (ai->id() == "needsStdlibDir") { sc._needsStdlibDir = get_bool(ai); } else if (ai->id() == "needsPathsFile") { sc._needsPathsFile = get_bool(ai); } else if (ai->id() == "tags") { sc._tags = get_string_list(ai); } else if (ai->id() == "stdFlags") { sc._stdFlags = get_string_list(ai); } else if (ai->id() == "requiredFlags") { sc._requiredFlags = get_string_list(ai); } else if (ai->id() == "extraFlags") { sc._extraFlags = get_extra_flag_list(ai); } else { std::ostringstream ss; ss << "invalid configuration item (" << ai->id() << ")"; throw ConfigException(ss.str()); } } else { throw ConfigException("invalid configuration item"); } } if (!hadId) { throw ConfigException("invalid solver configuration (missing id)"); } if (!hadVersion) { throw ConfigException("invalid solver configuration (missing version)"); } if (!hadName) { throw ConfigException("invalid solver configuration (missing name)"); } } else { throw ConfigException(errstream.str()); } } catch (ConfigException&) { throw; } catch (Exception& e) { throw ConfigException(e.what()); } return sc; } std::string SolverConfig::toJSON(const SolverConfigs& configs) const { GCLock lock; std::ostringstream oss; auto def_id = configs.defaultSolver(""); oss << "{\n"; oss << " \"extraInfo\": {\n"; if (!def_id.empty() && def_id == id()) { oss << " \"isDefault\": true,\n"; } if (!mznlibResolved().empty()) { oss << " \"mznlib\": \"" << Printer::escapeStringLit(mznlibResolved()) << "\",\n"; } if (!executableResolved().empty()) { oss << " \"executable\": \"" << Printer::escapeStringLit(executableResolved()) << "\",\n"; } oss << " \"configFile\": \"" << Printer::escapeStringLit(configFile()) << "\""; if (!defaultFlags().empty()) { oss << ",\n \"defaultFlags\": ["; for (unsigned int j = 0; j < defaultFlags().size(); j++) { oss << "\"" << Printer::escapeStringLit(defaultFlags()[j]) << "\""; if (j < defaultFlags().size() - 1) { oss << ","; } } oss << "]"; } oss << "\n"; oss << " },\n"; oss << " \"id\": \"" << Printer::escapeStringLit(id()) << "\",\n"; oss << " \"name\": \"" << Printer::escapeStringLit(name()) << "\",\n"; oss << " \"version\": \"" << Printer::escapeStringLit(version()) << "\",\n"; if (!mznlib().empty()) { oss << " \"mznlib\": \"" << Printer::escapeStringLit(mznlib()) << "\",\n"; } if (!executable().empty()) { oss << " \"executable\": \"" << Printer::escapeStringLit(executable()) << "\",\n"; } oss << " \"mznlibVersion\": " << mznlibVersion() << ",\n"; if (!description().empty()) { oss << " \"description\": \"" << Printer::escapeStringLit(description()) << "\",\n"; } if (!contact().empty()) { oss << " \"contact\": \"" << Printer::escapeStringLit(contact()) << "\",\n"; } if (!website().empty()) { oss << " \"website\": \"" << Printer::escapeStringLit(website()) << "\",\n"; } if (!requiredFlags().empty()) { oss << " \"requiredFlags\": ["; for (unsigned int j = 0; j < requiredFlags().size(); j++) { oss << "\"" << requiredFlags()[j] << "\""; if (j < requiredFlags().size() - 1) { oss << ","; } } oss << "],\n"; } if (!stdFlags().empty()) { oss << " \"stdFlags\": ["; for (unsigned int j = 0; j < stdFlags().size(); j++) { oss << "\"" << stdFlags()[j] << "\""; if (j < stdFlags().size() - 1) { oss << ","; } } oss << "],\n"; } if (!extraFlags().empty()) { oss << " \"extraFlags\": ["; for (unsigned int j = 0; j < extraFlags().size(); j++) { oss << "\n [" << "\"" << Printer::escapeStringLit(extraFlags()[j].flag) << "\",\"" << Printer::escapeStringLit(extraFlags()[j].description) << "\",\""; switch (extraFlags()[j].flagType) { case ExtraFlag::FlagType::T_BOOL: oss << "bool"; break; case ExtraFlag::FlagType::T_INT: oss << "int"; break; case ExtraFlag::FlagType::T_FLOAT: oss << "float"; break; case ExtraFlag::FlagType::T_STRING: oss << (extraFlags()[j].range.empty() ? "string" : "opt"); break; } for (const auto& v : extraFlags()[j].range) { oss << ":" << Printer::escapeStringLit(v); } oss << "\",\"" << Printer::escapeStringLit(extraFlags()[j].defaultValue) << "\"]"; if (j < extraFlags().size() - 1) { oss << ","; } } oss << "\n ],\n"; } if (!tags().empty()) { oss << " \"tags\": ["; for (unsigned int j = 0; j < tags().size(); j++) { oss << "\"" << Printer::escapeStringLit(tags()[j]) << "\""; if (j < tags().size() - 1) { oss << ","; } } oss << "],\n"; } oss << " \"supportsMzn\": " << (supportsMzn() ? "true" : "false") << ",\n"; oss << " \"supportsFzn\": " << (supportsFzn() ? "true" : "false") << ",\n"; oss << " \"supportsNL\": " << (supportsNL() ? "true" : "false") << ",\n"; oss << " \"needsSolns2Out\": " << (needsSolns2Out() ? "true" : "false") << ",\n"; oss << " \"needsMznExecutable\": " << (needsMznExecutable() ? "true" : "false") << ",\n"; oss << " \"needsStdlibDir\": " << (needsStdlibDir() ? "true" : "false") << ",\n"; oss << " \"needsPathsFile\": " << (needsPathsFile() ? "true" : "false") << ",\n"; oss << " \"isGUIApplication\": " << (isGUIApplication() ? "true" : "false") << "\n"; oss << "}"; return oss.str(); } class BuiltinSolverConfigs { public: std::unordered_map builtinSolvers; }; BuiltinSolverConfigs& builtin_solver_configs() { static BuiltinSolverConfigs c; return c; } void SolverConfigs::addConfig(const MiniZinc::SolverConfig& sc) { int newIdx = static_cast(_solvers.size()); _solvers.push_back(sc); std::vector sc_tags = sc.tags(); std::string id = string_to_lower(sc.id()); std::string name = string_to_lower(sc.name()); sc_tags.push_back(id); size_t last_dot = id.find_last_of('.'); if (last_dot != std::string::npos) { std::string last_id = id.substr(last_dot + 1); if (last_id != name) { sc_tags.push_back(last_id); } } sc_tags.push_back(name); for (const auto& t : sc_tags) { auto it = _tags.find(t); if (it == _tags.end()) { _tags.insert(std::make_pair(t, std::vector({newIdx}))); } else { it->second.push_back(newIdx); } } } std::vector SolverConfigs::solverConfigsPath() const { return _solverPath; } SolverConfigs::SolverConfigs(std::ostream& log) { #ifdef _MSC_VER const char* PATHSEP = ";"; #else const char* PATHSEP = ":"; #endif std::string mzn_solver_path = get_env("MZN_SOLVER_PATH"); while (!mzn_solver_path.empty()) { size_t next_sep = mzn_solver_path.find(PATHSEP); string cur_path = mzn_solver_path.substr(0, next_sep); _solverPath.push_back(cur_path); if (next_sep != string::npos) { mzn_solver_path = mzn_solver_path.substr(next_sep + 1, string::npos); } else { mzn_solver_path = ""; } } std::string userConfigDir = FileUtils::user_config_dir(); if (FileUtils::directory_exists(userConfigDir + "/solvers")) { _solverPath.push_back(userConfigDir + "/solvers"); } std::vector configFiles( {FileUtils::global_config_file(), FileUtils::user_config_file()}); for (auto& cf : configFiles) { if (!cf.empty() && FileUtils::file_exists(cf)) { ostringstream errstream; try { Env userconfenv; Model* m = nullptr; if (JSONParser::fileIsJSON(cf)) { JSONParser jp(userconfenv.envi()); try { m = new Model; GCLock lock; jp.parse(m, cf, false); } catch (JSONError&) { delete m; m = nullptr; } } if (m != nullptr) { for (auto& i : *m) { if (auto* ai = i->dynamicCast()) { if (ai->id() == "mzn_solver_path") { std::vector sp = get_string_list(ai); for (const auto& s : sp) { _solverPath.push_back(s); } } else if (ai->id() == "mzn_lib_dir") { _mznlibDir = get_string(ai); } else if (ai->id() == "tagDefaults") { std::vector > tagDefs = get_string_pair_list(ai); for (auto& td : tagDefs) { std::string tag = td.first; std::string solver_id = td.second; _tagDefault[tag] = solver_id; } } else if (ai->id() == "solverDefaults") { std::vector > solverDefs = get_default_option_list(ai); for (auto& sd : solverDefs) { assert(sd.size() == 3); std::string solver = sd[0]; auto it = _solverDefaultOptions.find(solver); if (it == _solverDefaultOptions.end()) { std::vector solverOptions({sd[1], sd[2]}); _solverDefaultOptions.insert(std::make_pair(solver, solverOptions)); } else { std::vector& opts = it->second; bool found = false; for (unsigned int i = 0; i < opts.size(); i += 2) { if (opts[i] == sd[1]) { // Override existing option value opts[i + 1] = sd[2]; found = true; break; } } if (!found) { // Option didn't exist, add to end of list opts.push_back(sd[1]); opts.push_back(sd[2]); } } } } else { throw ConfigException("invalid configuration item"); } } else { throw ConfigException("invalid configuration item"); } } } else { std::cerr << errstream.str(); throw ConfigException("internal error"); } } catch (ConfigException& e) { log << "Warning: invalid configuration file: " << e.msg() << "\n"; } catch (Exception& e) { log << "Warning: invalid configuration file: " << e.what() << "\n"; } } } if (_mznlibDir.empty()) { _mznlibDir = FileUtils::file_path(FileUtils::share_directory()); } if (!_mznlibDir.empty()) { _solverPath.push_back(_mznlibDir + "/solvers"); } #ifndef _MSC_VER if (_mznlibDir != "/usr/local/share/minizinc" && FileUtils::directory_exists("/usr/local/share")) { _solverPath.emplace_back("/usr/local/share/minizinc/solvers"); } if (_mznlibDir != "/usr/share/minizinc" && FileUtils::directory_exists("/usr/share")) { _solverPath.emplace_back("/usr/share/minizinc/solvers"); } #endif } void SolverConfigs::populate(std::ostream& log) { for (const auto& sc : builtin_solver_configs().builtinSolvers) { addConfig(sc.second); } for (const string& cur_path : _solverPath) { std::vector configFiles = FileUtils::directory_list(cur_path, "msc"); for (auto& configFile : configFiles) { try { SolverConfig sc = SolverConfig::load(cur_path + "/" + configFile); addConfig(sc); } catch (ConfigException& e) { log << "Warning: error loading solver configuration from file " << cur_path << "/" << configFile << "\n"; log << "Error was:\n" << e.msg() << "\n"; } } } // Add default options to loaded solver configurations for (auto& sc : _solvers) { sc.defaultFlags(defaultOptions(sc.id())); } } vector SolverConfigs::solvers() const { // Find default solver, if present std::string def_id; auto def_it = _tagDefault.find(""); if (def_it != _tagDefault.end()) { def_id = def_it->second; } // Create sorted list of solvers vector s; for (const auto& sc : _solvers) { if (std::find(sc.tags().begin(), sc.tags().end(), "__internal__") != sc.tags().end()) { continue; } std::ostringstream oss; oss << sc.name() << " " << sc.version() << " (" << sc.id(); if (!def_id.empty() && sc.id() == def_id) { oss << ", default solver"; } for (const std::string& t : sc.tags()) { oss << ", " << t; } oss << ")"; s.push_back(oss.str()); } SortByLowercase sortByLowercase; std::sort(s.begin(), s.end(), sortByLowercase); return s; } std::string SolverConfigs::solverConfigsJSON() const { std::ostringstream oss; SortByName sortByName(_solvers); std::vector solversIdx(_solvers.size()); for (unsigned int i = 0; i < solversIdx.size(); i++) { solversIdx[i] = i; } std::sort(solversIdx.begin(), solversIdx.end(), sortByName); bool hadSolver = false; oss << "["; for (unsigned int i = 0; i < _solvers.size(); i++) { const SolverConfig& sc = _solvers[solversIdx[i]]; if (std::find(sc.tags().begin(), sc.tags().end(), "__internal__") != sc.tags().end()) { continue; } if (hadSolver) { oss << ","; } hadSolver = true; std::istringstream iss(sc.toJSON(*this)); std::string line; while (std::getline(iss, line)) { oss << "\n " << line; } } oss << "\n]\n"; return oss.str(); } namespace { std::string get_tag(const std::string& t) { return t.substr(0, t.find('@')); } std::string get_version(const std::string& t) { size_t sep = t.find('@'); return sep == string::npos ? "" : t.substr(sep + 1); } } // namespace const SolverConfig& SolverConfigs::config(const std::string& _s) { std::string s; if (_s.size() > 4 && _s.substr(_s.size() - 4) == ".msc") { SolverConfig sc = SolverConfig::load(_s); addConfig(sc); s = sc.id() + "@" + sc.version(); } else { s = _s; } s = string_to_lower(s); std::vector tags; std::istringstream iss(s); std::string next_s; while (std::getline(iss, next_s, ',')) { tags.push_back(next_s); } std::set defaultSolvers; std::set selectedSolvers; std::string firstTag; if (tags.empty()) { DefaultMap::const_iterator def_it = _tagDefault.find(""); if (def_it != _tagDefault.end()) { firstTag = def_it->second; } else { throw ConfigException("no solver selected"); } } else { firstTag = tags[0]; } TagMap::const_iterator tag_it = _tags.find(get_tag(firstTag)); if (tag_it == _tags.end()) { throw ConfigException("no solver with tag " + get_tag(firstTag) + " found"); } std::string tv = get_version(firstTag); for (int sidx : tag_it->second) { if (tv.empty() || tv == _solvers[sidx].version()) { selectedSolvers.insert(sidx); } } if (selectedSolvers.empty()) { // No matching version, only matching tag for (int sidx : tag_it->second) { selectedSolvers.insert(sidx); } } DefaultMap::const_iterator def_it = _tagDefault.find(get_tag(firstTag)); if (def_it != _tagDefault.end()) { defaultSolvers.insert(def_it->second); } for (unsigned int i = 1; i < tags.size(); i++) { tag_it = _tags.find(get_tag(tags[i])); if (tag_it == _tags.end()) { throw ConfigException("no solver with tag " + tags[i] + " found"); } tv = get_version(tags[i]); std::set newSolvers; for (int sidx : tag_it->second) { if (tv.empty() || tv == _solvers[sidx].version()) { newSolvers.insert(sidx); } } std::set intersection; std::set_intersection(selectedSolvers.begin(), selectedSolvers.end(), newSolvers.begin(), newSolvers.end(), std::inserter(intersection, intersection.begin())); selectedSolvers = intersection; if (selectedSolvers.empty()) { throw ConfigException("no solver with tags " + s + " found"); } def_it = _tagDefault.find(get_tag(tags[i])); if (def_it != _tagDefault.end()) { defaultSolvers.insert(def_it->second); } } int selectedSolver = -1; if (selectedSolvers.size() > 1) { // use default information for the tags to select a solver for (int sc_idx : selectedSolvers) { if (defaultSolvers.find(_solvers[sc_idx].id()) != defaultSolvers.end()) { selectedSolver = sc_idx; break; } } if (selectedSolver == -1) { selectedSolver = *selectedSolvers.begin(); } } else { selectedSolver = *selectedSolvers.begin(); } return _solvers[selectedSolver]; } std::vector SolverConfigs::defaultOptions(const std::string& id) { auto it = _solverDefaultOptions.find(id); if (it != _solverDefaultOptions.end()) { std::vector defaultOptions; for (const auto& df : it->second) { if (!df.empty()) { defaultOptions.push_back(df); } } return defaultOptions; } return {}; } void SolverConfigs::registerBuiltinSolver(const SolverConfig& sc) { builtin_solver_configs().builtinSolvers.insert(make_pair(sc.id(), sc)); } } // namespace MiniZinc