| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449 | ////=======================================================================// Copyright 2002 Marc Wintermantel (wintermantel@even-ag.ch)// ETH Zurich, Center of Structure Technologies (www.imes.ethz.ch/st)//// Distributed under the Boost Software License, Version 1.0. (See// accompanying file LICENSE_1_0.txt or copy at// http://www.boost.org/LICENSE_1_0.txt)//=======================================================================//#ifndef BOOST_GRAPH_SLOAN_HPP#define BOOST_GRAPH_SLOAN_HPP#define WEIGHT1 1               //default weight for the distance in the Sloan algorithm#define WEIGHT2 2               //default weight for the degree in the Sloan algorithm#include <boost/config.hpp>#include <vector>#include <queue>#include <algorithm>#include <limits>#include <boost/pending/queue.hpp>#include <boost/graph/graph_traits.hpp>#include <boost/graph/breadth_first_search.hpp>#include <boost/graph/properties.hpp>#include <boost/pending/indirect_cmp.hpp>#include <boost/property_map/property_map.hpp>#include <boost/graph/visitors.hpp>#include <boost/graph/adjacency_list.hpp>#include <boost/graph/cuthill_mckee_ordering.hpp>////////////////////////////////////////////////////////////////Sloan-Algorithm for graph reordering//(optimzes profile and wavefront, not primiraly bandwidth//////////////////////////////////////////////////////////////namespace boost {          /////////////////////////////////////////////////////////////////////////  // Function that returns the maximum depth of   // a rooted level strucutre (RLS)  //  /////////////////////////////////////////////////////////////////////////  template<class Distance>  typename Distance::value_type RLS_depth(Distance& d)  {    typename Distance::value_type h_s = 0;    typename Distance::iterator iter;        for (iter = d.begin(); iter != d.end(); ++iter)      {        if(*iter > h_s)          {            h_s = *iter;          }      }        return h_s;  }      /////////////////////////////////////////////////////////////////////////  // Function that returns the width of the largest level of   // a rooted level strucutre (RLS)  //  /////////////////////////////////////////////////////////////////////////  template<class Distance, class my_int>  typename Distance::value_type RLS_max_width(Distance& d, my_int depth)  {      typedef typename Distance::value_type Degree;          //Searching for the maximum width of a level      std::vector<Degree> dummy_width(depth+1, 0);      typename std::vector<Degree>::iterator my_it;      typename Distance::iterator iter;      Degree w_max = 0;            for (iter = d.begin(); iter != d.end(); ++iter)      {          dummy_width[*iter]++;      }            for(my_it = dummy_width.begin(); my_it != dummy_width.end(); ++my_it)      {          if(*my_it > w_max) w_max = *my_it;      }            return w_max;        }      /////////////////////////////////////////////////////////////////////////  // Function for finding a good starting node for Sloan algorithm  //  // This is to find a good starting node. "good" is in the sense  // of the ordering generated.   /////////////////////////////////////////////////////////////////////////  template <class Graph, class ColorMap, class DegreeMap>   typename graph_traits<Graph>::vertex_descriptor   sloan_start_end_vertices(Graph& G,                            typename graph_traits<Graph>::vertex_descriptor &s,                            ColorMap color,                            DegreeMap degree)  {    typedef typename property_traits<DegreeMap>::value_type Degree;    typedef typename graph_traits<Graph>::vertex_descriptor Vertex;    typedef typename std::vector< typename graph_traits<Graph>::vertices_size_type>::iterator vec_iter;    typedef typename graph_traits<Graph>::vertices_size_type size_type;        typedef typename property_map<Graph, vertex_index_t>::const_type VertexID;        s = *(vertices(G).first);    Vertex e = s;    Vertex i;    Degree my_degree = get(degree, s );     Degree dummy, h_i, h_s, w_i, w_e;    bool new_start = true;    Degree maximum_degree = 0;        //Creating a std-vector for storing the distance from the start vertex in dist    std::vector<typename graph_traits<Graph>::vertices_size_type> dist(num_vertices(G), 0);    //Wrap a property_map_iterator around the std::iterator    boost::iterator_property_map<vec_iter, VertexID, size_type, size_type&> dist_pmap(dist.begin(), get(vertex_index, G));        //Creating a property_map for the indices of a vertex    typename property_map<Graph, vertex_index_t>::type index_map = get(vertex_index, G);        //Creating a priority queue    typedef indirect_cmp<DegreeMap, std::greater<Degree> > Compare;    Compare comp(degree);    std::priority_queue<Vertex, std::vector<Vertex>, Compare> degree_queue(comp);        //step 1    //Scan for the vertex with the smallest degree and the maximum degree    typename graph_traits<Graph>::vertex_iterator ui, ui_end;    for (boost::tie(ui, ui_end) = vertices(G); ui != ui_end; ++ui)    {      dummy = get(degree, *ui);            if(dummy < my_degree)      {        my_degree = dummy;        s = *ui;      }            if(dummy > maximum_degree)      {        maximum_degree = dummy;      }    }    //end 1        do{        new_start = false;     //Setting the loop repetition status to false            //step 2      //initialize the the disance std-vector with 0      for(typename std::vector<typename graph_traits<Graph>::vertices_size_type>::iterator iter = dist.begin(); iter != dist.end(); ++iter) *iter = 0;            //generating the RLS (rooted level structure)      breadth_first_search        (G, s, visitor         (           make_bfs_visitor(record_distances(dist_pmap, on_tree_edge() ) )           )          );            //end 2            //step 3      //calculating the depth of the RLS      h_s = RLS_depth(dist);            //step 4      //pushing one node of each degree in an ascending manner into degree_queue      std::vector<bool> shrink_trace(maximum_degree, false);      for (boost::tie(ui, ui_end) = vertices(G); ui != ui_end; ++ui)      {        dummy = get(degree, *ui);                if( (dist[index_map[*ui]] == h_s ) && ( !shrink_trace[ dummy ] ) )        {          degree_queue.push(*ui);          shrink_trace[ dummy ] = true;        }      }            //end 3 & 4            // step 5      // Initializing w      w_e = (std::numeric_limits<Degree>::max)();      //end 5                  //step 6      //Testing for termination      while( !degree_queue.empty() )      {        i = degree_queue.top();       //getting the node with the lowest degree from the degree queue        degree_queue.pop();           //ereasing the node with the lowest degree from the degree queue                //generating a RLS                  for(typename std::vector<typename graph_traits<Graph>::vertices_size_type>::iterator iter = dist.begin(); iter != dist.end(); ++iter) *iter = 0;                breadth_first_search          (G, i, boost::visitor           (             make_bfs_visitor(record_distances(dist_pmap, on_tree_edge() ) )             )            );                //Calculating depth and width of the rooted level        h_i = RLS_depth(dist);        w_i = RLS_max_width(dist, h_i);                //Testing for termination        if( (h_i > h_s) && (w_i < w_e) )         {          h_s = h_i;          s = i;          while(!degree_queue.empty()) degree_queue.pop();          new_start = true;                 }        else if(w_i < w_e)        {           w_e = w_i;          e = i;        }      }      //end 6            }while(new_start);        return e;  }  //////////////////////////////////////////////////////////////////////////  // Sloan algorithm with a given starting Vertex.  //  // This algorithm requires user to provide a starting vertex to  // compute Sloan ordering.  //////////////////////////////////////////////////////////////////////////  template <class Graph, class OutputIterator,            class ColorMap, class DegreeMap,            class PriorityMap, class Weight>  OutputIterator  sloan_ordering(Graph& g,                 typename graph_traits<Graph>::vertex_descriptor s,                 typename graph_traits<Graph>::vertex_descriptor e,                 OutputIterator permutation,                  ColorMap color,                  DegreeMap degree,                  PriorityMap priority,                  Weight W1,                  Weight W2)  {    //typedef typename property_traits<DegreeMap>::value_type Degree;    typedef typename property_traits<PriorityMap>::value_type Degree;    typedef typename property_traits<ColorMap>::value_type ColorValue;    typedef color_traits<ColorValue> Color;    typedef typename graph_traits<Graph>::vertex_descriptor Vertex;    typedef typename std::vector<typename graph_traits<Graph>::vertices_size_type>::iterator vec_iter;    typedef typename graph_traits<Graph>::vertices_size_type size_type;    typedef typename property_map<Graph, vertex_index_t>::const_type VertexID;        //Creating a std-vector for storing the distance from the end vertex in it    typename std::vector<typename graph_traits<Graph>::vertices_size_type> dist(num_vertices(g), 0);        //Wrap a property_map_iterator around the std::iterator    boost::iterator_property_map<vec_iter, VertexID, size_type, size_type&> dist_pmap(dist.begin(), get(vertex_index, g));         breadth_first_search      (g, e, visitor       (           make_bfs_visitor(record_distances(dist_pmap, on_tree_edge() ) )        )       );        //Creating a property_map for the indices of a vertex    typename property_map<Graph, vertex_index_t>::type index_map = get(vertex_index, g);        //Sets the color and priority to their initial status    Degree cdeg;        typename graph_traits<Graph>::vertex_iterator ui, ui_end;    for (boost::tie(ui, ui_end) = vertices(g); ui != ui_end; ++ui)    {        put(color, *ui, Color::white());        cdeg=get(degree, *ui)+1;        put(priority, *ui, W1*dist[index_map[*ui]]-W2*cdeg );      }        //Priority list    typedef indirect_cmp<PriorityMap, std::greater<Degree> > Compare;    Compare comp(priority);    std::list<Vertex> priority_list;    //Some more declarations    typename graph_traits<Graph>::out_edge_iterator ei, ei_end, ei2, ei2_end;    Vertex u, v, w;    put(color, s, Color::green());      //Sets the color of the starting vertex to gray    priority_list.push_front(s);                 //Puts s into the priority_list        while ( !priority_list.empty() )     {        priority_list.sort(comp);         //Orders the elements in the priority list in an ascending manner            u = priority_list.front();           //Accesses the last element in the priority list      priority_list.pop_front();               //Removes the last element in the priority list            if(get(color, u) == Color::green() )      {        //for-loop over all out-edges of vertex u        for (boost::tie(ei, ei_end) = out_edges(u, g); ei != ei_end; ++ei)         {          v = target(*ei, g);                    put( priority, v, get(priority, v) + W2 ); //updates the priority                    if (get(color, v) == Color::white() )      //test if the vertex is inactive          {            put(color, v, Color::green() );        //giving the vertex a preactive status            priority_list.push_front(v);                     //writing the vertex in the priority_queue          }                   }      }            //Here starts step 8      *permutation++ = u;                      //Puts u to the first position in the permutation-vector      put(color, u, Color::black() );          //Gives u an inactive status            //for loop over all the adjacent vertices of u      for (boost::tie(ei, ei_end) = out_edges(u, g); ei != ei_end; ++ei) {                v = target(*ei, g);                     if (get(color, v) == Color::green() ) {      //tests if the vertex is inactive                    put(color, v, Color::red() );        //giving the vertex an active status          put(priority, v, get(priority, v)+W2);  //updates the priority                            //for loop over alll adjacent vertices of v          for (boost::tie(ei2, ei2_end) = out_edges(v, g); ei2 != ei2_end; ++ei2) {            w = target(*ei2, g);                        if(get(color, w) != Color::black() ) {     //tests if vertex is postactive                            put(priority, w, get(priority, w)+W2);  //updates the priority                            if(get(color, w) == Color::white() ){                                put(color, w, Color::green() );   // gives the vertex a preactive status                priority_list.push_front(w);           // puts the vertex into the priority queue                              } //end if                          } //end if                      } //end for                  } //end if              } //end for          } //end while            return permutation;  }      /////////////////////////////////////////////////////////////////////////////////////////  // Same algorithm as before, but without the weights given (taking default weights  template <class Graph, class OutputIterator,            class ColorMap, class DegreeMap,            class PriorityMap>  OutputIterator  sloan_ordering(Graph& g,                 typename graph_traits<Graph>::vertex_descriptor s,                 typename graph_traits<Graph>::vertex_descriptor e,                 OutputIterator permutation,                  ColorMap color,                  DegreeMap degree,                  PriorityMap priority)  {    return sloan_ordering(g, s, e, permutation, color, degree, priority, WEIGHT1, WEIGHT2);   }  //////////////////////////////////////////////////////////////////////////  // Sloan algorithm without a given starting Vertex.  //  // This algorithm finds a good starting vertex itself to  // compute Sloan-ordering.  //////////////////////////////////////////////////////////////////////////   template < class Graph, class OutputIterator,              class Color, class Degree,             class Priority, class Weight>  inline OutputIterator  sloan_ordering(Graph& G,                  OutputIterator permutation,                  Color color,                  Degree degree,                  Priority priority,                  Weight W1,                  Weight W2 )  {    typedef typename boost::graph_traits<Graph>::vertex_descriptor Vertex;        Vertex s, e;    e = sloan_start_end_vertices(G, s, color, degree);        return sloan_ordering(G, s, e, permutation, color, degree, priority, W1, W2);  }  /////////////////////////////////////////////////////////////////////////////////////////  // Same as before, but without given weights (default weights are taken instead)  template < class Graph, class OutputIterator,              class Color, class Degree,             class Priority >  inline OutputIterator  sloan_ordering(Graph& G,                  OutputIterator permutation,                  Color color,                  Degree degree,                  Priority priority)  {     return sloan_ordering(G, permutation, color, degree, priority, WEIGHT1, WEIGHT2);  }    } // namespace boost#endif // BOOST_GRAPH_SLOAN_HPP
 |