//===-- HeuristcBase.h --- Heuristic base class for PBQP --------*- C++ -*-===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #ifndef LLVM_CODEGEN_PBQP_HEURISTICBASE_H #define LLVM_CODEGEN_PBQP_HEURISTICBASE_H #include "HeuristicSolver.h" namespace PBQP { /// \brief Abstract base class for heuristic implementations. /// /// This class provides a handy base for heuristic implementations with common /// solver behaviour implemented for a number of methods. /// /// To implement your own heuristic using this class as a base you'll have to /// implement, as a minimum, the following methods: /// <ul> /// <li> void addToHeuristicList(Graph::NodeItr) : Add a node to the /// heuristic reduction list. /// <li> void heuristicReduce() : Perform a single heuristic reduction. /// <li> void preUpdateEdgeCosts(Graph::EdgeItr) : Handle the (imminent) /// change to the cost matrix on the given edge (by R2). /// <li> void postUpdateEdgeCostts(Graph::EdgeItr) : Handle the new /// costs on the given edge. /// <li> void handleAddEdge(Graph::EdgeItr) : Handle the addition of a new /// edge into the PBQP graph (by R2). /// <li> void handleRemoveEdge(Graph::EdgeItr, Graph::NodeItr) : Handle the /// disconnection of the given edge from the given node. /// <li> A constructor for your derived class : to pass back a reference to /// the solver which is using this heuristic. /// </ul> /// /// These methods are implemented in this class for documentation purposes, /// but will assert if called. /// /// Note that this class uses the curiously recursive template idiom to /// forward calls to the derived class. These methods need not be made /// virtual, and indeed probably shouldn't for performance reasons. /// /// You'll also need to provide NodeData and EdgeData structs in your class. /// These can be used to attach data relevant to your heuristic to each /// node/edge in the PBQP graph. template <typename HImpl> class HeuristicBase { private: typedef std::list<Graph::NodeItr> OptimalList; HeuristicSolverImpl<HImpl> &s; Graph &g; OptimalList optimalList; // Return a reference to the derived heuristic. HImpl& impl() { return static_cast<HImpl&>(*this); } // Add the given node to the optimal reductions list. Keep an iterator to // its location for fast removal. void addToOptimalReductionList(Graph::NodeItr nItr) { optimalList.insert(optimalList.end(), nItr); } public: /// \brief Construct an instance with a reference to the given solver. /// @param solver The solver which is using this heuristic instance. HeuristicBase(HeuristicSolverImpl<HImpl> &solver) : s(solver), g(s.getGraph()) { } /// \brief Get the solver which is using this heuristic instance. /// @return The solver which is using this heuristic instance. /// /// You can use this method to get access to the solver in your derived /// heuristic implementation. HeuristicSolverImpl<HImpl>& getSolver() { return s; } /// \brief Get the graph representing the problem to be solved. /// @return The graph representing the problem to be solved. Graph& getGraph() { return g; } /// \brief Tell the solver to simplify the graph before the reduction phase. /// @return Whether or not the solver should run a simplification phase /// prior to the main setup and reduction. /// /// HeuristicBase returns true from this method as it's a sensible default, /// however you can over-ride it in your derived class if you want different /// behaviour. bool solverRunSimplify() const { return true; } /// \brief Decide whether a node should be optimally or heuristically /// reduced. /// @return Whether or not the given node should be listed for optimal /// reduction (via R0, R1 or R2). /// /// HeuristicBase returns true for any node with degree less than 3. This is /// sane and sensible for many situations, but not all. You can over-ride /// this method in your derived class if you want a different selection /// criteria. Note however that your criteria for selecting optimal nodes /// should be <i>at least</i> as strong as this. I.e. Nodes of degree 3 or /// higher should not be selected under any circumstances. bool shouldOptimallyReduce(Graph::NodeItr nItr) { if (g.getNodeDegree(nItr) < 3) return true; // else return false; } /// \brief Add the given node to the list of nodes to be optimally reduced. /// @return nItr Node iterator to be added. /// /// You probably don't want to over-ride this, except perhaps to record /// statistics before calling this implementation. HeuristicBase relies on /// its behaviour. void addToOptimalReduceList(Graph::NodeItr nItr) { optimalList.push_back(nItr); } /// \brief Initialise the heuristic. /// /// HeuristicBase iterates over all nodes in the problem and adds them to /// the appropriate list using addToOptimalReduceList or /// addToHeuristicReduceList based on the result of shouldOptimallyReduce. /// /// This behaviour should be fine for most situations. void setup() { for (Graph::NodeItr nItr = g.nodesBegin(), nEnd = g.nodesEnd(); nItr != nEnd; ++nItr) { if (impl().shouldOptimallyReduce(nItr)) { addToOptimalReduceList(nItr); } else { impl().addToHeuristicReduceList(nItr); } } } /// \brief Optimally reduce one of the nodes in the optimal reduce list. /// @return True if a reduction takes place, false if the optimal reduce /// list is empty. /// /// Selects a node from the optimal reduce list and removes it, applying /// R0, R1 or R2 as appropriate based on the selected node's degree. bool optimalReduce() { if (optimalList.empty()) return false; Graph::NodeItr nItr = optimalList.front(); optimalList.pop_front(); switch (s.getSolverDegree(nItr)) { case 0: s.applyR0(nItr); break; case 1: s.applyR1(nItr); break; case 2: s.applyR2(nItr); break; default: assert(false && "Optimal reductions of degree > 2 nodes is invalid."); } return true; } /// \brief Perform the PBQP reduction process. /// /// Reduces the problem to the empty graph by repeated application of the /// reduction rules R0, R1, R2 and RN. /// R0, R1 or R2 are always applied if possible before RN is used. void reduce() { bool finished = false; while (!finished) { if (!optimalReduce()) { if (impl().heuristicReduce()) { getSolver().recordRN(); } else { finished = true; } } } } /// \brief Add a node to the heuristic reduce list. /// @param nItr Node iterator to add to the heuristic reduce list. void addToHeuristicList(Graph::NodeItr nItr) { assert(false && "Must be implemented in derived class."); } /// \brief Heuristically reduce one of the nodes in the heuristic /// reduce list. /// @return True if a reduction takes place, false if the heuristic reduce /// list is empty. void heuristicReduce() { assert(false && "Must be implemented in derived class."); } /// \brief Prepare a change in the costs on the given edge. /// @param eItr Edge iterator. void preUpdateEdgeCosts(Graph::EdgeItr eItr) { assert(false && "Must be implemented in derived class."); } /// \brief Handle the change in the costs on the given edge. /// @param eItr Edge iterator. void postUpdateEdgeCostts(Graph::EdgeItr eItr) { assert(false && "Must be implemented in derived class."); } /// \brief Handle the addition of a new edge into the PBQP graph. /// @param eItr Edge iterator for the added edge. void handleAddEdge(Graph::EdgeItr eItr) { assert(false && "Must be implemented in derived class."); } /// \brief Handle disconnection of an edge from a node. /// @param eItr Edge iterator for edge being disconnected. /// @param nItr Node iterator for the node being disconnected from. /// /// Edges are frequently removed due to the removal of a node. This /// method allows for the effect to be computed only for the remaining /// node in the graph. void handleRemoveEdge(Graph::EdgeItr eItr, Graph::NodeItr nItr) { assert(false && "Must be implemented in derived class."); } /// \brief Clean up any structures used by HeuristicBase. /// /// At present this just performs a sanity check: that the optimal reduce /// list is empty now that reduction has completed. /// /// If your derived class has more complex structures which need tearing /// down you should over-ride this method but include a call back to this /// implementation. void cleanup() { assert(optimalList.empty() && "Nodes left over in optimal reduce list?"); } }; } #endif // LLVM_CODEGEN_PBQP_HEURISTICBASE_H