1#ifndef ZONOOPT_BRANCH_AND_BOUND_
2#define ZONOOPT_BRANCH_AND_BOUND_
18#include <condition_variable>
22#include <memory_resource>
29namespace ZonoOpt::detail
34 explicit BranchAndBound(
const MI_data& data);
40 std::pair<std::vector<OptSolution>, OptSolution> multi_solve(
int max_sols = std::numeric_limits<int>::max());
43 void warmstart(
const Eigen::Vector<zono_float, -1>& xi_ws,
const Eigen::Vector<zono_float, -1>& u_ws)
52 std::pmr::synchronized_pool_resource* pool_ptr;
54 explicit NodeDeleter(std::pmr::synchronized_pool_resource* pool_ptr) : pool_ptr(pool_ptr)
58 void operator()(Node* node)
const
63 pool_ptr->deallocate(node,
sizeof(Node),
alignof(Node));
70 bool operator()(
const std::unique_ptr<Node, NodeDeleter>& n1,
71 const std::unique_ptr<Node, NodeDeleter>& n2)
const
73 if (n2->is_priority() && !n1->is_priority())
75 else if (!n2->is_priority() && n1->is_priority())
78 return n1->solution.J > n2->solution.J;
84 ThreadSafeMultiset& J_threads;
86 bool specified =
false;
88 explicit JThreadGuard(ThreadSafeMultiset& J_threads): J_threads(J_threads)
97 this->specified =
true;
98 this->J_threads.add(J);
105 this->J_threads.remove(J);
110 std::pmr::synchronized_pool_resource pool;
113 PriorityQueuePrunable<std::unique_ptr<Node, NodeDeleter>, NodeCompare> node_queue;
114 mutable std::mutex pq_mtx;
115 std::condition_variable pq_cv_bnb, pq_cv_admm_fp;
118 bool multi_sol =
false;
119 std::shared_ptr<ADMM_data> bnb_data, admm_fp_data;
121 std::atomic<bool> converged =
false;
122 std::atomic<bool> done =
false;
123 std::atomic<bool> feasible =
false;
124 std::atomic<bool> admm_fp_incumbent =
false;
125 std::atomic<long int> qp_iter = 0;
126 std::atomic<int> iter = 0;
127 std::atomic<int> iter_admm_fp = 0;
128 std::atomic<zono_float> J_max = std::numeric_limits<zono_float>::infinity();
129 ThreadSafeAccess<Eigen::Vector<
zono_float, -1>> z, x, u;
130 std::atomic<zono_float> primal_residual = std::numeric_limits<zono_float>::infinity();
131 std::atomic<zono_float> dual_residual = std::numeric_limits<zono_float>::infinity();
132 ThreadSafeIncrementable<double> total_startup_time{0.0};
133 ThreadSafeIncrementable<double> total_run_time{0.0};
134 ThreadSafeMultiset J_threads;
135 ThreadSafeVector<OptSolution> solutions;
141 std::unique_ptr<Node, NodeDeleter> make_node(
const std::shared_ptr<ADMM_data>& admm_data);
143 std::unique_ptr<Node, NodeDeleter> clone_node(
const std::unique_ptr<Node, NodeDeleter>& other);
146 std::variant<OptSolution, std::pair<std::vector<OptSolution>, OptSolution>> solver_core(
147 int max_sols = std::numeric_limits<int>::max());
150 void solve_and_branch(
const std::unique_ptr<Node, NodeDeleter>& node);
152 void admm_fp_solve(
const std::unique_ptr<ADMM_FP_solver>& node);
155 bool is_integer_feasible(
const Eigen::Ref<
const Eigen::Vector<zono_float, -1>> xb)
const;
158 void branch_most_frac(
const std::unique_ptr<Node, NodeDeleter>& node);
163 void admm_fp_loop(std::unique_ptr<ADMM_FP_solver>&& node);
166 void push_node(std::unique_ptr<Node, NodeDeleter>&& node);
172 bool check_bin_equal(
const OptSolution& sol1,
const OptSolution& sol2)
const;
Convex and mixed-integer ADMM implementations used within ZonoOpt.
Data structures for mixed-integer optimization in ZonoOpt library.
Optimization settings and solution data structures for ZonoOpt library.
#define zono_float
Defines the floating-point type used in ZonoOpt.
Definition ZonoOpt.hpp:45