astar_search.hpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564
  1. //
  2. //=======================================================================
  3. // Copyright (c) 2004 Kristopher Beevers
  4. //
  5. // Distributed under the Boost Software License, Version 1.0. (See
  6. // accompanying file LICENSE_1_0.txt or copy at
  7. // http://www.boost.org/LICENSE_1_0.txt)
  8. //=======================================================================
  9. //
  10. #ifndef BOOST_GRAPH_ASTAR_SEARCH_HPP
  11. #define BOOST_GRAPH_ASTAR_SEARCH_HPP
  12. #include <functional>
  13. #include <vector>
  14. #include <boost/limits.hpp>
  15. #include <boost/throw_exception.hpp>
  16. #include <boost/graph/named_function_params.hpp>
  17. #include <boost/graph/relax.hpp>
  18. #include <boost/graph/exception.hpp>
  19. #include <boost/graph/breadth_first_search.hpp>
  20. #include <boost/graph/iteration_macros.hpp>
  21. #include <boost/graph/detail/d_ary_heap.hpp>
  22. #include <boost/graph/property_maps/constant_property_map.hpp>
  23. #include <boost/property_map/property_map.hpp>
  24. #include <boost/property_map/vector_property_map.hpp>
  25. #include <boost/property_map/function_property_map.hpp>
  26. #include <boost/concept/assert.hpp>
  27. namespace boost {
  28. template <class Heuristic, class Graph>
  29. struct AStarHeuristicConcept {
  30. void constraints()
  31. {
  32. BOOST_CONCEPT_ASSERT(( CopyConstructibleConcept<Heuristic> ));
  33. h(u);
  34. }
  35. Heuristic h;
  36. typename graph_traits<Graph>::vertex_descriptor u;
  37. };
  38. template <class Graph, class CostType>
  39. class astar_heuristic : public std::unary_function<
  40. typename graph_traits<Graph>::vertex_descriptor, CostType>
  41. {
  42. public:
  43. typedef typename graph_traits<Graph>::vertex_descriptor Vertex;
  44. astar_heuristic() {}
  45. CostType operator()(Vertex u) { return static_cast<CostType>(0); }
  46. };
  47. template <class Visitor, class Graph>
  48. struct AStarVisitorConcept {
  49. void constraints()
  50. {
  51. BOOST_CONCEPT_ASSERT(( CopyConstructibleConcept<Visitor> ));
  52. vis.initialize_vertex(u, g);
  53. vis.discover_vertex(u, g);
  54. vis.examine_vertex(u, g);
  55. vis.examine_edge(e, g);
  56. vis.edge_relaxed(e, g);
  57. vis.edge_not_relaxed(e, g);
  58. vis.black_target(e, g);
  59. vis.finish_vertex(u, g);
  60. }
  61. Visitor vis;
  62. Graph g;
  63. typename graph_traits<Graph>::vertex_descriptor u;
  64. typename graph_traits<Graph>::edge_descriptor e;
  65. };
  66. template <class Visitors = null_visitor>
  67. class astar_visitor : public bfs_visitor<Visitors> {
  68. public:
  69. astar_visitor() {}
  70. astar_visitor(Visitors vis)
  71. : bfs_visitor<Visitors>(vis) {}
  72. template <class Edge, class Graph>
  73. void edge_relaxed(Edge e, const Graph& g) {
  74. invoke_visitors(this->m_vis, e, g, on_edge_relaxed());
  75. }
  76. template <class Edge, class Graph>
  77. void edge_not_relaxed(Edge e, const Graph& g) {
  78. invoke_visitors(this->m_vis, e, g, on_edge_not_relaxed());
  79. }
  80. private:
  81. template <class Edge, class Graph>
  82. void tree_edge(Edge e, const Graph& g) {}
  83. template <class Edge, class Graph>
  84. void non_tree_edge(Edge e, const Graph& g) {}
  85. };
  86. template <class Visitors>
  87. astar_visitor<Visitors>
  88. make_astar_visitor(Visitors vis) {
  89. return astar_visitor<Visitors>(vis);
  90. }
  91. typedef astar_visitor<> default_astar_visitor;
  92. namespace detail {
  93. template <class AStarHeuristic, class UniformCostVisitor,
  94. class UpdatableQueue, class PredecessorMap,
  95. class CostMap, class DistanceMap, class WeightMap,
  96. class ColorMap, class BinaryFunction,
  97. class BinaryPredicate>
  98. struct astar_bfs_visitor
  99. {
  100. typedef typename property_traits<CostMap>::value_type C;
  101. typedef typename property_traits<ColorMap>::value_type ColorValue;
  102. typedef color_traits<ColorValue> Color;
  103. typedef typename property_traits<DistanceMap>::value_type distance_type;
  104. astar_bfs_visitor(AStarHeuristic h, UniformCostVisitor vis,
  105. UpdatableQueue& Q, PredecessorMap p,
  106. CostMap c, DistanceMap d, WeightMap w,
  107. ColorMap col, BinaryFunction combine,
  108. BinaryPredicate compare, C zero)
  109. : m_h(h), m_vis(vis), m_Q(Q), m_predecessor(p), m_cost(c),
  110. m_distance(d), m_weight(w), m_color(col),
  111. m_combine(combine), m_compare(compare), m_zero(zero) {}
  112. template <class Vertex, class Graph>
  113. void initialize_vertex(Vertex u, const Graph& g) {
  114. m_vis.initialize_vertex(u, g);
  115. }
  116. template <class Vertex, class Graph>
  117. void discover_vertex(Vertex u, const Graph& g) {
  118. m_vis.discover_vertex(u, g);
  119. }
  120. template <class Vertex, class Graph>
  121. void examine_vertex(Vertex u, const Graph& g) {
  122. m_vis.examine_vertex(u, g);
  123. }
  124. template <class Vertex, class Graph>
  125. void finish_vertex(Vertex u, const Graph& g) {
  126. m_vis.finish_vertex(u, g);
  127. }
  128. template <class Edge, class Graph>
  129. void examine_edge(Edge e, const Graph& g) {
  130. if (m_compare(get(m_weight, e), m_zero))
  131. BOOST_THROW_EXCEPTION(negative_edge());
  132. m_vis.examine_edge(e, g);
  133. }
  134. template <class Edge, class Graph>
  135. void non_tree_edge(Edge, const Graph&) {}
  136. template <class Edge, class Graph>
  137. void tree_edge(Edge e, const Graph& g) {
  138. using boost::get;
  139. bool m_decreased =
  140. relax(e, g, m_weight, m_predecessor, m_distance,
  141. m_combine, m_compare);
  142. if(m_decreased) {
  143. m_vis.edge_relaxed(e, g);
  144. put(m_cost, target(e, g),
  145. m_combine(get(m_distance, target(e, g)),
  146. m_h(target(e, g))));
  147. } else
  148. m_vis.edge_not_relaxed(e, g);
  149. }
  150. template <class Edge, class Graph>
  151. void gray_target(Edge e, const Graph& g) {
  152. using boost::get;
  153. bool m_decreased =
  154. relax(e, g, m_weight, m_predecessor, m_distance,
  155. m_combine, m_compare);
  156. if(m_decreased) {
  157. put(m_cost, target(e, g),
  158. m_combine(get(m_distance, target(e, g)),
  159. m_h(target(e, g))));
  160. m_Q.update(target(e, g));
  161. m_vis.edge_relaxed(e, g);
  162. } else
  163. m_vis.edge_not_relaxed(e, g);
  164. }
  165. template <class Edge, class Graph>
  166. void black_target(Edge e, const Graph& g) {
  167. using boost::get;
  168. bool m_decreased =
  169. relax(e, g, m_weight, m_predecessor, m_distance,
  170. m_combine, m_compare);
  171. if(m_decreased) {
  172. m_vis.edge_relaxed(e, g);
  173. put(m_cost, target(e, g),
  174. m_combine(get(m_distance, target(e, g)),
  175. m_h(target(e, g))));
  176. m_Q.push(target(e, g));
  177. put(m_color, target(e, g), Color::gray());
  178. m_vis.black_target(e, g);
  179. } else
  180. m_vis.edge_not_relaxed(e, g);
  181. }
  182. AStarHeuristic m_h;
  183. UniformCostVisitor m_vis;
  184. UpdatableQueue& m_Q;
  185. PredecessorMap m_predecessor;
  186. CostMap m_cost;
  187. DistanceMap m_distance;
  188. WeightMap m_weight;
  189. ColorMap m_color;
  190. BinaryFunction m_combine;
  191. BinaryPredicate m_compare;
  192. C m_zero;
  193. };
  194. } // namespace detail
  195. template <typename VertexListGraph, typename AStarHeuristic,
  196. typename AStarVisitor, typename PredecessorMap,
  197. typename CostMap, typename DistanceMap,
  198. typename WeightMap, typename ColorMap,
  199. typename VertexIndexMap,
  200. typename CompareFunction, typename CombineFunction,
  201. typename CostInf, typename CostZero>
  202. inline void
  203. astar_search_no_init
  204. (const VertexListGraph &g,
  205. typename graph_traits<VertexListGraph>::vertex_descriptor s,
  206. AStarHeuristic h, AStarVisitor vis,
  207. PredecessorMap predecessor, CostMap cost,
  208. DistanceMap distance, WeightMap weight,
  209. ColorMap color, VertexIndexMap index_map,
  210. CompareFunction compare, CombineFunction combine,
  211. CostInf /*inf*/, CostZero zero)
  212. {
  213. typedef typename graph_traits<VertexListGraph>::vertex_descriptor
  214. Vertex;
  215. typedef boost::vector_property_map<std::size_t, VertexIndexMap> IndexInHeapMap;
  216. IndexInHeapMap index_in_heap(index_map);
  217. typedef d_ary_heap_indirect<Vertex, 4, IndexInHeapMap, CostMap, CompareFunction>
  218. MutableQueue;
  219. MutableQueue Q(cost, index_in_heap, compare);
  220. detail::astar_bfs_visitor<AStarHeuristic, AStarVisitor,
  221. MutableQueue, PredecessorMap, CostMap, DistanceMap,
  222. WeightMap, ColorMap, CombineFunction, CompareFunction>
  223. bfs_vis(h, vis, Q, predecessor, cost, distance, weight,
  224. color, combine, compare, zero);
  225. breadth_first_visit(g, s, Q, bfs_vis, color);
  226. }
  227. namespace graph_detail {
  228. template <typename A, typename B>
  229. struct select1st {
  230. typedef std::pair<A, B> argument_type;
  231. typedef A result_type;
  232. A operator()(const std::pair<A, B>& p) const {return p.first;}
  233. };
  234. }
  235. template <typename VertexListGraph, typename AStarHeuristic,
  236. typename AStarVisitor, typename PredecessorMap,
  237. typename CostMap, typename DistanceMap,
  238. typename WeightMap,
  239. typename CompareFunction, typename CombineFunction,
  240. typename CostInf, typename CostZero>
  241. inline void
  242. astar_search_no_init_tree
  243. (const VertexListGraph &g,
  244. typename graph_traits<VertexListGraph>::vertex_descriptor s,
  245. AStarHeuristic h, AStarVisitor vis,
  246. PredecessorMap predecessor, CostMap cost,
  247. DistanceMap distance, WeightMap weight,
  248. CompareFunction compare, CombineFunction combine,
  249. CostInf /*inf*/, CostZero zero)
  250. {
  251. typedef typename graph_traits<VertexListGraph>::vertex_descriptor
  252. Vertex;
  253. typedef typename property_traits<DistanceMap>::value_type Distance;
  254. typedef d_ary_heap_indirect<
  255. std::pair<Distance, Vertex>,
  256. 4,
  257. null_property_map<std::pair<Distance, Vertex>, std::size_t>,
  258. function_property_map<graph_detail::select1st<Distance, Vertex>, std::pair<Distance, Vertex> >,
  259. CompareFunction>
  260. MutableQueue;
  261. MutableQueue Q(
  262. make_function_property_map<std::pair<Distance, Vertex> >(graph_detail::select1st<Distance, Vertex>()),
  263. null_property_map<std::pair<Distance, Vertex>, std::size_t>(),
  264. compare);
  265. vis.discover_vertex(s, g);
  266. Q.push(std::make_pair(get(cost, s), s));
  267. while (!Q.empty()) {
  268. Vertex v;
  269. Distance v_rank;
  270. boost::tie(v_rank, v) = Q.top();
  271. Q.pop();
  272. vis.examine_vertex(v, g);
  273. BGL_FORALL_OUTEDGES_T(v, e, g, VertexListGraph) {
  274. Vertex w = target(e, g);
  275. vis.examine_edge(e, g);
  276. Distance e_weight = get(weight, e);
  277. if (compare(e_weight, zero))
  278. BOOST_THROW_EXCEPTION(negative_edge());
  279. bool decreased =
  280. relax(e, g, weight, predecessor, distance,
  281. combine, compare);
  282. Distance w_d = combine(get(distance, v), e_weight);
  283. if (decreased) {
  284. vis.edge_relaxed(e, g);
  285. Distance w_rank = combine(get(distance, w), h(w));
  286. put(cost, w, w_rank);
  287. vis.discover_vertex(w, g);
  288. Q.push(std::make_pair(w_rank, w));
  289. } else {
  290. vis.edge_not_relaxed(e, g);
  291. }
  292. }
  293. vis.finish_vertex(v, g);
  294. }
  295. }
  296. // Non-named parameter interface
  297. template <typename VertexListGraph, typename AStarHeuristic,
  298. typename AStarVisitor, typename PredecessorMap,
  299. typename CostMap, typename DistanceMap,
  300. typename WeightMap, typename VertexIndexMap,
  301. typename ColorMap,
  302. typename CompareFunction, typename CombineFunction,
  303. typename CostInf, typename CostZero>
  304. inline void
  305. astar_search
  306. (const VertexListGraph &g,
  307. typename graph_traits<VertexListGraph>::vertex_descriptor s,
  308. AStarHeuristic h, AStarVisitor vis,
  309. PredecessorMap predecessor, CostMap cost,
  310. DistanceMap distance, WeightMap weight,
  311. VertexIndexMap index_map, ColorMap color,
  312. CompareFunction compare, CombineFunction combine,
  313. CostInf inf, CostZero zero)
  314. {
  315. typedef typename property_traits<ColorMap>::value_type ColorValue;
  316. typedef color_traits<ColorValue> Color;
  317. typename graph_traits<VertexListGraph>::vertex_iterator ui, ui_end;
  318. for (boost::tie(ui, ui_end) = vertices(g); ui != ui_end; ++ui) {
  319. put(color, *ui, Color::white());
  320. put(distance, *ui, inf);
  321. put(cost, *ui, inf);
  322. put(predecessor, *ui, *ui);
  323. vis.initialize_vertex(*ui, g);
  324. }
  325. put(distance, s, zero);
  326. put(cost, s, h(s));
  327. astar_search_no_init
  328. (g, s, h, vis, predecessor, cost, distance, weight,
  329. color, index_map, compare, combine, inf, zero);
  330. }
  331. // Non-named parameter interface
  332. template <typename VertexListGraph, typename AStarHeuristic,
  333. typename AStarVisitor, typename PredecessorMap,
  334. typename CostMap, typename DistanceMap,
  335. typename WeightMap,
  336. typename CompareFunction, typename CombineFunction,
  337. typename CostInf, typename CostZero>
  338. inline void
  339. astar_search_tree
  340. (const VertexListGraph &g,
  341. typename graph_traits<VertexListGraph>::vertex_descriptor s,
  342. AStarHeuristic h, AStarVisitor vis,
  343. PredecessorMap predecessor, CostMap cost,
  344. DistanceMap distance, WeightMap weight,
  345. CompareFunction compare, CombineFunction combine,
  346. CostInf inf, CostZero zero)
  347. {
  348. typename graph_traits<VertexListGraph>::vertex_iterator ui, ui_end;
  349. for (boost::tie(ui, ui_end) = vertices(g); ui != ui_end; ++ui) {
  350. put(distance, *ui, inf);
  351. put(cost, *ui, inf);
  352. put(predecessor, *ui, *ui);
  353. vis.initialize_vertex(*ui, g);
  354. }
  355. put(distance, s, zero);
  356. put(cost, s, h(s));
  357. astar_search_no_init_tree
  358. (g, s, h, vis, predecessor, cost, distance, weight,
  359. compare, combine, inf, zero);
  360. }
  361. // Named parameter interfaces
  362. template <typename VertexListGraph,
  363. typename AStarHeuristic,
  364. typename P, typename T, typename R>
  365. void
  366. astar_search
  367. (const VertexListGraph &g,
  368. typename graph_traits<VertexListGraph>::vertex_descriptor s,
  369. AStarHeuristic h, const bgl_named_params<P, T, R>& params)
  370. {
  371. using namespace boost::graph::keywords;
  372. typedef bgl_named_params<P, T, R> params_type;
  373. BOOST_GRAPH_DECLARE_CONVERTED_PARAMETERS(params_type, params)
  374. // Distance type is the value type of the distance map if there is one,
  375. // otherwise the value type of the weight map.
  376. typedef
  377. typename detail::override_const_property_result<
  378. arg_pack_type, tag::weight_map, edge_weight_t, VertexListGraph>::type
  379. weight_map_type;
  380. typedef typename boost::property_traits<weight_map_type>::value_type W;
  381. typedef
  382. typename detail::map_maker<VertexListGraph, arg_pack_type, tag::distance_map, W>::map_type
  383. distance_map_type;
  384. typedef typename boost::property_traits<distance_map_type>::value_type D;
  385. const D inf = arg_pack[_distance_inf || detail::get_max<D>()];
  386. astar_search
  387. (g, s, h,
  388. arg_pack[_visitor | make_astar_visitor(null_visitor())],
  389. arg_pack[_predecessor_map | dummy_property_map()],
  390. detail::make_property_map_from_arg_pack_gen<tag::rank_map, D>(D())(g, arg_pack),
  391. detail::make_property_map_from_arg_pack_gen<tag::distance_map, W>(W())(g, arg_pack),
  392. detail::override_const_property(arg_pack, _weight_map, g, edge_weight),
  393. detail::override_const_property(arg_pack, _vertex_index_map, g, vertex_index),
  394. detail::make_color_map_from_arg_pack(g, arg_pack),
  395. arg_pack[_distance_compare | std::less<D>()],
  396. arg_pack[_distance_combine | closed_plus<D>(inf)],
  397. inf,
  398. arg_pack[_distance_zero | D()]);
  399. }
  400. // Named parameter interfaces
  401. template <typename VertexListGraph,
  402. typename AStarHeuristic,
  403. typename P, typename T, typename R>
  404. void
  405. astar_search_tree
  406. (const VertexListGraph &g,
  407. typename graph_traits<VertexListGraph>::vertex_descriptor s,
  408. AStarHeuristic h, const bgl_named_params<P, T, R>& params)
  409. {
  410. using namespace boost::graph::keywords;
  411. typedef bgl_named_params<P, T, R> params_type;
  412. BOOST_GRAPH_DECLARE_CONVERTED_PARAMETERS(params_type, params)
  413. // Distance type is the value type of the distance map if there is one,
  414. // otherwise the value type of the weight map.
  415. typedef
  416. typename detail::override_const_property_result<
  417. arg_pack_type, tag::weight_map, edge_weight_t, VertexListGraph>::type
  418. weight_map_type;
  419. typedef typename boost::property_traits<weight_map_type>::value_type W;
  420. typedef
  421. typename detail::map_maker<VertexListGraph, arg_pack_type, tag::distance_map, W>::map_type
  422. distance_map_type;
  423. typedef typename boost::property_traits<distance_map_type>::value_type D;
  424. const D inf = arg_pack[_distance_inf || detail::get_max<D>()];
  425. astar_search_tree
  426. (g, s, h,
  427. arg_pack[_visitor | make_astar_visitor(null_visitor())],
  428. arg_pack[_predecessor_map | dummy_property_map()],
  429. detail::make_property_map_from_arg_pack_gen<tag::rank_map, D>(D())(g, arg_pack),
  430. detail::make_property_map_from_arg_pack_gen<tag::distance_map, W>(W())(g, arg_pack),
  431. detail::override_const_property(arg_pack, _weight_map, g, edge_weight),
  432. arg_pack[_distance_compare | std::less<D>()],
  433. arg_pack[_distance_combine | closed_plus<D>(inf)],
  434. inf,
  435. arg_pack[_distance_zero | D()]);
  436. }
  437. template <typename VertexListGraph,
  438. typename AStarHeuristic,
  439. typename P, typename T, typename R>
  440. void
  441. astar_search_no_init
  442. (const VertexListGraph &g,
  443. typename graph_traits<VertexListGraph>::vertex_descriptor s,
  444. AStarHeuristic h, const bgl_named_params<P, T, R>& params)
  445. {
  446. using namespace boost::graph::keywords;
  447. typedef bgl_named_params<P, T, R> params_type;
  448. BOOST_GRAPH_DECLARE_CONVERTED_PARAMETERS(params_type, params)
  449. typedef
  450. typename detail::override_const_property_result<
  451. arg_pack_type, tag::weight_map, edge_weight_t, VertexListGraph>::type
  452. weight_map_type;
  453. typedef typename boost::property_traits<weight_map_type>::value_type D;
  454. const D inf = arg_pack[_distance_inf || detail::get_max<D>()];
  455. astar_search_no_init
  456. (g, s, h,
  457. arg_pack[_visitor | make_astar_visitor(null_visitor())],
  458. arg_pack[_predecessor_map | dummy_property_map()],
  459. detail::make_property_map_from_arg_pack_gen<tag::rank_map, D>(D())(g, arg_pack),
  460. detail::make_property_map_from_arg_pack_gen<tag::distance_map, D>(D())(g, arg_pack),
  461. detail::override_const_property(arg_pack, _weight_map, g, edge_weight),
  462. detail::make_color_map_from_arg_pack(g, arg_pack),
  463. detail::override_const_property(arg_pack, _vertex_index_map, g, vertex_index),
  464. arg_pack[_distance_compare | std::less<D>()],
  465. arg_pack[_distance_combine | closed_plus<D>(inf)],
  466. inf,
  467. arg_pack[_distance_zero | D()]);
  468. }
  469. template <typename VertexListGraph,
  470. typename AStarHeuristic,
  471. typename P, typename T, typename R>
  472. void
  473. astar_search_no_init_tree
  474. (const VertexListGraph &g,
  475. typename graph_traits<VertexListGraph>::vertex_descriptor s,
  476. AStarHeuristic h, const bgl_named_params<P, T, R>& params)
  477. {
  478. using namespace boost::graph::keywords;
  479. typedef bgl_named_params<P, T, R> params_type;
  480. BOOST_GRAPH_DECLARE_CONVERTED_PARAMETERS(params_type, params)
  481. typedef
  482. typename detail::override_const_property_result<
  483. arg_pack_type, tag::weight_map, edge_weight_t, VertexListGraph>::type
  484. weight_map_type;
  485. typedef typename boost::property_traits<weight_map_type>::value_type D;
  486. const D inf = arg_pack[_distance_inf || detail::get_max<D>()];
  487. astar_search_no_init_tree
  488. (g, s, h,
  489. arg_pack[_visitor | make_astar_visitor(null_visitor())],
  490. arg_pack[_predecessor_map | dummy_property_map()],
  491. detail::make_property_map_from_arg_pack_gen<tag::rank_map, D>(D())(g, arg_pack),
  492. detail::make_property_map_from_arg_pack_gen<tag::distance_map, D>(D())(g, arg_pack),
  493. detail::override_const_property(arg_pack, _weight_map, g, edge_weight),
  494. arg_pack[_distance_compare | std::less<D>()],
  495. arg_pack[_distance_combine | closed_plus<D>(inf)],
  496. inf,
  497. arg_pack[_distance_zero | D()]);
  498. }
  499. } // namespace boost
  500. #endif // BOOST_GRAPH_ASTAR_SEARCH_HPP