From ea0f93ac2416276961edbf49b93096a52b396211 Mon Sep 17 00:00:00 2001 From: Jendrik Seipp Date: Wed, 29 May 2024 17:08:19 +0200 Subject: [PATCH 1/2] [issue1138] Cleanly shut down the planner when it terminates. * Raise exception in exit_with() to ensure that destructors are called upon program exit. * Add and use non-reentrant report_exit_code() function. --- src/search/planner.cc | 72 ++++++++++++++++-------------- src/search/utils/system.cc | 17 ++++++- src/search/utils/system.h | 13 ++++++ src/search/utils/system_windows.cc | 13 ++---- 4 files changed, 70 insertions(+), 45 deletions(-) diff --git a/src/search/planner.cc b/src/search/planner.cc index e86ef8891f..a524825698 100644 --- a/src/search/planner.cc +++ b/src/search/planner.cc @@ -13,38 +13,44 @@ using namespace std; using utils::ExitCode; int main(int argc, const char **argv) { - utils::register_event_handlers(); - - if (argc < 2) { - utils::g_log << usage(argv[0]) << endl; - utils::exit_with(ExitCode::SEARCH_INPUT_ERROR); - } - - bool unit_cost = false; - if (static_cast(argv[1]) != "--help") { - utils::g_log << "reading input..." << endl; - tasks::read_root_task(cin); - utils::g_log << "done reading input!" << endl; - TaskProxy task_proxy(*tasks::g_root_task); - unit_cost = task_properties::is_unit_cost(task_proxy); + try { + utils::register_event_handlers(); + + if (argc < 2) { + utils::g_log << usage(argv[0]) << endl; + utils::exit_with(ExitCode::SEARCH_INPUT_ERROR); + } + + bool unit_cost = false; + if (static_cast(argv[1]) != "--help") { + utils::g_log << "reading input..." << endl; + tasks::read_root_task(cin); + utils::g_log << "done reading input!" << endl; + TaskProxy task_proxy(*tasks::g_root_task); + unit_cost = task_properties::is_unit_cost(task_proxy); + } + + shared_ptr search_algorithm = + parse_cmd_line(argc, argv, unit_cost); + + + utils::Timer search_timer; + search_algorithm->search(); + search_timer.stop(); + utils::g_timer.stop(); + + search_algorithm->save_plan_if_necessary(); + search_algorithm->print_statistics(); + utils::g_log << "Search time: " << search_timer << endl; + utils::g_log << "Total time: " << utils::g_timer << endl; + + ExitCode exitcode = search_algorithm->found_solution() + ? ExitCode::SUCCESS + : ExitCode::SEARCH_UNSOLVED_INCOMPLETE; + exit_with(exitcode); + } catch (const utils::ExitException &e) { + /* To ensure that all destructors are called before the program exits, + we raise an exception in utils::exit_with() and let main() return. */ + return static_cast(e.get_exitcode()); } - - shared_ptr search_algorithm = - parse_cmd_line(argc, argv, unit_cost); - - - utils::Timer search_timer; - search_algorithm->search(); - search_timer.stop(); - utils::g_timer.stop(); - - search_algorithm->save_plan_if_necessary(); - search_algorithm->print_statistics(); - utils::g_log << "Search time: " << search_timer << endl; - utils::g_log << "Total time: " << utils::g_timer << endl; - - ExitCode exitcode = search_algorithm->found_solution() - ? ExitCode::SUCCESS - : ExitCode::SEARCH_UNSOLVED_INCOMPLETE; - exit_with(exitcode); } diff --git a/src/search/utils/system.cc b/src/search/utils/system.cc index 001f68542a..46a2438908 100644 --- a/src/search/utils/system.cc +++ b/src/search/utils/system.cc @@ -44,9 +44,22 @@ bool is_exit_code_error_reentrant(ExitCode exitcode) { } } +void report_exit_code(ExitCode exitcode) { + const char *message = get_exit_code_message_reentrant(exitcode); + bool is_error = is_exit_code_error_reentrant(exitcode); + if (message) { + ostream &stream = is_error ? cerr : cout; + stream << message << endl; + } else { + cerr << "Exitcode: " << static_cast(exitcode) << endl + << "Unknown exitcode." << endl; + abort(); + } +} + void exit_with(ExitCode exitcode) { - report_exit_code_reentrant(exitcode); - exit(static_cast(exitcode)); + report_exit_code(exitcode); + throw ExitException(exitcode); } void exit_with_reentrant(ExitCode exitcode) { diff --git a/src/search/utils/system.h b/src/search/utils/system.h index 1f23ef19c7..9d5218afcb 100644 --- a/src/search/utils/system.h +++ b/src/search/utils/system.h @@ -53,6 +53,18 @@ enum class ExitCode { SEARCH_UNSUPPORTED = 34 }; +class ExitException : public std::exception { + ExitCode exitcode; +public: + explicit ExitException(ExitCode exitcode) + : exitcode(exitcode) { + } + + ExitCode get_exitcode() const { + return exitcode; + } +}; + NO_RETURN extern void exit_with(ExitCode returncode); NO_RETURN extern void exit_with_reentrant(ExitCode returncode); @@ -60,6 +72,7 @@ int get_peak_memory_in_kb(); const char *get_exit_code_message_reentrant(ExitCode exitcode); bool is_exit_code_error_reentrant(ExitCode exitcode); void register_event_handlers(); +void report_exit_code(ExitCode exitcode); void report_exit_code_reentrant(ExitCode exitcode); int get_process_id(); } diff --git a/src/search/utils/system_windows.cc b/src/search/utils/system_windows.cc index a244b68468..3fbe316d34 100644 --- a/src/search/utils/system_windows.cc +++ b/src/search/utils/system_windows.cc @@ -58,16 +58,9 @@ void register_event_handlers() { } void report_exit_code_reentrant(ExitCode exitcode) { - const char *message = get_exit_code_message_reentrant(exitcode); - bool is_error = is_exit_code_error_reentrant(exitcode); - if (message) { - ostream &stream = is_error ? cerr : cout; - stream << message << endl; - } else { - cerr << "Exitcode: " << static_cast(exitcode) << endl - << "Unknown exitcode." << endl; - abort(); - } + /* We call a function that uses ostreams even though this is unsafe in + reentrant code, because we don't know how to do it otherwise on Windows. */ + report_exit_code(exitcode); } int get_process_id() { From 00625e76ea789bd4c4d9d3512e4c421b03530b21 Mon Sep 17 00:00:00 2001 From: remochristen Date: Fri, 5 Jul 2024 17:29:59 +0200 Subject: [PATCH 2/2] [issue1134] Implement delete-relaxation operator-counting constraints by Rankooh and Rintanen. We implement the delete-relaxation operator-counting constraints described in "Efficient Computation and Informative Estimation of h^+ by Integer and Linear Programming" ICAPS 2022 by Masood Feyzbakhsh Rankooh and Jussi Rintanen. The command line option for the existing delete-relaxation constraint formulation by Tatsuya Imai and Alex Fukunaga changed from `delete_relaxation_constraints(...)` to `delete_relaxation_if_constraints(...)`. The Rankooh and Rintanen formulation generally shows better performance. We also fix a small bug in named vector. --- src/search/CMakeLists.txt | 3 +- src/search/algorithms/named_vector.h | 13 +- src/search/lp/cplex_solver_interface.h | 14 +- ...cc => delete_relaxation_if_constraints.cc} | 39 +- ...s.h => delete_relaxation_if_constraints.h} | 8 +- .../delete_relaxation_rr_constraints.cc | 637 ++++++++++++++++++ .../delete_relaxation_rr_constraints.h | 116 ++++ .../operator_counting_heuristic.cc | 4 + src/search/utils/hash.h | 7 +- 9 files changed, 810 insertions(+), 31 deletions(-) rename src/search/operator_counting/{delete_relaxation_constraints.cc => delete_relaxation_if_constraints.cc} (86%) rename src/search/operator_counting/{delete_relaxation_constraints.h => delete_relaxation_if_constraints.h} (89%) create mode 100644 src/search/operator_counting/delete_relaxation_rr_constraints.cc create mode 100644 src/search/operator_counting/delete_relaxation_rr_constraints.h diff --git a/src/search/CMakeLists.txt b/src/search/CMakeLists.txt index e9a8209a25..7d8177ba14 100644 --- a/src/search/CMakeLists.txt +++ b/src/search/CMakeLists.txt @@ -856,7 +856,8 @@ create_fast_downward_library( HELP "Plugin containing the code for operator-counting heuristics" SOURCES operator_counting/constraint_generator - operator_counting/delete_relaxation_constraints + operator_counting/delete_relaxation_if_constraints + operator_counting/delete_relaxation_rr_constraints operator_counting/lm_cut_constraints operator_counting/operator_counting_heuristic operator_counting/pho_constraints diff --git a/src/search/algorithms/named_vector.h b/src/search/algorithms/named_vector.h index 619053acf6..66814dde21 100644 --- a/src/search/algorithms/named_vector.h +++ b/src/search/algorithms/named_vector.h @@ -46,7 +46,8 @@ class NamedVector { void set_name(int index, const std::string &name) { assert(index >= 0 && index < size()); - if (index >= names.size()) { + int num_names = names.size(); + if (index >= num_names) { if (name.empty()) { // All unspecified names are empty by default. return; @@ -56,14 +57,18 @@ class NamedVector { names[index] = name; } - std::string get_name(int index) const { + const std::string &get_name(int index) const { assert(index >= 0 && index < size()); int num_names = names.size(); if (index < num_names) { return names[index]; } else { - // All unspecified names are empty by default. - return ""; + /* + All unspecified names are empty by default. We use a static + string here to avoid returning a reference to a local object. + */ + static std::string empty; + return empty; } } diff --git a/src/search/lp/cplex_solver_interface.h b/src/search/lp/cplex_solver_interface.h index db8d7ad64e..5ba35cd532 100644 --- a/src/search/lp/cplex_solver_interface.h +++ b/src/search/lp/cplex_solver_interface.h @@ -7,6 +7,7 @@ #include "../algorithms/named_vector.h" #include "../utils/memory.h" +#include #include namespace lp { @@ -144,15 +145,20 @@ class CplexSolverInterface : public SolverInterface { template explicit CplexNameData(const named_vector::NamedVector &values) { if (values.has_names()) { - names.resize(values.size()); - indices.resize(values.size()); + names.reserve(values.size()); + indices.reserve(values.size()); int num_values = values.size(); for (int i = 0; i < num_values; ++i) { - names[i] = values.get_name(i).data(); - indices[i] = i; + const std::string &name = values.get_name(i); + if (!name.empty()) { + // CPLEX copies the names, so the const_cast should be fine. + names.push_back(const_cast(name.data())); + indices.push_back(i); + } } } } + int size() {return names.size();} int *get_indices() { if (indices.empty()) { diff --git a/src/search/operator_counting/delete_relaxation_constraints.cc b/src/search/operator_counting/delete_relaxation_if_constraints.cc similarity index 86% rename from src/search/operator_counting/delete_relaxation_constraints.cc rename to src/search/operator_counting/delete_relaxation_if_constraints.cc index 959de4bad9..6890d0c2ae 100644 --- a/src/search/operator_counting/delete_relaxation_constraints.cc +++ b/src/search/operator_counting/delete_relaxation_if_constraints.cc @@ -1,4 +1,4 @@ -#include "delete_relaxation_constraints.h" +#include "delete_relaxation_if_constraints.h" #include "../task_proxy.h" @@ -21,37 +21,37 @@ static void add_lp_variables(int count, LPVariables &variables, vector &ind } -DeleteRelaxationConstraints::DeleteRelaxationConstraints(const plugins::Options &opts) +DeleteRelaxationIFConstraints::DeleteRelaxationIFConstraints(const plugins::Options &opts) : use_time_vars(opts.get("use_time_vars")), use_integer_vars(opts.get("use_integer_vars")) { } -int DeleteRelaxationConstraints::get_var_op_used(const OperatorProxy &op) { +int DeleteRelaxationIFConstraints::get_var_op_used(const OperatorProxy &op) { return lp_var_id_op_used[op.get_id()]; } -int DeleteRelaxationConstraints::get_var_fact_reached(FactPair f) { +int DeleteRelaxationIFConstraints::get_var_fact_reached(FactPair f) { return lp_var_id_fact_reached[f.var][f.value]; } -int DeleteRelaxationConstraints::get_var_first_achiever( +int DeleteRelaxationIFConstraints::get_var_first_achiever( const OperatorProxy &op, FactPair f) { return lp_var_id_first_achiever[op.get_id()][f.var][f.value]; } -int DeleteRelaxationConstraints::get_var_op_time(const OperatorProxy &op) { +int DeleteRelaxationIFConstraints::get_var_op_time(const OperatorProxy &op) { return lp_var_id_op_time[op.get_id()]; } -int DeleteRelaxationConstraints::get_var_fact_time(FactPair f) { +int DeleteRelaxationIFConstraints::get_var_fact_time(FactPair f) { return lp_var_id_fact_time[f.var][f.value]; } -int DeleteRelaxationConstraints::get_constraint_id(FactPair f) { +int DeleteRelaxationIFConstraints::get_constraint_id(FactPair f) { return constraint_ids[f.var][f.value]; } -void DeleteRelaxationConstraints::create_auxiliary_variables( +void DeleteRelaxationIFConstraints::create_auxiliary_variables( const TaskProxy &task_proxy, LPVariables &variables) { OperatorsProxy ops = task_proxy.get_operators(); int num_ops = ops.size(); @@ -94,8 +94,8 @@ void DeleteRelaxationConstraints::create_auxiliary_variables( } } -void DeleteRelaxationConstraints::create_constraints(const TaskProxy &task_proxy, - lp::LinearProgram &lp) { +void DeleteRelaxationIFConstraints::create_constraints(const TaskProxy &task_proxy, + lp::LinearProgram &lp) { LPVariables &variables = lp.get_variables(); LPConstraints &constraints = lp.get_constraints(); double infinity = lp.get_infinity(); @@ -213,7 +213,7 @@ void DeleteRelaxationConstraints::create_constraints(const TaskProxy &task_proxy } -void DeleteRelaxationConstraints::initialize_constraints( +void DeleteRelaxationIFConstraints::initialize_constraints( const shared_ptr &task, lp::LinearProgram &lp) { TaskProxy task_proxy(*task); create_auxiliary_variables(task_proxy, lp.get_variables()); @@ -221,7 +221,7 @@ void DeleteRelaxationConstraints::initialize_constraints( } -bool DeleteRelaxationConstraints::update_constraints( +bool DeleteRelaxationIFConstraints::update_constraints( const State &state, lp::LPSolver &lp_solver) { // Unset old bounds. for (FactPair f : last_state) { @@ -236,9 +236,9 @@ bool DeleteRelaxationConstraints::update_constraints( return false; } -class DeleteRelaxationConstraintsFeature : public plugins::TypedFeature { +class DeleteRelaxationIFConstraintsFeature : public plugins::TypedFeature { public: - DeleteRelaxationConstraintsFeature() : TypedFeature("delete_relaxation_constraints") { + DeleteRelaxationIFConstraintsFeature() : TypedFeature("delete_relaxation_if_constraints") { document_title("Delete relaxation constraints"); document_synopsis( "Operator-counting constraints based on the delete relaxation. By " @@ -277,10 +277,15 @@ class DeleteRelaxationConstraintsFeature : public plugins::TypedFeature _plugin; +static plugins::FeaturePlugin _plugin; } diff --git a/src/search/operator_counting/delete_relaxation_constraints.h b/src/search/operator_counting/delete_relaxation_if_constraints.h similarity index 89% rename from src/search/operator_counting/delete_relaxation_constraints.h rename to src/search/operator_counting/delete_relaxation_if_constraints.h index a28fed269e..7adfb543e9 100644 --- a/src/search/operator_counting/delete_relaxation_constraints.h +++ b/src/search/operator_counting/delete_relaxation_if_constraints.h @@ -1,5 +1,5 @@ -#ifndef OPERATOR_COUNTING_DELETE_RELAXATION_CONSTRAINTS_H -#define OPERATOR_COUNTING_DELETE_RELAXATION_CONSTRAINTS_H +#ifndef OPERATOR_COUNTING_DELETE_RELAXATION_IF_CONSTRAINTS_H +#define OPERATOR_COUNTING_DELETE_RELAXATION_IF_CONSTRAINTS_H #include "constraint_generator.h" @@ -20,7 +20,7 @@ namespace operator_counting { using LPConstraints = named_vector::NamedVector; using LPVariables = named_vector::NamedVector; -class DeleteRelaxationConstraints : public ConstraintGenerator { +class DeleteRelaxationIFConstraints : public ConstraintGenerator { bool use_time_vars; bool use_integer_vars; @@ -63,7 +63,7 @@ class DeleteRelaxationConstraints : public ConstraintGenerator { const TaskProxy &task_proxy, LPVariables &variables); void create_constraints(const TaskProxy &task_proxy, lp::LinearProgram &lp); public: - explicit DeleteRelaxationConstraints(const plugins::Options &opts); + explicit DeleteRelaxationIFConstraints(const plugins::Options &opts); virtual void initialize_constraints( const std::shared_ptr &task, diff --git a/src/search/operator_counting/delete_relaxation_rr_constraints.cc b/src/search/operator_counting/delete_relaxation_rr_constraints.cc new file mode 100644 index 0000000000..0e305279ec --- /dev/null +++ b/src/search/operator_counting/delete_relaxation_rr_constraints.cc @@ -0,0 +1,637 @@ +#include "delete_relaxation_rr_constraints.h" + +#include "../algorithms/priority_queues.h" +#include "../lp/lp_solver.h" +#include "../plugins/plugin.h" +#include "../algorithms/priority_queues.h" +#include "../task_proxy.h" +#include "../utils/markup.h" + +#include +#include +#include + +using namespace std; + +namespace operator_counting { +class VEGraph { + struct Node { + vector predecessors; + vector successors; + bool is_eliminated = false; + int in_degree; + }; + + /* + Vertex Elimination Graphs have one node per fact. We index them by + variable and value. + */ + vector> nodes; + utils::HashSet> delta; + utils::HashSet> edges; + priority_queues::AdaptiveQueue elimination_queue; + + Node &get_node(FactPair fact) { + return nodes[fact.var][fact.value]; + } + + const Node &get_node(FactPair fact) const { + return nodes[fact.var][fact.value]; + } + + void add_edge(FactPair from_fact, FactPair to_fact) { + pair edge = make_pair(from_fact, to_fact); + if (!edges.count(edge)) { + get_node(from_fact).successors.push_back(to_fact); + get_node(to_fact).predecessors.push_back(from_fact); + edges.insert(edge); + } + } + + void push_fact(FactPair fact) { + Node &node = get_node(fact); + if (node.is_eliminated) { + return; + } + int in_degree = 0; + for (FactPair predecessor : node.predecessors) { + if (!get_node(predecessor).is_eliminated) { + ++in_degree; + } + } + node.in_degree = in_degree; + elimination_queue.push(in_degree, fact); + } + + optional pop_fact() { + while (!elimination_queue.empty()) { + const auto [key, fact] = elimination_queue.pop(); + Node &node = get_node(fact); + if (node.in_degree == key && !node.is_eliminated) { + return fact; + } + } + return nullopt; + } + + void eliminate(FactPair fact) { + Node &node = get_node(fact); + /* + When eliminating the given fact from the graph, we add shorcut edges + from all its (non-eliminated) predecessors, to all its + (non-eliminated) successors. + */ + vector> new_shortcuts; + for (FactPair predecessor : node.predecessors) { + if (get_node(predecessor).is_eliminated) { + continue; + } + for (FactPair successor : node.successors) { + if (get_node(successor).is_eliminated) { + continue; + } + if (predecessor != successor) { + new_shortcuts.push_back(make_tuple(predecessor, fact, successor)); + } + } + } + node.is_eliminated = true; + + for (tuple shortcut : new_shortcuts) { + auto [from, _, to] = shortcut; + add_edge(from, to); + delta.insert(shortcut); + } + + /* + The elimination can affect the priority queue which uses the number of + incoming edges from non-eliminated nodes as a key. However, this can + only change for successors of 'fact'. We add them back into the queue + with updated keys and lazily filter out the outdated values. + */ + for (FactPair successor : node.successors) { + if (!get_node(successor).is_eliminated) { + push_fact(successor); + } + } + } + + void construct_task_graph(const TaskProxy &task_proxy) { + nodes.resize(task_proxy.get_variables().size()); + for (VariableProxy var : task_proxy.get_variables()) { + nodes[var.get_id()].resize(var.get_domain_size()); + } + for (OperatorProxy op : task_proxy.get_operators()) { + for (FactProxy pre_proxy : op.get_preconditions()) { + FactPair pre = pre_proxy.get_pair(); + for (EffectProxy eff_proxy : op.get_effects()) { + FactPair eff = eff_proxy.get_fact().get_pair(); + if (pre != eff) { + add_edge(pre, eff); + } + } + } + } + } + + void initialize_queue(const TaskProxy &task_proxy) { + for (VariableProxy var : task_proxy.get_variables()) { + int num_values = var.get_domain_size(); + for (int val = 0; val < num_values; ++val) { + push_fact(var.get_fact(val).get_pair()); + } + } + } + +public: + VEGraph(const TaskProxy &task_proxy) { + construct_task_graph(task_proxy); + initialize_queue(task_proxy); + while (optional fact = pop_fact()) { + eliminate(*fact); + } + } + + const utils::HashSet> &get_delta() const { + return delta; + } + + const utils::HashSet> &get_edges() const { + return edges; + } +}; + +int DeleteRelaxationRRConstraints::LPVariableIDs::id_of_fp(FactPair f) const { + return fp_offsets[f.var] + f.value; +} + +int DeleteRelaxationRRConstraints::LPVariableIDs::id_of_fpa( + FactPair f, const OperatorProxy &op) const { + return fpa_ids[op.get_id()].at(f); +} + +int DeleteRelaxationRRConstraints::LPVariableIDs::id_of_e( + pair edge) const { + return e_ids.at(edge); +} + +int DeleteRelaxationRRConstraints::LPVariableIDs::has_e( + pair edge) const { + return e_ids.find(edge) != e_ids.end(); +} + +int DeleteRelaxationRRConstraints::LPVariableIDs::id_of_t(FactPair f) const { + return t_offsets[f.var] + f.value; +} + +DeleteRelaxationRRConstraints::DeleteRelaxationRRConstraints( + const plugins::Options &opts) + : acyclicity_type(opts.get("acyclicity_type")), + use_integer_vars(opts.get("use_integer_vars")) { +} + +int DeleteRelaxationRRConstraints::get_constraint_id(FactPair f) const { + return constraint_offsets[f.var] + f.value; +} + +DeleteRelaxationRRConstraints::LPVariableIDs +DeleteRelaxationRRConstraints::create_auxiliary_variables( + const TaskProxy &task_proxy, LPVariables &variables) const { + OperatorsProxy ops = task_proxy.get_operators(); + VariablesProxy task_variables = task_proxy.get_variables(); + int num_vars = task_variables.size(); + LPVariableIDs lp_var_ids; + + // Add f_p variables. + lp_var_ids.fp_offsets.reserve(num_vars); + for (VariableProxy var : task_variables) { + lp_var_ids.fp_offsets.push_back(variables.size()); + int num_values = var.get_domain_size(); + for (int value = 0; value < num_values; ++value) { + variables.emplace_back(0, 1, 0, use_integer_vars); +#ifndef NDEBUG + variables.set_name(variables.size() - 1, + "f_" + var.get_name() + "_" + + var.get_fact(value).get_name()); +#endif + } + } + + // Add f_{p,a} variables. + lp_var_ids.fpa_ids.resize(ops.size()); + for (OperatorProxy op : ops) { + for (EffectProxy eff_proxy : op.get_effects()) { + FactPair eff = eff_proxy.get_fact().get_pair(); + lp_var_ids.fpa_ids[op.get_id()][eff] = variables.size(); + variables.emplace_back(0, 1, 0, use_integer_vars); +#ifndef NDEBUG + variables.set_name(variables.size() - 1, + "f_" + eff_proxy.get_fact().get_name() + + "_achieved_by_" + op.get_name()); +#endif + } + } + return lp_var_ids; +} + +void DeleteRelaxationRRConstraints::create_auxiliary_variables_ve( + const TaskProxy &task_proxy, const VEGraph &ve_graph, LPVariables &variables, + DeleteRelaxationRRConstraints::LPVariableIDs &lp_var_ids) const { + utils::unused_variable(task_proxy); + // Add e_{i,j} variables. + for (pair edge : ve_graph.get_edges()) { + lp_var_ids.e_ids[edge] = variables.size(); + variables.emplace_back(0, 1, 0, use_integer_vars); +#ifndef NDEBUG + auto [f1, f2] = edge; + FactProxy f1_proxy = task_proxy.get_variables()[f1.var].get_fact(f1.value); + FactProxy f2_proxy = task_proxy.get_variables()[f2.var].get_fact(f2.value); + variables.set_name(variables.size() - 1, + "e_" + f1_proxy.get_name() + + "_before_" + f2_proxy.get_name()); +#endif + } +} + +void DeleteRelaxationRRConstraints::create_auxiliary_variables_tl( + const TaskProxy &task_proxy, LPVariables &variables, + DeleteRelaxationRRConstraints::LPVariableIDs &lp_var_ids) const { + int num_facts = 0; + for (VariableProxy var : task_proxy.get_variables()) { + num_facts += var.get_domain_size(); + } + + lp_var_ids.t_offsets.resize(task_proxy.get_variables().size()); + for (VariableProxy var : task_proxy.get_variables()) { + lp_var_ids.t_offsets[var.get_id()] = variables.size(); + int num_values = var.get_domain_size(); + for (int value = 0; value < num_values; ++value) { + variables.emplace_back(1, num_facts, 0, use_integer_vars); +#ifndef NDEBUG + variables.set_name(variables.size() - 1, + "t_" + var.get_fact(value).get_name()); +#endif + } + } +} + +void DeleteRelaxationRRConstraints::create_constraints( + const TaskProxy &task_proxy, + const DeleteRelaxationRRConstraints::LPVariableIDs &lp_var_ids, + lp::LinearProgram &lp) { + LPVariables &variables = lp.get_variables(); + LPConstraints &constraints = lp.get_constraints(); + double infinity = lp.get_infinity(); + OperatorsProxy ops = task_proxy.get_operators(); + VariablesProxy vars = task_proxy.get_variables(); + + /* + Constraint (2) in paper: + + f_p = [p in s] + sum_{a in A where p in add(a)} f_{p,a} + for all facts p. + + Intuition: p is reached iff we selected exactly one achiever for it, or + if it is true in state s. + Implementation notes: we will set the state-dependent part ([p in s]) in + the update function and leave the right-hand side at 0 for now. The first + loop creates all constraints and adds the term "f_p", the second loop adds + the terms f_{p,a} to the appropriate constraints. + */ + constraint_offsets.reserve(vars.size()); + for (VariableProxy var_p : vars) { + int var_id_p = var_p.get_id(); + constraint_offsets.push_back(constraints.size()); + for (int value_p = 0; value_p < var_p.get_domain_size(); ++value_p) { + FactPair fact_p(var_id_p, value_p); + lp::LPConstraint constraint(0, 0); + constraint.insert(lp_var_ids.id_of_fp(fact_p), 1); + constraints.push_back(move(constraint)); + } + } + for (OperatorProxy op : ops) { + for (EffectProxy eff_proxy : op.get_effects()) { + FactPair eff = eff_proxy.get_fact().get_pair(); + lp::LPConstraint &constraint = constraints[get_constraint_id(eff)]; + constraint.insert(lp_var_ids.id_of_fpa(eff, op), -1); + } + } + + /* + Constraint (3) in paper: + + sum_{a in A where q in pre(a) and p in add(a)} f_{p,a} <= f_q + for all facts p, q. + + Intuition: If q is the precondition of an action that is selected as an + achiever for p, then q must be reached. (Also, at most one action may be + selected as the achiever of p.) + Implementation notes: if there is no action in the sum for a pair (p, q), + the constraint trivializes to 0 <= f_q which is guaranteed by the variable + bounds. We thus only loop over pairs (p, q) that occur as effect and + precondition in some action. + */ + utils::HashMap, int> constraint3_ids; + for (OperatorProxy op : ops) { + for (EffectProxy eff_proxy : op.get_effects()) { + FactPair eff = eff_proxy.get_fact().get_pair(); + for (FactProxy pre_proxy : op.get_preconditions()) { + FactPair pre = pre_proxy.get_pair(); + if (pre == eff) { + continue; + } + pair key = make_pair(pre, eff); + if (!constraint3_ids.contains(key)) { + constraint3_ids[key] = constraints.size(); + lp::LPConstraint constraint(0, 1); + constraint.insert(lp_var_ids.id_of_fp(pre), 1); + constraints.push_back(move(constraint)); + } + int constraint_id = constraint3_ids[key]; + lp::LPConstraint &constraint = constraints[constraint_id]; + constraint.insert(lp_var_ids.id_of_fpa(eff, op), -1); + } + } + } + + /* + Constraint (4) in paper: + + f_p = 1 for all goal facts p. + + Intuition: We have to reach all goal facts. + Implementation notes: we don't add a constraint but instead raise the + lower bound of the (binary) variable to 1. A further optimization step + would be to replace all occurrences of f_p with 1 in all other constraints + but this would be more complicated. + */ + for (FactProxy goal : task_proxy.get_goals()) { + variables[lp_var_ids.id_of_fp(goal.get_pair())].lower_bound = 1; + } + + /* + Constraint (5) in paper: + + f_{p,a} <= count_a for all a in A and p in add(a). + + Intuition: if we use an action as an achiever for some fact, we have to + use it at least once. + Implementation notes: the paper uses a binary variable f_a instead of the + operator-counting variable count_a. We can make this change without + problems as f_a does not occur in any other constraint. + */ + for (OperatorProxy op : ops) { + for (EffectProxy eff_proxy : op.get_effects()) { + FactPair eff = eff_proxy.get_fact().get_pair(); + lp::LPConstraint constraint(0, infinity); + constraint.insert(lp_var_ids.id_of_fpa(eff, op), -1); + constraint.insert(op.get_id(), 1); + constraints.push_back(move(constraint)); + } + } +} + +void DeleteRelaxationRRConstraints::create_constraints_ve( + const TaskProxy &task_proxy, const VEGraph &ve_graph, + const DeleteRelaxationRRConstraints::LPVariableIDs &lp_var_ids, + lp::LinearProgram &lp) { + LPConstraints &constraints = lp.get_constraints(); + double infinity = lp.get_infinity(); + OperatorsProxy ops = task_proxy.get_operators(); + + /* + Constraint (6) in paper: + + f_{p_j,a} <= e_{i,j} for all a in A, p_i in pre(a), and p_j in add(a). + + Intuition: if we use a as the achiever of p_j, then its preconditions (in + particular p_i) must be achieved earlier than p_j. + */ + for (OperatorProxy op : ops) { + for (FactProxy pre_proxy : op.get_preconditions()) { + FactPair pre = pre_proxy.get_pair(); + for (EffectProxy eff_proxy : op.get_effects()) { + FactPair eff = eff_proxy.get_fact().get_pair(); + lp::LPConstraint constraint(0, infinity); + constraint.insert(lp_var_ids.id_of_e(make_pair(pre, eff)), 1); + constraint.insert(lp_var_ids.id_of_fpa(eff, op), -1); + constraints.push_back(move(constraint)); + } + } + } + + /* + Constraint (7) in paper: + + e_{i,j} + e_{j,i} <= 1 for all (p_i, p_j) in E_Pi^*. + + Intuition: if there is a 2-cycle in the elimination graph, we have to + avoid it by either ordering i before j or vice versa. + Implementation note: the paper is not explicit about this but the + constraint only makes sense if the reverse edge is in the graph. + */ + for (const pair &edge : ve_graph.get_edges()) { + pair reverse_edge = make_pair(edge.second, edge.first); + if (lp_var_ids.has_e(reverse_edge)) { + lp::LPConstraint constraint(-infinity, 1); + constraint.insert(lp_var_ids.id_of_e(edge), 1); + constraint.insert(lp_var_ids.id_of_e(reverse_edge), 1); + constraints.push_back(move(constraint)); + } + } + + /* + Constraint (8) in paper: + + e_{i,j} + e_{j,k} - 1 <= e_{i,k} for all (p_i, p_j, p_k) in Delta. + + Intuition: if we introduced shortcut edge (p_i, p_k) while eliminating p_j + cycles involving the new edge represents cycles containing the edges + (p_i, p_j) and (p_j, p_k). If we don't order p_i before p_k, we also may + not have both p_i ordered before p_j, and p_j ordered before p_k. + */ + for (auto [pi, pj, pk] : ve_graph.get_delta()) { + lp::LPConstraint constraint(-infinity, 1); + constraint.insert(lp_var_ids.id_of_e(make_pair(pi, pj)), 1); + constraint.insert(lp_var_ids.id_of_e(make_pair(pj, pk)), 1); + constraint.insert(lp_var_ids.id_of_e(make_pair(pi, pk)), -1); + constraints.push_back(move(constraint)); + } +} + +void DeleteRelaxationRRConstraints::create_constraints_tl( + const TaskProxy &task_proxy, + const DeleteRelaxationRRConstraints::LPVariableIDs &lp_var_ids, + lp::LinearProgram &lp) { + /* + Constraint (9) in paper: + + t_i - t_j + 1 <= |P|(1 - f_{p_j, a}) + for all a in A, p_i in pre(a) and p_j in add(a) + Equivalent form: + t_i - t_j + |P|f_{p_j, a} <= |P| - 1 + + Intuition: if a is used to achieve p_j and p_i is one of a's + preconditions, we have to achieve p_i before p_j. + */ + LPConstraints &constraints = lp.get_constraints(); + double infinity = lp.get_infinity(); + int num_facts = 0; + for (VariableProxy var : task_proxy.get_variables()) { + num_facts += var.get_domain_size(); + } + + for (OperatorProxy op : task_proxy.get_operators()) { + for (FactProxy pre_proxy : op.get_preconditions()) { + FactPair pre = pre_proxy.get_pair(); + for (EffectProxy eff_proxy : op.get_effects()) { + FactPair eff = eff_proxy.get_fact().get_pair(); + if (pre == eff) { + // Prevail conditions are compiled away in the paper. + continue; + } + lp::LPConstraint constraint(-infinity, num_facts - 1); + constraint.insert(lp_var_ids.id_of_t(pre), 1); + constraint.insert(lp_var_ids.id_of_t(eff), -1); + constraint.insert(lp_var_ids.id_of_fpa(eff, op), num_facts); + constraints.push_back(move(constraint)); + } + } + } +} + +void DeleteRelaxationRRConstraints::initialize_constraints( + const shared_ptr &task, lp::LinearProgram &lp) { + TaskProxy task_proxy(*task); + LPVariableIDs lp_var_ids = create_auxiliary_variables( + task_proxy, lp.get_variables()); + create_constraints(task_proxy, lp_var_ids, lp); + + switch (acyclicity_type) { + case AcyclicityType::VERTEX_ELIMINATION: + { + VEGraph ve_graph(task_proxy); + create_auxiliary_variables_ve( + task_proxy, ve_graph, lp.get_variables(), lp_var_ids); + create_constraints_ve(task_proxy, ve_graph, lp_var_ids, lp); + break; + } + case AcyclicityType::TIME_LABELS: + { + create_auxiliary_variables_tl( + task_proxy, lp.get_variables(), lp_var_ids); + create_constraints_tl(task_proxy, lp_var_ids, lp); + break; + } + case AcyclicityType::NONE: + { + break; + } + default: + ABORT("Unknown AcyclicityType"); + } +} + +bool DeleteRelaxationRRConstraints::update_constraints( + const State &state, lp::LPSolver &lp_solver) { + // Unset old bounds. + int con_id; + for (FactPair f : last_state) { + con_id = get_constraint_id(f); + lp_solver.set_constraint_lower_bound(con_id, 0); + lp_solver.set_constraint_upper_bound(con_id, 0); + } + last_state.clear(); + // Set new bounds. + for (FactProxy f : state) { + con_id = get_constraint_id(f.get_pair()); + lp_solver.set_constraint_lower_bound(con_id, 1); + lp_solver.set_constraint_upper_bound(con_id, 1); + last_state.push_back(f.get_pair()); + } + return false; +} + +class DeleteRelaxationRRConstraintsFeature + : public plugins::TypedFeature { +public: + DeleteRelaxationRRConstraintsFeature() + : TypedFeature("delete_relaxation_rr_constraints") { + document_title( + "Delete relaxation constraints from Rankooh and Rintanen"); + document_synopsis( + "Operator-counting constraints based on the delete relaxation. By " + "default the constraints encode an easy-to-compute relaxation of " + "h^+^. " + "With the right settings, these constraints can be used to compute " + "the " + "optimal delete-relaxation heuristic h^+^ (see example below). " + "For details, see" + + utils::format_journal_reference( + {"Masood Feyzbakhsh Rankooh", "Jussi Rintanen"}, + "Efficient Computation and Informative Estimation of" + "h+ by Integer and Linear Programming" + "", + "https://ojs.aaai.org/index.php/ICAPS/article/view/19787/19546", + "Proceedings of the Thirty-Second International Conference on " + "Automated Planning and Scheduling (ICAPS2022)", + "32", "71-79", "2022")); + + add_option( + "acyclicity_type", + "The most relaxed version of this constraint only enforces that " + "achievers of facts are picked in such a way that all goal facts " + "have an achiever, and the preconditions all achievers are either " + "true in the current state or have achievers themselves. In this " + "version, cycles in the achiever relation can occur. Such cycles " + "can be excluded with additional auxilliary varibles and " + "constraints.", + "vertex_elimination"); + add_option( + "use_integer_vars", + "restrict auxiliary variables to integer values. These variables " + "encode whether facts are reached, which operator first achieves " + "which fact, and (depending on the acyclicity_type) in which order " + "the operators are used. Restricting them to integers generally " + "improves the heuristic value at the cost of increased runtime.", + "false"); + + document_note( + "Example", + "To compute the optimal delete-relaxation heuristic h^+^, use" + "integer variables and some way of enforcing acyclicity (other " + "than \"none\"). For example\n" + "{{{\noperatorcounting([delete_relaxation_rr_constraints(" + "acyclicity_type=vertex_elimination, use_integer_vars=true)], " + "use_integer_operator_counts=true))\n}}}\n"); + document_note( + "Note", + "While the delete-relaxation constraints by Imai and Fukunaga " + "(accessible via option {{{delete_relaxation_if_constraints}}}) " + "serve a similar purpose to the constraints implemented here, we " + "recommend using this formulation as it can generally be solved " + "more efficiently, in particular in case of the h^+^ " + "configuration, and some relaxations offer tighter bounds.\n"); + } +}; + +static plugins::FeaturePlugin _plugin; + +static plugins::TypedEnumPlugin _enum_plugin({ + {"time_labels", + "introduces MIP variables that encode the time at which each fact is " + "reached. Acyclicity is enforced with constraints that ensure that " + "preconditions of actions are reached before their effects."}, + {"vertex_elimination", + "introduces binary variables based on vertex elimination. These " + "variables encode that one fact has to be reached before another " + "fact. Instead of adding such variables for every pair of states, " + "they are only added for a subset sufficient to ensure acyclicity. " + "Constraints enforce that preconditions of actions are reached before " + "their effects and that the assignment encodes a valid order."}, + {"none", + "No acyclicity is enforced. The resulting heuristic is a relaxation " + "of the delete-relaxation heuristic."} + }); +} diff --git a/src/search/operator_counting/delete_relaxation_rr_constraints.h b/src/search/operator_counting/delete_relaxation_rr_constraints.h new file mode 100644 index 0000000000..cb43921cd9 --- /dev/null +++ b/src/search/operator_counting/delete_relaxation_rr_constraints.h @@ -0,0 +1,116 @@ +#ifndef OPERATOR_COUNTING_DELETE_RELAXATION_RR_CONSTRAINTS_H +#define OPERATOR_COUNTING_DELETE_RELAXATION_RR_CONSTRAINTS_H + +#include "constraint_generator.h" + +#include "../task_proxy.h" +#include "../utils/hash.h" + +#include +#include + +namespace lp { +class LPConstraint; +struct LPVariable; +} + +namespace plugins { +class Options; +} + +namespace operator_counting { +class VEGraph; +using LPConstraints = named_vector::NamedVector; +using LPVariables = named_vector::NamedVector; + +enum class AcyclicityType { + TIME_LABELS, VERTEX_ELIMINATION, NONE +}; + +class DeleteRelaxationRRConstraints : public ConstraintGenerator { + struct LPVariableIDs { + /* + The variables f_p in the paper represent if a fact p is reached by the + relaxed plan encoded in the LP solution. We only store the offset for + each variable (LP variables for different facts of the same variable + are consecutive). + */ + std::vector fp_offsets; + + /* + The variables f_{p,a} in the paper represent if an action a is used to + achieve a fact p in the relaxed plan encoded in the LP solution. The + variable is only needed for combinations of p and a where p is an + effect of a. We store one hash map for each operator a that maps facts + p to LP variable IDs. + */ + std::vector> fpa_ids; + + /* + The variable e_{i,j} in the paper is used as part of the vertex + elimination method. It represents that fact p_i is used before fact + p_j. Not all pairs of facts have to be ordered, the vertex elimination + graph ensures that enough variables are created to exclude all cycles. + */ + utils::HashMap, int> e_ids; + + /* + The variable t_p in the paper is used as part of the time labels + method. It represents the time at which fact p is first made true. + We store the offsets here, analogous to fp. + */ + std::vector t_offsets; + + int id_of_fp(FactPair f) const; + int id_of_fpa(FactPair f, const OperatorProxy &op) const; + int id_of_e(std::pair edge) const; + int has_e(std::pair edge) const; + int id_of_t(FactPair f) const; + }; + + AcyclicityType acyclicity_type; + bool use_integer_vars; + + /* + Store offsets to identify Constraints (2) in the paper. We need to + reference them when updating constraints for a given state. The constraint + for a fact with variable v and value d has ID (constraint_offsets[v] + d). + */ + std::vector constraint_offsets; + + /* The state that is currently used for setting the bounds. Remembering + this makes it faster to unset the bounds when the state changes. */ + std::vector last_state; + + + int get_constraint_id(FactPair f) const; + + LPVariableIDs create_auxiliary_variables( + const TaskProxy &task_proxy, LPVariables &variables) const; + void create_auxiliary_variables_ve( + const TaskProxy &task_proxy, const VEGraph &ve_graph, + LPVariables &variables, LPVariableIDs &lp_var_ids) const; + void create_auxiliary_variables_tl( + const TaskProxy &task_proxy, LPVariables &variables, + LPVariableIDs &lp_var_ids) const; + void create_constraints( + const TaskProxy &task_proxy, const LPVariableIDs &lp_var_ids, + lp::LinearProgram &lp); + void create_constraints_ve( + const TaskProxy &task_proxy, const VEGraph &ve_graph, + const LPVariableIDs &lp_var_ids, lp::LinearProgram &lp); + void create_constraints_tl( + const TaskProxy &task_proxy, const LPVariableIDs &lp_var_ids, + lp::LinearProgram &lp); +public: + explicit DeleteRelaxationRRConstraints(const plugins::Options &opts); + + virtual void initialize_constraints( + const std::shared_ptr &task, + lp::LinearProgram &lp) override; + virtual bool update_constraints( + const State &state, lp::LPSolver &lp_solver) override; +}; +} + +#endif diff --git a/src/search/operator_counting/operator_counting_heuristic.cc b/src/search/operator_counting/operator_counting_heuristic.cc index 05fa85d698..a844b377bb 100644 --- a/src/search/operator_counting/operator_counting_heuristic.cc +++ b/src/search/operator_counting/operator_counting_heuristic.cc @@ -4,6 +4,7 @@ #include "../plugins/plugin.h" #include "../utils/markup.h" +#include "../utils/strings.h" #include @@ -22,6 +23,9 @@ OperatorCountingHeuristic::OperatorCountingHeuristic(const plugins::Options &opt for (OperatorProxy op : task_proxy.get_operators()) { int op_cost = op.get_cost(); variables.push_back(lp::LPVariable(0, infinity, op_cost, use_integer_operator_counts)); +#ifndef NDEBUG + variables.set_name(op.get_id(), op.get_name()); +#endif } lp::LinearProgram lp(lp::LPObjectiveSense::MINIMIZE, move(variables), {}, infinity); for (const auto &generator : constraint_generators) { diff --git a/src/search/utils/hash.h b/src/search/utils/hash.h index db13437ad3..bc755d7558 100644 --- a/src/search/utils/hash.h +++ b/src/search/utils/hash.h @@ -26,7 +26,7 @@ namespace utils { that are "fed" to the main hashing function (implemented in class HashState) one by one. This allows a compositional approach to hashing. For example, the code for a pair p is the concatenation of - code(x.first) and code(x.second). + code(p.first) and code(p.second). A simpler compositional approach to hashing would first hash the components of an object and then combine the hash values, and this @@ -258,6 +258,11 @@ void feed(HashState &hash_state, const std::vector &vec) { } } +template +void feed(HashState &hash_state, const std::tuple &t) { + std::apply([&](auto &&... element) {((feed(hash_state, element)), ...);}, t); +} + /* Public hash functions.