From 8ddb3a126c1bb3b72807d381d2d1c7a80fec4fad Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Tue, 12 Sep 2023 14:30:32 +0800 Subject: [PATCH 01/30] Always do lane entry event even when starting mid-lane Signed-off-by: Michael X. Grey --- .../agv/planning/DifferentialDrivePlanner.cpp | 61 ++++++++++++++++--- 1 file changed, 52 insertions(+), 9 deletions(-) diff --git a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp index 0c285aa2..5422e4f6 100644 --- a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp @@ -114,8 +114,6 @@ struct OrientationTimeMap const auto& node_low = it_low->second; const auto& node_high = it_high->second; - assert(node_low->route_from_parent.back().map() - == node_high->route_from_parent.back().map()); const auto& start_wp = node_low->route_from_parent.back().trajectory().back(); @@ -920,11 +918,56 @@ class ScheduledDifferentialDriveExpander const auto& wp = _supergraph->original().waypoints[target_waypoint_index]; const Eigen::Vector2d wp_location = wp.get_location(); + SearchNodePtr node = top; + if (start.lane().has_value() && start.location().has_value()) + { + const auto lane_index = start.lane().value(); + const Eigen::Vector2d p = start.location().value(); + const auto& lane = _supergraph->original().lanes[lane_index]; + if (lane.entry().event()) + { + Graph::Lane::EventPtr entry_event = lane.entry().event()->clone(); + const double event_cost = time::to_seconds(entry_event->duration()); + Trajectory hold; + Eigen::Vector3d p_start = {p.x(), p.y(), start.orientation()}; + const auto zero = Eigen::Vector3d::Zero(); + hold.insert(node->time, p_start, zero); + hold.insert(node->time + entry_event->duration(), p_start, zero); + Route route{wp.get_map_name(), std::move(hold)}; + if (_validator && !is_valid(node, route)) + { + // If we cannot wait for the entry event to happen then this is not a + // feasible start. + return; + } + + node = std::make_shared( + SearchNode{ + std::nullopt, + top->waypoint, + {}, + p, + start.orientation(), + node->time + entry_event->duration(), + node->remaining_cost_estimate, + {std::move(route)}, + std::move(entry_event), + node->current_cost + event_cost, + std::nullopt, + node + }); + } + } + + Graph::Lane::EventPtr entry_event; + double entry_event_cost = 0.0; + Duration entry_event_duration = Duration(0); + // If this start node did not have a waypoint, then it must have a location assert(start.location().has_value()); const auto approach_info = make_start_approach_trajectories( - top->start.value(), top->current_cost); + top->start.value(), node->current_cost); if (approach_info.trajectories.empty()) { @@ -974,7 +1017,7 @@ class ScheduledDifferentialDriveExpander for (const auto& map : map_names) { Route route{map, approach_trajectory}; - if (_validator && !is_valid(top, route)) + if (_validator && !is_valid(node, route)) { all_valid = false; break; @@ -998,7 +1041,7 @@ class ScheduledDifferentialDriveExpander for (const auto& map : map_names) { Route route{map, hold}; - if (_validator && !is_valid(top, route)) + if (_validator && !is_valid(node, route)) { all_valid = false; break; @@ -1018,7 +1061,7 @@ class ScheduledDifferentialDriveExpander // TODO(MXG): We can actually specify the orientation for this. We just // need to be smarter with make_start_approach_trajectories(). We should // really have it return a Traversal. - auto node = std::make_shared( + node = std::make_shared( SearchNode{ std::nullopt, target_waypoint_index, @@ -1026,12 +1069,12 @@ class ScheduledDifferentialDriveExpander wp_location, approach_yaw, approach_time, - top->remaining_cost_estimate - approach_cost, + node->remaining_cost_estimate - approach_cost, std::move(approach_routes), exit_event, - top->current_cost + approach_cost, + node->current_cost + approach_cost, std::nullopt, - top + node }); if (exit_event) From aae1bdde34097e28db1796ad478d02d532792a8f Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Tue, 12 Sep 2023 14:30:47 +0800 Subject: [PATCH 02/30] Allow compute_plan_starts to work inside of lifts Signed-off-by: Michael X. Grey --- rmf_traffic/src/rmf_traffic/agv/Planner.cpp | 20 +++++++++++++++----- 1 file changed, 15 insertions(+), 5 deletions(-) diff --git a/rmf_traffic/src/rmf_traffic/agv/Planner.cpp b/rmf_traffic/src/rmf_traffic/agv/Planner.cpp index 769e3b35..518a068c 100644 --- a/rmf_traffic/src/rmf_traffic/agv/Planner.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/Planner.cpp @@ -1185,11 +1185,8 @@ std::vector compute_plan_starts( { const auto& lane = graph.get_lane(i); const auto& wp0 = graph.get_waypoint(lane.entry().waypoint_index()); - if (wp0.get_map_name() != map_name) - continue; - const auto& wp1 = graph.get_waypoint(lane.exit().waypoint_index()); - if (wp1.get_map_name() != map_name) + if (wp0.get_map_name() != map_name && wp1.get_map_name() != map_name) continue; const Eigen::Vector2d p0 = wp0.get_location(); @@ -1197,9 +1194,22 @@ std::vector compute_plan_starts( const double lane_length = (p1 - p0).norm(); - // This "lane" is effectively a single point, so we'll skip it + // This "lane" is either two points stacked very close or is moving + // vertically through a lift. if (lane_length < min_lane_length) + { + const double dp0 = (p_location - p0).norm(); + const double dp1 = (p_location - p1).norm(); + if (dp0 < max_merge_lane_distance || dp1 < max_merge_lane_distance) + { + starts.emplace_back( + Plan::Start( + start_time, lane.exit().waypoint_index(), + start_yaw, p_location, i)); + } + continue; + } const Eigen::Vector2d pn = (p1 - p0) / lane_length; const Eigen::Vector2d p_l = p_location - p0; From 90bce776c8c47908199f1756dcaa95da5906a8d2 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Tue, 3 Oct 2023 08:16:53 +0000 Subject: [PATCH 03/30] Adding waypoint property for waypoints inside of a lift Signed-off-by: Michael X. Grey --- rmf_traffic/include/rmf_traffic/agv/Graph.hpp | 9 +- rmf_traffic/src/rmf_traffic/agv/Graph.cpp | 18 ++++ .../agv/planning/DifferentialDrivePlanner.cpp | 93 ++++++++++++++++--- 3 files changed, 104 insertions(+), 16 deletions(-) diff --git a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp index 2e7b97aa..116a2fd2 100644 --- a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp +++ b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp @@ -79,7 +79,6 @@ class Graph /// Set this Waypoint to be a parking spot. Waypoint& set_parking_spot(bool _is_parking_spot); - /// Returns true if this Waypoint is a charger spot. Robots are routed to /// these spots when their batteries charge levels drop below the threshold /// value. @@ -88,6 +87,14 @@ class Graph /// Set this Waypoint to be a parking spot. Waypoint& set_charger(bool _is_charger); + /// If this waypoint is inside the lift then this will return a pointer to + /// a string of the lift's name. Otherwise this will be a nullptr. + const std::string* in_lift() const; + + /// Set the name of the lift that the waypoint is inside of, or provide a + /// nullopt if it is not inside a lift. + Waypoint& set_in_lift(std::optional lift_name); + /// The index of this waypoint within the Graph. This cannot be changed /// after the waypoint is created. std::size_t index() const; diff --git a/rmf_traffic/src/rmf_traffic/agv/Graph.cpp b/rmf_traffic/src/rmf_traffic/agv/Graph.cpp index f97b25b4..75b31a40 100644 --- a/rmf_traffic/src/rmf_traffic/agv/Graph.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/Graph.cpp @@ -46,6 +46,8 @@ class Graph::Waypoint::Implementation bool charger = false; + std::optional in_lift; + template static Waypoint make(Args&& ... args) { @@ -140,6 +142,22 @@ auto Graph::Waypoint::set_charger(bool _is_charger) -> Waypoint& return *this; } +//============================================================================== +const std::string* Graph::Waypoint::in_lift() const +{ + if (_pimpl->in_lift.has_value()) + return &*_pimpl->in_lift; + return nullptr; +} + +//============================================================================== +auto Graph::Waypoint::set_in_lift( + std::optional lift_name) -> Waypoint& +{ + _pimpl->in_lift = lift_name; + return *this; +} + //============================================================================== std::size_t Graph::Waypoint::index() const { diff --git a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp index 5422e4f6..8c533bf6 100644 --- a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp @@ -32,6 +32,24 @@ namespace rmf_traffic { namespace agv { namespace planning { +class Printer : public rmf_traffic::agv::Graph::Lane::Executor +{ +public: + Printer() + { + // Do nothing + } + + void execute(const DoorOpen&) override { std::cout << "event " << __LINE__ << std::endl;; } + void execute(const DoorClose&) override { std::cout << "event " << __LINE__ << std::endl;; } + void execute(const LiftSessionBegin&) override { std::cout << "event " << __LINE__ << std::endl;; } + void execute(const LiftDoorOpen&) override { std::cout << "event " << __LINE__ << std::endl;; } + void execute(const LiftSessionEnd&) override { std::cout << "event " << __LINE__ << std::endl;; } + void execute(const LiftMove&) override { std::cout << "event " << __LINE__ << std::endl;; } + void execute(const Wait&) override { std::cout << "event " << __LINE__ << std::endl;; } + void execute(const Dock& dock) override { std::cout << "event " << __LINE__ << std::endl;; } +}; + //============================================================================== template std::vector reconstruct_nodes(const NodePtr& finish_node) @@ -45,6 +63,22 @@ std::vector reconstruct_nodes(const NodePtr& finish_node) } std::reverse(node_sequence.begin(), node_sequence.end()); + std::cout << " --- node sequence --- " << std::endl; + const auto t0 = node_sequence.front()->time; + for (const NodePtr node : node_sequence) + { + std::cout << " -- LINE:" << node->line << " | t=" << time::to_seconds(node->time - t0) << " "; + if (node->waypoint.has_value()) + std::cout << "index " << *node->waypoint << " "; + std::cout << " <" << node->position.transpose() << "> yaw=" << node->yaw << " "; + std::cout << std::endl; + if (node->event) + { + Printer printer; + node->event->execute(printer); + } + } + return node_sequence; } @@ -184,6 +218,7 @@ std::vector reconstruct_nodes( const double alpha_nom, const double rotational_threshold) { + std::cout << "==================================" << std::endl; auto node_sequence = reconstruct_nodes(finish_node); // Remove "cruft" from plans. This means making sure vehicles don't do any @@ -723,6 +758,7 @@ class ScheduledDifferentialDriveExpander double current_cost; std::optional start; SearchNodePtr parent; + std::size_t line; double get_total_cost_estimate() const { @@ -754,7 +790,8 @@ class ScheduledDifferentialDriveExpander Graph::Lane::EventPtr event_, double current_cost_, std::optional start_, - SearchNodePtr parent_) + SearchNodePtr parent_, + std::size_t line_) : entry(entry_), waypoint(waypoint_), approach_lanes(std::move(approach_lanes_)), @@ -766,7 +803,8 @@ class ScheduledDifferentialDriveExpander event(event_), current_cost(current_cost_), start(std::move(start_)), - parent(std::move(parent_)) + parent(std::move(parent_)), + line(line_) { assert(!route_from_parent.empty()); @@ -954,7 +992,8 @@ class ScheduledDifferentialDriveExpander std::move(entry_event), node->current_cost + event_cost, std::nullopt, - node + node, + __LINE__ }); } } @@ -1074,7 +1113,8 @@ class ScheduledDifferentialDriveExpander exit_event, node->current_cost + approach_cost, std::nullopt, - node + node, + __LINE__ }); if (exit_event) @@ -1092,7 +1132,8 @@ class ScheduledDifferentialDriveExpander nullptr, node->current_cost + exit_event_cost, std::nullopt, - node + node, + __LINE__ }); } @@ -1134,7 +1175,8 @@ class ScheduledDifferentialDriveExpander nullptr, top->current_cost + hold_cost, start, - top + top, + __LINE__ })); } @@ -1180,7 +1222,8 @@ class ScheduledDifferentialDriveExpander nullptr, top->current_cost + cost, std::nullopt, - top + top, + __LINE__ }); } @@ -1236,7 +1279,8 @@ class ScheduledDifferentialDriveExpander nullptr, top->current_cost + cost, std::nullopt, - top + top, + __LINE__ }); } @@ -1468,6 +1512,19 @@ class ScheduledDifferentialDriveExpander const double cost = calculate_cost(approach_route.trajectory()); const double yaw = approach_wp.position()[2]; const auto time = approach_wp.time(); + auto parent = node; + std::vector route_from_parent; + if (approach_route.trajectory().size() < 2) + { + // This is just an entry event so we will skip the unnecessary + // intermediate node + parent = node->parent; + route_from_parent = node->route_from_parent; + } + else + { + route_from_parent = {std::move(approach_route)}; + } node = std::make_shared( SearchNode{ @@ -1483,11 +1540,12 @@ class ScheduledDifferentialDriveExpander time, *remaining_cost_estimate + entry_event_cost + alt->cost + exit_event_cost, - {std::move(approach_route)}, + route_from_parent, traversal.entry_event, node->current_cost + cost, std::nullopt, - node + parent, + __LINE__ }); } @@ -1526,7 +1584,8 @@ class ScheduledDifferentialDriveExpander traversal.exit_event, node->current_cost + entry_event_cost + alt->cost, std::nullopt, - node + node, + __LINE__ }); if (traversal.exit_event && exit_event_route.trajectory().size() >= 2) @@ -1544,7 +1603,8 @@ class ScheduledDifferentialDriveExpander nullptr, node->current_cost + exit_event_cost, std::nullopt, - node + node, + __LINE__ }); } @@ -1594,7 +1654,8 @@ class ScheduledDifferentialDriveExpander solution_root->info.event, search_node->current_cost + approach_info.cost, std::nullopt, - search_node + search_node, + __LINE__ }); } @@ -1642,7 +1703,8 @@ class ScheduledDifferentialDriveExpander solution_node->info.event, search_node->current_cost + solution_node->info.cost_from_parent, std::nullopt, - search_node + search_node, + __LINE__ }); solution_node = solution_node->child; @@ -1945,7 +2007,8 @@ class ScheduledDifferentialDriveExpander nullptr, 0.0, start, - nullptr + nullptr, + __LINE__ }); } From 82117e50dbcf3a4ba361d3930c745e4e00c58898 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Thu, 5 Oct 2023 06:12:37 +0000 Subject: [PATCH 04/30] Provide lift properties Signed-off-by: Michael X. Grey --- rmf_traffic/include/rmf_traffic/agv/Graph.hpp | 57 +++++++++- rmf_traffic/src/rmf_traffic/agv/Graph.cpp | 103 ++++++++++++++++-- .../src/rmf_traffic/agv/internal_Graph.hpp | 1 + 3 files changed, 147 insertions(+), 14 deletions(-) diff --git a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp index 116a2fd2..842573c2 100644 --- a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp +++ b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp @@ -27,6 +27,7 @@ #include #include +#include #include namespace rmf_traffic { @@ -37,10 +38,45 @@ class Graph { public: - class Waypoint + /// Properties related to lifts (elevators) that exist in the graph + class LiftProperties { public: + /// Get the name of the lift. + const std::string& name() const; + + /// Get the (x, y) location of the lift in RMF canonical coordinates. + Eigen::Vector2d location() const; + + /// Get the orientation (in radians) of the lift in RMF canonical + /// coordinates. + double orientation() const; + + /// Get the dimensions of the lift, aligned with the lift's local (x, y) + /// coordinates. + Eigen::Vector2d dimensions() const; + + /// Get whether the specified position, given in RMF canonical coordinates, + /// is inside the lift. + bool is_in_lift(Eigen::Vector2d position) const; + + /// Constructor + LiftProperties( + std::string name, + Eigen::Vector2d location, + double orientations, + Eigen::Vector2d dimensions); + + private: + class Implementation; + rmf_utils::impl_ptr _pimpl; + }; + using LiftPropertiesPtr = std::shared_ptr; + /// Properties assigned to each waypoint (vertex) in the graph + class Waypoint + { + public: /// Get the name of the map that this Waypoint exists on. const std::string& get_map_name() const; @@ -88,12 +124,12 @@ class Graph Waypoint& set_charger(bool _is_charger); /// If this waypoint is inside the lift then this will return a pointer to - /// a string of the lift's name. Otherwise this will be a nullptr. - const std::string* in_lift() const; + /// the properties of the lift. Otherwise this will be a nullptr. + LiftPropertiesPtr in_lift() const; - /// Set the name of the lift that the waypoint is inside of, or provide a - /// nullopt if it is not inside a lift. - Waypoint& set_in_lift(std::optional lift_name); + /// Set the properties of the lift that the waypoint is inside of, or + /// provide a nullptr if it is not inside a lift. + Waypoint& set_in_lift(LiftPropertiesPtr properties); /// The index of this waypoint within the Graph. This cannot be changed /// after the waypoint is created. @@ -596,6 +632,15 @@ class Graph /// const-qualified lane_from() const Lane* lane_from(std::size_t from_wp, std::size_t to_wp) const; + /// Get the lifts that are known to exist for this graph. + /// NOTE: There is no mechanism to automatically keep known_lifts synced with + /// the actual lifts used by the vertices, so this must be kept in sync + /// manually. + const std::unordered_set& known_lifts() const; + + /// Mutable reference to the known lifts + std::unordered_set& known_lifts(); + class Implementation; private: rmf_utils::impl_ptr _pimpl; diff --git a/rmf_traffic/src/rmf_traffic/agv/Graph.cpp b/rmf_traffic/src/rmf_traffic/agv/Graph.cpp index 75b31a40..a97bb5bb 100644 --- a/rmf_traffic/src/rmf_traffic/agv/Graph.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/Graph.cpp @@ -25,6 +25,84 @@ namespace rmf_traffic { namespace agv { +//============================================================================== +class Graph::LiftProperties::Implementation +{ +public: + std::string name; + Eigen::Vector2d location; + double orientation; + Eigen::Vector2d half_dimensions; + Eigen::Isometry2d tf_inv; +}; + +//============================================================================== +const std::string& Graph::LiftProperties::name() const +{ + return _pimpl->name; +} + +//============================================================================== +Eigen::Vector2d Graph::LiftProperties::location() const +{ + return _pimpl->location; +} + +//============================================================================== +double Graph::LiftProperties::orientation() const +{ + return _pimpl->orientation; +} + +//============================================================================== +Eigen::Vector2d Graph::LiftProperties::dimensions() const +{ + return 2.0 * _pimpl->half_dimensions; +} + +//============================================================================== +bool Graph::LiftProperties::is_in_lift(Eigen::Vector2d position) const +{ + Eigen::Vector2d p_local = _pimpl->tf_inv * position; + for (int i = 0; i < 2; ++i) + { + if (p_local[i] < -_pimpl->half_dimensions[i]) + return false; + + if (_pimpl->half_dimensions[i] < p_local[i]) + return false; + } + + return true; +} + +//============================================================================== +Eigen::Isometry2d make_lift_tf_inv(Eigen::Vector2d location, double orientation) +{ + Eigen::Isometry2d tf = Eigen::Isometry2d::Identity(); + tf.translate(location); + tf.rotate(orientation); + return tf.inverse(); +} + +//============================================================================== +Graph::LiftProperties::LiftProperties( + std::string name, + Eigen::Vector2d location, + double orientation, + Eigen::Vector2d dimensions) +: _pimpl(rmf_utils::make_impl( + Implementation { + std::move(name), + location, + orientation, + dimensions / 2.0, + make_lift_tf_inv(location, orientation) + })) +{ + // Do nothing +} + //============================================================================== class Graph::Waypoint::Implementation { @@ -46,7 +124,7 @@ class Graph::Waypoint::Implementation bool charger = false; - std::optional in_lift; + LiftPropertiesPtr in_lift = nullptr; template static Waypoint make(Args&& ... args) @@ -143,18 +221,15 @@ auto Graph::Waypoint::set_charger(bool _is_charger) -> Waypoint& } //============================================================================== -const std::string* Graph::Waypoint::in_lift() const +auto Graph::Waypoint::in_lift() const -> LiftPropertiesPtr { - if (_pimpl->in_lift.has_value()) - return &*_pimpl->in_lift; - return nullptr; + return _pimpl->in_lift; } //============================================================================== -auto Graph::Waypoint::set_in_lift( - std::optional lift_name) -> Waypoint& +auto Graph::Waypoint::set_in_lift(LiftPropertiesPtr lift) -> Waypoint& { - _pimpl->in_lift = lift_name; + _pimpl->in_lift = lift; return *this; } @@ -967,6 +1042,18 @@ auto Graph::lane_from(std::size_t from_wp, std::size_t to_wp) -> Lane* return &_pimpl->lanes.at(it->second); } +//============================================================================== +auto Graph::known_lifts() const -> const std::unordered_set& +{ + return _pimpl->lifts; +} + +//============================================================================== +auto Graph::known_lifts() -> std::unordered_set& +{ + return _pimpl->lifts; +} + //============================================================================== auto Graph::lane_from(std::size_t from_wp, std::size_t to_wp) const -> const Lane* diff --git a/rmf_traffic/src/rmf_traffic/agv/internal_Graph.hpp b/rmf_traffic/src/rmf_traffic/agv/internal_Graph.hpp index d1c2385f..fc321f41 100644 --- a/rmf_traffic/src/rmf_traffic/agv/internal_Graph.hpp +++ b/rmf_traffic/src/rmf_traffic/agv/internal_Graph.hpp @@ -31,6 +31,7 @@ class Graph::Implementation std::vector waypoints; std::vector lanes; std::unordered_map keys; + std::unordered_set lifts; // A map from a waypoint index to the set of lanes that can exit from it std::vector> lanes_from; From 910af569a44d907a9dbde230e59ac1b09f7889b4 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Thu, 5 Oct 2023 07:26:25 +0000 Subject: [PATCH 05/30] Remove debug output Signed-off-by: Michael X. Grey --- .../agv/planning/DifferentialDrivePlanner.cpp | 34 ------------------- 1 file changed, 34 deletions(-) diff --git a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp index 8c533bf6..1b8eeee3 100644 --- a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp @@ -32,24 +32,6 @@ namespace rmf_traffic { namespace agv { namespace planning { -class Printer : public rmf_traffic::agv::Graph::Lane::Executor -{ -public: - Printer() - { - // Do nothing - } - - void execute(const DoorOpen&) override { std::cout << "event " << __LINE__ << std::endl;; } - void execute(const DoorClose&) override { std::cout << "event " << __LINE__ << std::endl;; } - void execute(const LiftSessionBegin&) override { std::cout << "event " << __LINE__ << std::endl;; } - void execute(const LiftDoorOpen&) override { std::cout << "event " << __LINE__ << std::endl;; } - void execute(const LiftSessionEnd&) override { std::cout << "event " << __LINE__ << std::endl;; } - void execute(const LiftMove&) override { std::cout << "event " << __LINE__ << std::endl;; } - void execute(const Wait&) override { std::cout << "event " << __LINE__ << std::endl;; } - void execute(const Dock& dock) override { std::cout << "event " << __LINE__ << std::endl;; } -}; - //============================================================================== template std::vector reconstruct_nodes(const NodePtr& finish_node) @@ -63,21 +45,6 @@ std::vector reconstruct_nodes(const NodePtr& finish_node) } std::reverse(node_sequence.begin(), node_sequence.end()); - std::cout << " --- node sequence --- " << std::endl; - const auto t0 = node_sequence.front()->time; - for (const NodePtr node : node_sequence) - { - std::cout << " -- LINE:" << node->line << " | t=" << time::to_seconds(node->time - t0) << " "; - if (node->waypoint.has_value()) - std::cout << "index " << *node->waypoint << " "; - std::cout << " <" << node->position.transpose() << "> yaw=" << node->yaw << " "; - std::cout << std::endl; - if (node->event) - { - Printer printer; - node->event->execute(printer); - } - } return node_sequence; } @@ -218,7 +185,6 @@ std::vector reconstruct_nodes( const double alpha_nom, const double rotational_threshold) { - std::cout << "==================================" << std::endl; auto node_sequence = reconstruct_nodes(finish_node); // Remove "cruft" from plans. This means making sure vehicles don't do any From 8bf9ab241a88e47ce457535c7c66667a30b71399 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Mon, 23 Oct 2023 23:09:18 +0000 Subject: [PATCH 06/30] Fix plan starts that have entry events Signed-off-by: Michael X. Grey --- .../agv/planning/DifferentialDrivePlanner.cpp | 30 +++++++++++++++++-- 1 file changed, 27 insertions(+), 3 deletions(-) diff --git a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp index 1b8eeee3..5dca3681 100644 --- a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp @@ -186,6 +186,20 @@ std::vector reconstruct_nodes( const double rotational_threshold) { auto node_sequence = reconstruct_nodes(finish_node); + std::optional start; + if (!node_sequence.empty()) + { + start = node_sequence.front()->start; + if (!start.has_value()) + { + // *INDENT-OFF* + throw std::runtime_error( + "[rmf_traffic::agv::Planner::plan][1] The root node of a solved plan " + "is missing its Start information. This should not happen. Please " + "report this critical bug to the maintainers of rmf_traffic."); + // *INDENT-ON* + } + } // Remove "cruft" from plans. This means making sure vehicles don't do any // unnecessary motions. @@ -234,7 +248,12 @@ std::vector reconstruct_nodes( duplicate.squash(validator); } - return reconstruct_nodes(finish_node); + auto final_node_sequence = reconstruct_nodes(finish_node); + if (!final_node_sequence.empty()) + { + final_node_sequence.front()->start = start; + } + return final_node_sequence; } //============================================================================== @@ -679,8 +698,8 @@ Plan::Start find_start(NodePtr node) { // *INDENT-OFF* throw std::runtime_error( - "[rmf_traffic::agv::Planner::plan] The root node of a solved plan is " - "missing its Start information. This should not happen. Please report " + "[rmf_traffic::agv::Planner::plan][2] The root node of a processed plan " + "is missing its Start information. This should not happen. Please report " "this critical bug to the maintainers of rmf_traffic."); // *INDENT-ON* } @@ -1485,6 +1504,11 @@ class ScheduledDifferentialDriveExpander // This is just an entry event so we will skip the unnecessary // intermediate node parent = node->parent; + if (!parent) + { + // If the top node is a root node, then don't skip it + parent = node; + } route_from_parent = node->route_from_parent; } else From 773ac04ed28edb5536060a989f5244e07f97e510 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Tue, 31 Oct 2023 03:16:56 +0000 Subject: [PATCH 07/30] Add mutex groups to lanes and break the supergraph at mutex group switches Signed-off-by: Michael X. Grey --- rmf_traffic/include/rmf_traffic/agv/Graph.hpp | 23 ++++++++++++++ rmf_traffic/src/rmf_traffic/agv/Graph.cpp | 30 +++++++++++++++++++ .../rmf_traffic/agv/planning/Supergraph.cpp | 28 +++++++++++++++++ 3 files changed, 81 insertions(+) diff --git a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp index 842573c2..35ae2120 100644 --- a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp +++ b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp @@ -162,6 +162,17 @@ class Graph const std::string& name_format = "%s", const std::string& index_format = "#%d") const; + /// Get the mutex group that this waypoint is associated with. An empty + /// string implies that it is not associated with any mutex group. + /// + /// Only one robot at a time is allowed to occupy any waypoint or lane + /// associated with a particular mutex group. + const std::string& in_mutex_group() const; + + /// Set what mutex group this waypoint is associated with. Passing in an + /// empty string will disasscoiate the waypoint from any mutex group. + Waypoint& set_in_mutex_group(std::string group_name); + class Implementation; private: Waypoint(); @@ -511,6 +522,7 @@ class Graph /// Construct a default set of properties /// * speed_limit: nullopt + /// * mutex_group: "" Properties(); /// Get the speed limit along this lane. If a std::nullopt is returned, @@ -521,6 +533,17 @@ class Graph /// indicates that there is no speed limit for the lane. Properties& speed_limit(std::optional value); + /// Get the mutex group that this lane is associated with. An empty string + /// implies that it is not associated with any mutex group. + /// + /// Only one robot at a time is allowed to occupy any waypoint or lane + /// associated with a particular mutex group. + const std::string& in_mutex_group() const; + + /// Set what mutex group this lane is associated with. Passing in an + /// empty string will disassociate the lane from any mutex group. + Properties& set_in_mutex_group(std::string group_name); + class Implementation; private: rmf_utils::impl_ptr _pimpl; diff --git a/rmf_traffic/src/rmf_traffic/agv/Graph.cpp b/rmf_traffic/src/rmf_traffic/agv/Graph.cpp index a97bb5bb..136613a5 100644 --- a/rmf_traffic/src/rmf_traffic/agv/Graph.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/Graph.cpp @@ -126,6 +126,8 @@ class Graph::Waypoint::Implementation LiftPropertiesPtr in_lift = nullptr; + std::string mutex_group = ""; + template static Waypoint make(Args&& ... args) { @@ -273,6 +275,19 @@ std::string Graph::Waypoint::name_or_index( + index_format.substr(it+2); } +//============================================================================== +const std::string& Graph::Waypoint::in_mutex_group() const +{ + return _pimpl->mutex_group; +} + +//============================================================================== +auto Graph::Waypoint::set_in_mutex_group(std::string group_name) -> Waypoint& +{ + _pimpl->mutex_group = std::move(group_name); + return *this; +} + //============================================================================== Graph::Waypoint::Waypoint() { @@ -769,6 +784,7 @@ class Graph::Lane::Properties::Implementation public: std::optional speed_limit; + std::string mutex_group; }; @@ -793,6 +809,20 @@ auto Graph::Lane::Properties::speed_limit(std::optional value) return *this; } +//============================================================================== +const std::string& Graph::Lane::Properties::in_mutex_group() const +{ + return _pimpl->mutex_group; +} + +//============================================================================== +auto Graph::Lane::Properties::set_in_mutex_group(std::string group_name) +-> Properties& +{ + _pimpl->mutex_group = std::move(group_name); + return *this; +} + //============================================================================== class Graph::Lane::Implementation { diff --git a/rmf_traffic/src/rmf_traffic/agv/planning/Supergraph.cpp b/rmf_traffic/src/rmf_traffic/agv/planning/Supergraph.cpp index 69b7b78e..5f5f8eb6 100644 --- a/rmf_traffic/src/rmf_traffic/agv/planning/Supergraph.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/planning/Supergraph.cpp @@ -80,6 +80,8 @@ struct TraversalNode Graph::Lane::EventPtr entry_event; Graph::Lane::EventPtr exit_event; + std::string mutex_group; + // TODO(MXG): Replace this with a more intelligent way of handling speed limit // changes that may occur when traversing multiple consecutive lanes std::optional lowest_speed_limit; @@ -299,6 +301,7 @@ void perform_traversal( node.initial_waypoint_index = wp_index_0; node.finish_waypoint_index = wp_index_1; node.lowest_speed_limit = lane.properties().speed_limit(); + node.mutex_group = lane.properties().in_mutex_group(); if (parent) { @@ -309,6 +312,31 @@ void perform_traversal( return; } + const auto& wp0_mutex = wp0.in_mutex_group(); + if (!parent->mutex_group.empty() && wp0_mutex.empty()) + { + // We are exiting a mutex group, so we should do a quick stop here to make + // sure that we release the mutex. + return; + } + + if (!node.mutex_group.empty() && parent->mutex_group != node.mutex_group) + { + // The lane belongs to a different mutex group than the parent, so we + // cannot continuously traverse. + return; + } + + const auto& wp1_mutex = wp1.in_mutex_group(); + if (!wp1_mutex.empty() && wp1_mutex != node.mutex_group) + { + // The end waypoint of this lane belongs to a different mutex group than + // the lane leading up to it. The waypoint's mutex must be locked before + // traversing this lane, so we should stop before traversing this lane + // instead of continuing from a parent. + return; + } + node.initial_lane_index = parent->initial_lane_index; node.initial_waypoint_index = parent->initial_waypoint_index; node.initial_p = parent->initial_p; From 3efab183d4445c826ec84f89549a50c18e226d44 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Thu, 2 Nov 2023 23:30:15 +0000 Subject: [PATCH 08/30] Catch which node is missing its waypoint Signed-off-by: Michael X. Grey --- .../agv/planning/DifferentialDrivePlanner.cpp | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp index 5dca3681..419a4e5e 100644 --- a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp @@ -1150,7 +1150,7 @@ class ScheduledDifferentialDriveExpander std::make_shared( SearchNode{ std::nullopt, - std::nullopt, + top->waypoint, {}, p, yaw, @@ -1721,7 +1721,15 @@ class ScheduledDifferentialDriveExpander if (!top->waypoint.has_value()) { // If the node does not have a waypoint, then it must be a start node. - assert(top->start.has_value()); + if (!top->start.has_value()) + { + throw std::runtime_error( + "[rmf_traffic::agv::planning::DifferentialDrivePlanner::expand] " + "Node has no waypoint and also no start information. It was produced " + "on line [" + std::to_string(top->line) + "]. This should not be " + "possible. Please report this critical bug to the maintainers of " + "rmf_traffic."); + } expand_start(top, queue); return; } From bd44823821a2c2f6ae08bfb405e2381280ba3732 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Wed, 8 Nov 2023 06:59:18 +0000 Subject: [PATCH 09/30] Fix the way start nodes are expanded Signed-off-by: Michael X. Grey --- .../agv/planning/DifferentialDrivePlanner.cpp | 42 +++++++++++-------- 1 file changed, 25 insertions(+), 17 deletions(-) diff --git a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp index 419a4e5e..d2a86117 100644 --- a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp @@ -941,7 +941,7 @@ class ScheduledDifferentialDriveExpander const auto& wp = _supergraph->original().waypoints[target_waypoint_index]; const Eigen::Vector2d wp_location = wp.get_location(); - SearchNodePtr node = top; + SearchNodePtr parent = top; if (start.lane().has_value() && start.location().has_value()) { const auto lane_index = start.lane().value(); @@ -954,30 +954,30 @@ class ScheduledDifferentialDriveExpander Trajectory hold; Eigen::Vector3d p_start = {p.x(), p.y(), start.orientation()}; const auto zero = Eigen::Vector3d::Zero(); - hold.insert(node->time, p_start, zero); - hold.insert(node->time + entry_event->duration(), p_start, zero); + hold.insert(parent->time, p_start, zero); + hold.insert(parent->time + entry_event->duration(), p_start, zero); Route route{wp.get_map_name(), std::move(hold)}; - if (_validator && !is_valid(node, route)) + if (_validator && !is_valid(parent, route)) { // If we cannot wait for the entry event to happen then this is not a // feasible start. return; } - node = std::make_shared( + parent = std::make_shared( SearchNode{ std::nullopt, top->waypoint, {}, p, start.orientation(), - node->time + entry_event->duration(), - node->remaining_cost_estimate, + parent->time + entry_event->duration(), + parent->remaining_cost_estimate, {std::move(route)}, std::move(entry_event), - node->current_cost + event_cost, + parent->current_cost + event_cost, std::nullopt, - node, + parent, __LINE__ }); } @@ -991,7 +991,7 @@ class ScheduledDifferentialDriveExpander assert(start.location().has_value()); const auto approach_info = make_start_approach_trajectories( - top->start.value(), node->current_cost); + top->start.value(), parent->current_cost); if (approach_info.trajectories.empty()) { @@ -1041,7 +1041,7 @@ class ScheduledDifferentialDriveExpander for (const auto& map : map_names) { Route route{map, approach_trajectory}; - if (_validator && !is_valid(node, route)) + if (_validator && !is_valid(parent, route)) { all_valid = false; break; @@ -1065,7 +1065,7 @@ class ScheduledDifferentialDriveExpander for (const auto& map : map_names) { Route route{map, hold}; - if (_validator && !is_valid(node, route)) + if (_validator && !is_valid(parent, route)) { all_valid = false; break; @@ -1085,7 +1085,7 @@ class ScheduledDifferentialDriveExpander // TODO(MXG): We can actually specify the orientation for this. We just // need to be smarter with make_start_approach_trajectories(). We should // really have it return a Traversal. - node = std::make_shared( + auto node = std::make_shared( SearchNode{ std::nullopt, target_waypoint_index, @@ -1093,12 +1093,12 @@ class ScheduledDifferentialDriveExpander wp_location, approach_yaw, approach_time, - node->remaining_cost_estimate - approach_cost, + parent->remaining_cost_estimate - approach_cost, std::move(approach_routes), exit_event, - node->current_cost + approach_cost, + parent->current_cost + approach_cost, std::nullopt, - node, + parent, __LINE__ }); @@ -2045,7 +2045,7 @@ class ScheduledDifferentialDriveExpander for (const auto& void_node : nodes) { bool skip = false; - const auto original_node = + auto original_node = std::static_pointer_cast(void_node.first); const auto original_t = void_node.second; @@ -2080,6 +2080,14 @@ class ScheduledDifferentialDriveExpander if (skip) continue; + while (original_node && !original_node->waypoint.has_value() && !original_node->start.has_value()) + { + original_node = original_node->parent; + } + + if (!original_node) + continue; + rollout_queue.emplace_back( RolloutEntry{ original_t, From a5e475d749f28abb1c0d6c4d7a9f94ecdb6c2520 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Sun, 12 Nov 2023 09:42:47 +0000 Subject: [PATCH 10/30] Provide distanced add/substract to trajectory iterator Signed-off-by: Michael X. Grey --- .../include/rmf_traffic/Trajectory.hpp | 7 ++++ rmf_traffic/src/rmf_traffic/Trajectory.cpp | 32 +++++++++++++++++++ 2 files changed, 39 insertions(+) diff --git a/rmf_traffic/include/rmf_traffic/Trajectory.hpp b/rmf_traffic/include/rmf_traffic/Trajectory.hpp index b5968160..0cea4ae3 100644 --- a/rmf_traffic/include/rmf_traffic/Trajectory.hpp +++ b/rmf_traffic/include/rmf_traffic/Trajectory.hpp @@ -367,6 +367,13 @@ class Trajectory::base_iterator /// \return a copy of the iterator before it was decremented base_iterator operator--(int); + /// Get the iterator that would be offset from this one by the specified + /// amount + base_iterator operator+(int offset) const; + + /// Get the iterator that would be offset from this one by the specified + /// amount in the opposite direction. + base_iterator operator-(int offset) const; // TODO(MXG): Consider the spaceship operator when we can use C++20 diff --git a/rmf_traffic/src/rmf_traffic/Trajectory.cpp b/rmf_traffic/src/rmf_traffic/Trajectory.cpp index 79a325b9..3f29c7d2 100644 --- a/rmf_traffic/src/rmf_traffic/Trajectory.cpp +++ b/rmf_traffic/src/rmf_traffic/Trajectory.cpp @@ -690,6 +690,38 @@ auto Trajectory::base_iterator::operator--(int) -> base_iterator return _pimpl->post_decrement(); } +//============================================================================== +template +auto Trajectory::base_iterator::operator+(int offset) const +-> base_iterator +{ + const std::size_t N = std::abs(offset); + const bool negative = offset < 0; + + base_iterator result = *this; + for (std::size_t i=0; i < N; ++i) + { + if (negative) + { + --result; + } + else + { + ++result; + } + } + + return result; +} + +//============================================================================== +template +auto Trajectory::base_iterator::operator-(int offset) const +-> base_iterator +{ + return *this + (-offset); +} + //============================================================================== #define DEFINE_BASIC_ITERATOR_OP(op) \ template \ From c95a491f6c24854c49177495531e5011f21219da Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Thu, 16 Nov 2023 16:59:26 +0000 Subject: [PATCH 11/30] Fix progress change updates Signed-off-by: Michael X. Grey --- .../rmf_traffic/schedule/Participant.hpp | 3 ++ .../src/rmf_traffic/schedule/Database.cpp | 31 ++++++++++--------- .../src/rmf_traffic/schedule/Participant.cpp | 6 ++++ 3 files changed, 25 insertions(+), 15 deletions(-) diff --git a/rmf_traffic/include/rmf_traffic/schedule/Participant.hpp b/rmf_traffic/include/rmf_traffic/schedule/Participant.hpp index bae9b2e5..950e3de8 100644 --- a/rmf_traffic/include/rmf_traffic/schedule/Participant.hpp +++ b/rmf_traffic/include/rmf_traffic/schedule/Participant.hpp @@ -100,6 +100,9 @@ class Participant // TODO(MXG): This function needs to be unit tested. ItineraryVersion version() const; + /// Get the current progress version for this participant. + ProgressVersion progress_version() const; + /// Get the description of this participant. const ParticipantDescription& description() const; diff --git a/rmf_traffic/src/rmf_traffic/schedule/Database.cpp b/rmf_traffic/src/rmf_traffic/schedule/Database.cpp index 35db7448..3f6dd48c 100644 --- a/rmf_traffic/src/rmf_traffic/schedule/Database.cpp +++ b/rmf_traffic/src/rmf_traffic/schedule/Database.cpp @@ -1488,10 +1488,10 @@ auto Database::changes( } std::vector part_patches; - for (const auto& p : changes) + for (const auto p : _pimpl->participant_ids) { - const auto& changeset = p.second; - const auto& state = _pimpl->states.at(p.first); + const auto& changeset = changes[p]; + const auto& state = _pimpl->states.at(p); std::vector delays; for (const auto& d : changeset.delays) @@ -1502,15 +1502,6 @@ auto Database::changes( }); } - if (changeset.erasures.empty() - && delays.empty() - && changeset.additions.empty()) - { - // There aren't actually any changes for this participant, so we will - // leave it out of the patch. - continue; - } - std::optional progress; if (state.schedule_version_of_progress.has_value()) { @@ -1522,13 +1513,23 @@ auto Database::changes( } } + if (changeset.erasures.empty() + && delays.empty() + && changeset.additions.empty() + && !progress.has_value()) + { + // There aren't actually any changes for this participant, so we will + // leave it out of the patch. + continue; + } + part_patches.emplace_back( Patch::Participant{ - p.first, + p, state.tracker->last_known_version(), - Change::Erase(std::move(p.second.erasures)), + Change::Erase(std::move(changeset.erasures)), std::move(delays), - Change::Add(state.latest_plan_id, std::move(p.second.additions)), + Change::Add(state.latest_plan_id, std::move(changeset.additions)), std::move(progress) }); } diff --git a/rmf_traffic/src/rmf_traffic/schedule/Participant.cpp b/rmf_traffic/src/rmf_traffic/schedule/Participant.cpp index 2743a792..ffdd4695 100644 --- a/rmf_traffic/src/rmf_traffic/schedule/Participant.cpp +++ b/rmf_traffic/src/rmf_traffic/schedule/Participant.cpp @@ -482,6 +482,12 @@ ItineraryVersion Participant::version() const return _pimpl->_shared->_version; } +//============================================================================== +ProgressVersion Participant::progress_version() const +{ + return _pimpl->_shared->_progress.version; +} + //============================================================================== const ParticipantDescription& Participant::description() const { From cc92e2fc8d9a538e62ba2c2c63d56c3bc9b6e760 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Mon, 20 Nov 2023 04:19:15 +0000 Subject: [PATCH 12/30] Add doors to nav graph information Signed-off-by: Michael X. Grey --- rmf_traffic/include/rmf_traffic/agv/Graph.hpp | 57 +++++++- rmf_traffic/src/rmf_traffic/agv/Graph.cpp | 138 +++++++++++++++++- .../src/rmf_traffic/agv/internal_Graph.hpp | 3 +- 3 files changed, 182 insertions(+), 16 deletions(-) diff --git a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp index 35ae2120..9e2b89c9 100644 --- a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp +++ b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp @@ -67,11 +67,39 @@ class Graph double orientations, Eigen::Vector2d dimensions); + class Implementation; private: + rmf_utils::impl_ptr _pimpl; + }; + using LiftPropertiesPtr = std::shared_ptr; + + class DoorProperties + { + public: + /// Get the name of the door. + const std::string& name() const; + + /// Get the start position of the door. + Eigen::Vector2d start() const; + + /// Get the end position of the door. + Eigen::Vector2d end() const; + + /// Get the name of the map that this door is on. + const std::string& map() const; + + /// Constructor + DoorProperties( + std::string name, + Eigen::Vector2d start, + Eigen::Vector2d end, + std::string map); + class Implementation; + private: rmf_utils::impl_ptr _pimpl; }; - using LiftPropertiesPtr = std::shared_ptr; + using DoorPropertiesPtr = std::shared_ptr; /// Properties assigned to each waypoint (vertex) in the graph class Waypoint @@ -655,14 +683,27 @@ class Graph /// const-qualified lane_from() const Lane* lane_from(std::size_t from_wp, std::size_t to_wp) const; - /// Get the lifts that are known to exist for this graph. - /// NOTE: There is no mechanism to automatically keep known_lifts synced with - /// the actual lifts used by the vertices, so this must be kept in sync - /// manually. - const std::unordered_set& known_lifts() const; + /// Add a known lift to the graph. If this lift has the same name as one + /// previously added, we will continue to use the same pointer as the original + /// and override the properties because lift names are expected to be unique. + LiftPropertiesPtr set_known_lift(LiftProperties lift); + + /// Get all the known lifts. + std::vector all_known_lifts() const; + + /// Find a known lift based on its name. + LiftPropertiesPtr find_known_lift(const std::string& name) const; + + /// Add a known door to the graph. If this door has the same name as one + /// previously added, we will continue to use the same pointer as the original + /// and override the properties because door names are expected to be unique. + DoorPropertiesPtr set_known_door(DoorProperties door); + + /// Get all the known doors. + std::vector all_known_doors() const; - /// Mutable reference to the known lifts - std::unordered_set& known_lifts(); + /// Find a known door based on its name. + DoorPropertiesPtr find_known_door(const std::string& name) const; class Implementation; private: diff --git a/rmf_traffic/src/rmf_traffic/agv/Graph.cpp b/rmf_traffic/src/rmf_traffic/agv/Graph.cpp index 136613a5..23c71901 100644 --- a/rmf_traffic/src/rmf_traffic/agv/Graph.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/Graph.cpp @@ -34,6 +34,13 @@ class Graph::LiftProperties::Implementation double orientation; Eigen::Vector2d half_dimensions; Eigen::Isometry2d tf_inv; + + static void update( + LiftProperties& original, + const LiftProperties& incoming) + { + *original._pimpl = *incoming._pimpl; + } }; //============================================================================== @@ -103,6 +110,57 @@ Graph::LiftProperties::LiftProperties( // Do nothing } +//============================================================================== +class Graph::DoorProperties::Implementation +{ +public: + std::string name; + Eigen::Vector2d start; + Eigen::Vector2d end; + std::string map; +}; + +//============================================================================== +const std::string& Graph::DoorProperties::name() const +{ + return _pimpl->name; +} + +//============================================================================== +Eigen::Vector2d Graph::DoorProperties::start() const +{ + return _pimpl->start; +} + +//============================================================================== +Eigen::Vector2d Graph::DoorProperties::end() const +{ + return _pimpl->end; +} + +//============================================================================== +const std::string& Graph::DoorProperties::map() const +{ + return _pimpl->map; +} + +//============================================================================== +Graph::DoorProperties::DoorProperties( + std::string name, + Eigen::Vector2d start, + Eigen::Vector2d end, + std::string map) +: _pimpl(rmf_utils::make_impl( + Implementation { + std::move(name), + start, + end, + std::move(map) + })) +{ + // Do nothing +} + //============================================================================== class Graph::Waypoint::Implementation { @@ -1073,22 +1131,88 @@ auto Graph::lane_from(std::size_t from_wp, std::size_t to_wp) -> Lane* } //============================================================================== -auto Graph::known_lifts() const -> const std::unordered_set& +auto Graph::lane_from(std::size_t from_wp, std::size_t to_wp) const +-> const Lane* { - return _pimpl->lifts; + return const_cast(*this).lane_from(from_wp, to_wp); } //============================================================================== -auto Graph::known_lifts() -> std::unordered_set& +auto Graph::set_known_lift(LiftProperties lift) -> LiftPropertiesPtr { - return _pimpl->lifts; + const auto [l_it, inserted] = _pimpl->lifts.insert({lift.name(), nullptr}); + if (inserted) + { + l_it->second = std::make_shared(std::move(lift)); + } + else + { + *l_it->second = std::move(lift); + } + + return l_it->second; } //============================================================================== -auto Graph::lane_from(std::size_t from_wp, std::size_t to_wp) const --> const Lane* +auto Graph::all_known_lifts() const -> std::vector { - return const_cast(*this).lane_from(from_wp, to_wp); + std::vector lifts; + lifts.reserve(_pimpl->lifts.size()); + for (const auto& [_, lift] : _pimpl->lifts) + { + lifts.push_back(lift); + } + + return lifts; +} + +//============================================================================== +auto Graph::find_known_lift(const std::string& name) const -> LiftPropertiesPtr +{ + const auto l_it = _pimpl->lifts.find(name); + if (l_it == _pimpl->lifts.end()) + return nullptr; + + return l_it->second; +} + +//============================================================================== +auto Graph::set_known_door(DoorProperties door) -> DoorPropertiesPtr +{ + const auto [d_it, inserted] = _pimpl->doors.insert({door.name(), nullptr}); + if (inserted) + { + d_it->second = std::make_shared(std::move(door)); + } + else + { + *d_it->second = std::move(door); + } + + return d_it->second; +} + +//============================================================================== +auto Graph::all_known_doors() const -> std::vector +{ + std::vector doors; + doors.reserve(_pimpl->doors.size()); + for (const auto& [_, door] : _pimpl->doors) + { + doors.push_back(door); + } + + return doors; +} + +//============================================================================== +auto Graph::find_known_door(const std::string& name) const -> DoorPropertiesPtr +{ + const auto d_it = _pimpl->doors.find(name); + if (d_it == _pimpl->doors.end()) + return nullptr; + + return d_it->second; } } // namespace avg diff --git a/rmf_traffic/src/rmf_traffic/agv/internal_Graph.hpp b/rmf_traffic/src/rmf_traffic/agv/internal_Graph.hpp index fc321f41..0e3372a3 100644 --- a/rmf_traffic/src/rmf_traffic/agv/internal_Graph.hpp +++ b/rmf_traffic/src/rmf_traffic/agv/internal_Graph.hpp @@ -31,7 +31,8 @@ class Graph::Implementation std::vector waypoints; std::vector lanes; std::unordered_map keys; - std::unordered_set lifts; + std::unordered_map lifts; + std::unordered_map doors; // A map from a waypoint index to the set of lanes that can exit from it std::vector> lanes_from; From 9ce2aa02dbfba279fc64ae22173e520f5b958a9c Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Wed, 22 Nov 2023 15:47:58 +0000 Subject: [PATCH 13/30] Fix approach lanes bug Signed-off-by: Michael X. Grey --- .../src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp index d2a86117..a6181517 100644 --- a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp @@ -1499,11 +1499,13 @@ class ScheduledDifferentialDriveExpander const auto time = approach_wp.time(); auto parent = node; std::vector route_from_parent; + std::vector approach_lanes; if (approach_route.trajectory().size() < 2) { // This is just an entry event so we will skip the unnecessary // intermediate node parent = node->parent; + approach_lanes = node->approach_lanes; if (!parent) { // If the top node is a root node, then don't skip it @@ -1524,7 +1526,7 @@ class ScheduledDifferentialDriveExpander Side::Start }, initial_waypoint_index, - {}, + approach_lanes, p0, yaw, time, From a57591de2c88b6dbba00b2e2a594ea793183024e Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Fri, 24 Nov 2023 12:17:44 +0000 Subject: [PATCH 14/30] Allow an envelope to be specified for is_in_lift Signed-off-by: Michael X. Grey --- rmf_traffic/include/rmf_traffic/agv/Graph.hpp | 5 +++-- rmf_traffic/src/rmf_traffic/agv/Graph.cpp | 8 +++++--- 2 files changed, 8 insertions(+), 5 deletions(-) diff --git a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp index 35ae2120..4274ee6f 100644 --- a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp +++ b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp @@ -57,8 +57,9 @@ class Graph Eigen::Vector2d dimensions() const; /// Get whether the specified position, given in RMF canonical coordinates, - /// is inside the lift. - bool is_in_lift(Eigen::Vector2d position) const; + /// is inside the lift. The envelope will expand the footprint of the lift + /// that is used in the calculation. + bool is_in_lift(Eigen::Vector2d position, double envelope = 0.0) const; /// Constructor LiftProperties( diff --git a/rmf_traffic/src/rmf_traffic/agv/Graph.cpp b/rmf_traffic/src/rmf_traffic/agv/Graph.cpp index 136613a5..8005c14d 100644 --- a/rmf_traffic/src/rmf_traffic/agv/Graph.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/Graph.cpp @@ -61,15 +61,17 @@ Eigen::Vector2d Graph::LiftProperties::dimensions() const } //============================================================================== -bool Graph::LiftProperties::is_in_lift(Eigen::Vector2d position) const +bool Graph::LiftProperties::is_in_lift( + Eigen::Vector2d position, + double envelope) const { Eigen::Vector2d p_local = _pimpl->tf_inv * position; for (int i = 0; i < 2; ++i) { - if (p_local[i] < -_pimpl->half_dimensions[i]) + if (p_local[i] < -_pimpl->half_dimensions[i] - envelope) return false; - if (_pimpl->half_dimensions[i] < p_local[i]) + if (_pimpl->half_dimensions[i] + envelope < p_local[i]) return false; } From 388355d946c3e5a1bc3a4bf7139478bcd3e9d272 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Fri, 24 Nov 2023 13:58:57 +0000 Subject: [PATCH 15/30] Provide function that checks if a line segment intersects a door Signed-off-by: Michael X. Grey --- rmf_traffic/include/rmf_traffic/agv/Graph.hpp | 6 ++ rmf_traffic/src/rmf_traffic/agv/Graph.cpp | 71 +++++++++++++++++++ 2 files changed, 77 insertions(+) diff --git a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp index 42d18500..64e8ebd5 100644 --- a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp +++ b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp @@ -89,6 +89,12 @@ class Graph /// Get the name of the map that this door is on. const std::string& map() const; + /// Check if the line formed by p0 -> p1 intersects this door. + bool intersects( + Eigen::Vector2d p0, + Eigen::Vector2d p1, + double envelope = 0.0) const; + /// Constructor DoorProperties( std::string name, diff --git a/rmf_traffic/src/rmf_traffic/agv/Graph.cpp b/rmf_traffic/src/rmf_traffic/agv/Graph.cpp index aa5bb25e..8bce988e 100644 --- a/rmf_traffic/src/rmf_traffic/agv/Graph.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/Graph.cpp @@ -146,6 +146,77 @@ const std::string& Graph::DoorProperties::map() const return _pimpl->map; } +//============================================================================== +namespace { +double distance_from_point_to_segment( + Eigen::Vector2d q, + Eigen::Vector2d p0, + Eigen::Vector2d p1) +{ + const double endpoint_distance = std::min((q - p0).norm(), (q - p1).norm()); + const auto L = (p1 - p0).norm(); + if (L < 1e-3) + { + return endpoint_distance; + } + + const Eigen::Vector2d n = (p1 - p0) / L; + const Eigen::Vector2d v = q - p0; + const Eigen::Vector2d q_proj = (v.dot(n)) * n; + const double ortho_distance = (q - q_proj).norm(); + return std::min(ortho_distance, endpoint_distance); +} +} // anonymous namespace + +//============================================================================== +bool Graph::DoorProperties::intersects( + Eigen::Vector2d p0, + Eigen::Vector2d p1, + double envelope) const +{ + const auto q0 = _pimpl->start; + const auto q1 = _pimpl->end; + for (const auto test : std::vector>{ + [&]{ return distance_from_point_to_segment(p0, q0, q1); }, + [&]{ return distance_from_point_to_segment(p1, q0, q1); }, + [&]{ return distance_from_point_to_segment(q0, p0, p1); }, + [&]{ return distance_from_point_to_segment(_pimpl->end, p0, p1); } + }) + { + const double distance = test(); + if (distance <= envelope) + return true; + } + + // If none of the endpoints are within range of the other lines, then the only + // way for an intersection to exist is if the lines truly cross each other. + const double det = (p0.x() - p1.x()) * (q0.y() - q1.y()) + - (p0.y() - p1.y()) * (q0.x() - q1.x()); + + if (std::abs(det) < 1e-8) + { + // The lines are essentially parallel and their endpoints aren't close + // enough, so there is no intersection. + return false; + } + + const double t = ( (p0.x() - q0.x()) * (q0.y() - q1.y()) + - (p0.y() - q0.y()) * (q0.x() - q1.x()) ) + / det; + + if (t < 0.0 || 1.0 < t) + return false; + + const double u = ( (p0.x() - q0.x()) * (p0.y() - p1.y()) + - (p0.y() - q0.y()) * (p0.x() - p1.x()) ) + / det; + + if (u < 0.0 || 1.0 < u) + return false; + + return true; +} + //============================================================================== Graph::DoorProperties::DoorProperties( std::string name, From a4a42f32c41266cb5accad3382b4ca5c43e9c031 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Sat, 25 Nov 2023 11:40:34 +0000 Subject: [PATCH 16/30] Introduce merge radius for graph waypoints Signed-off-by: Michael X. Grey --- rmf_traffic/include/rmf_traffic/agv/Graph.hpp | 8 +++++ rmf_traffic/src/rmf_traffic/agv/Graph.cpp | 15 ++++++++++ rmf_traffic/src/rmf_traffic/agv/Planner.cpp | 29 +++++++++++++++++-- 3 files changed, 49 insertions(+), 3 deletions(-) diff --git a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp index 64e8ebd5..4f9d162b 100644 --- a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp +++ b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp @@ -208,6 +208,14 @@ class Graph /// empty string will disasscoiate the waypoint from any mutex group. Waypoint& set_in_mutex_group(std::string group_name); + /// Get a merge radius specific to this waypoint, if it has one. The radius + /// indicates that any robot within this distance of the waypoint can merge + /// onto this waypoint. + std::optional merge_radius() const; + + /// Set the merge radius specific to this waypoint. + Waypoint& set_merge_radius(std::optional valeu); + class Implementation; private: Waypoint(); diff --git a/rmf_traffic/src/rmf_traffic/agv/Graph.cpp b/rmf_traffic/src/rmf_traffic/agv/Graph.cpp index 8bce988e..82f5a892 100644 --- a/rmf_traffic/src/rmf_traffic/agv/Graph.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/Graph.cpp @@ -259,6 +259,8 @@ class Graph::Waypoint::Implementation std::string mutex_group = ""; + std::optional merge_radius = std::nullopt; + template static Waypoint make(Args&& ... args) { @@ -419,6 +421,19 @@ auto Graph::Waypoint::set_in_mutex_group(std::string group_name) -> Waypoint& return *this; } +//============================================================================== +std::optional Graph::Waypoint::merge_radius() const +{ + return _pimpl->merge_radius; +} + +//============================================================================== +auto Graph::Waypoint::set_merge_radius(std::optional value) -> Waypoint& +{ + _pimpl->merge_radius = value; + return *this; +} + //============================================================================== Graph::Waypoint::Waypoint() { diff --git a/rmf_traffic/src/rmf_traffic/agv/Planner.cpp b/rmf_traffic/src/rmf_traffic/agv/Planner.cpp index 518a068c..d7cbccf5 100644 --- a/rmf_traffic/src/rmf_traffic/agv/Planner.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/Planner.cpp @@ -1158,6 +1158,8 @@ std::vector compute_plan_starts( const double max_merge_lane_distance, const double min_lane_length) { + std::stringstream ss; + ss << "Computing plan starts:"; const Eigen::Vector2d p_location = {pose[0], pose[1]}; const double start_yaw = pose[2]; @@ -1196,12 +1198,19 @@ std::vector compute_plan_starts( // This "lane" is either two points stacked very close or is moving // vertically through a lift. + const double merge_dist = std::max( + wp0.merge_radius().value_or(max_merge_lane_distance), + wp1.merge_radius().value_or(max_merge_lane_distance)); + if (lane_length < min_lane_length) { const double dp0 = (p_location - p0).norm(); const double dp1 = (p_location - p1).norm(); - if (dp0 < max_merge_lane_distance || dp1 < max_merge_lane_distance) + if (dp0 < merge_dist || dp1 < merge_dist) { + ss << "\n -- " << __LINE__ << ": " << wp0.name_or_index() << " -> " + << wp1.name_or_index() + << " | " << dp0 << ", " << dp1 << " vs " << merge_dist; starts.emplace_back( Plan::Start( start_time, lane.exit().waypoint_index(), @@ -1221,11 +1230,16 @@ std::vector compute_plan_starts( const double dist_to_entry = p_l.norm(); const std::size_t entry_waypoint_index = lane.entry().waypoint_index(); - if (dist_to_entry < max_merge_lane_distance) + const double merge_dist = + wp0.merge_radius().value_or(max_merge_lane_distance); + if (dist_to_entry < merge_dist) { if (!raw_starts.insert(entry_waypoint_index).second) continue; + ss << "\n -- " << __LINE__ << ": " << wp0.name_or_index() + << " -> " << wp1.name_or_index() + << " " << dist_to_entry << " vs " << merge_dist; starts.emplace_back( Plan::Start( start_time, entry_waypoint_index, start_yaw, p_location)); @@ -1238,11 +1252,16 @@ std::vector compute_plan_starts( const double dist_to_exit = (p_location - p1).norm(); const std::size_t exit_waypoint_index = lane.exit().waypoint_index(); - if (dist_to_exit < max_merge_lane_distance) + const double merge_dist = + wp1.merge_radius().value_or(max_merge_lane_distance); + if (dist_to_exit < merge_dist) { if (!raw_starts.insert(exit_waypoint_index).second) continue; + ss << "\n -- " << __LINE__ << ": " << wp0.name_or_index() + << " -> " << wp1.name_or_index() + << " " << dist_to_exit << " vs " << merge_dist; starts.emplace_back( Plan::Start( start_time, exit_waypoint_index, start_yaw, p_location)); @@ -1257,12 +1276,16 @@ std::vector compute_plan_starts( if (lane_dist < max_merge_lane_distance) { + ss << "\n -- " << __LINE__ << ": " << wp0.name_or_index() + << " -> " << wp1.name_or_index() + << " " << lane_dist << " vs " << max_merge_lane_distance; starts.emplace_back( Plan::Start( start_time, exit_waypoint_index, start_yaw, p_location, i)); } } } + std::cout << ss.str() << std::endl; return starts; } From e26a30fc068a9968bd2dbbcd03d3082f8db7c3df Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Sun, 26 Nov 2023 11:10:05 +0000 Subject: [PATCH 17/30] Filter out bad routes Signed-off-by: Michael X. Grey --- .../agv/planning/DifferentialDrivePlanner.cpp | 105 +++++++++++++++++- 1 file changed, 103 insertions(+), 2 deletions(-) diff --git a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp index a6181517..b6f3f8aa 100644 --- a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp @@ -186,6 +186,21 @@ std::vector reconstruct_nodes( const double rotational_threshold) { auto node_sequence = reconstruct_nodes(finish_node); + std::stringstream ss; + ss << " ---------- Plan Nodes"; + for (const auto& node : node_sequence) + { + ss << "\n -- line " << node->line; + if (node->waypoint.has_value()) + ss << " wp:" << *node->waypoint; + else + ss << " wp:(null)"; + ss << " " << node->position.transpose() << " approach: "; + for (const auto& l : node->approach_lanes) + ss << " " << l; + } + std::cout << ss.str() << std::endl; + std::optional start; if (!node_sequence.empty()) { @@ -297,6 +312,7 @@ std::vector find_dependencies( // There may be duplicate insertions because of event waypoints, but // that's okay. We just use the first relevant plan waypoint and allow the // insertion to fail for the rest. + std::cout << "inserting candidate " << c.route_id << ":" << c.checkpoint_id << std::endl; checkpoint_maps.at(c.route_id).insert({c.checkpoint_id, i}); } } @@ -654,8 +670,48 @@ reconstruct_waypoints( } else { - itinerary.push_back(next_route); - extended_route = &itinerary.back(); + bool merged = false; + if (last_route.trajectory().size() < 2) + { + // The last itinerary did not have enough waypoints, so we should + // discard it. + std::cout << "popping back route " << itinerary.size()-1 << std::endl; + const std::size_t remove = itinerary.size() - 1; + itinerary.pop_back(); + for (auto& candidate : candidates) + { + const auto r_it = std::remove_if( + candidate.waypoint.arrival.begin(), + candidate.waypoint.arrival.end(), + [remove](const Plan::Checkpoint& c) + { + return c.route_id == remove; + }); + candidate.waypoint.arrival.erase( + r_it, + candidate.waypoint.arrival.end()); + } + + if (!itinerary.empty()) + { + Route& last_route = itinerary.back(); + if (next_route.map() == last_route.map()) + { + merged = true; + extended_route = &last_route; + for (const auto& waypoint : next_route.trajectory()) + { + last_route.trajectory().insert(waypoint); + } + } + } + } + + if (!merged) + { + itinerary.push_back(next_route); + extended_route = &itinerary.back(); + } } if (!node->approach_lanes.empty()) @@ -672,14 +728,59 @@ reconstruct_waypoints( candidate.velocity).it->index(); candidate.waypoint.arrival.push_back({itinerary.size()-1, index}); + std::cout << __LINE__ << " push candidate " << candidate.waypoint.arrival.back().route_id + << ":" << candidate.waypoint.arrival.back().checkpoint_id << std::endl; } } candidates.back().waypoint.arrival .push_back({itinerary.size()-1, itinerary.back().trajectory().size()-1}); + std::cout << __LINE__ << " push candidate " << candidates.back().waypoint.arrival.back().route_id + << ":" << candidates.back().waypoint.arrival.back().checkpoint_id << std::endl; } } + std::stringstream ss; + std::vector removals; + for (std::size_t i=0; i < itinerary.size(); ++i) + { + if (itinerary[i].trajectory().size() < 2) + { + ss << "\n -- removing " << i; + removals.push_back(i); + } + } + + for (auto r_it = removals.rbegin(); r_it != removals.rend(); ++r_it) + { + std::size_t remove = *r_it; + itinerary.erase(itinerary.begin() + remove); + for (WaypointCandidate& candidate : candidates) + { + auto c_it = candidate.waypoint.arrival.begin(); + while (c_it != candidate.waypoint.arrival.end()) + { + Plan::Checkpoint& c = *c_it; + if (c.route_id == remove) + { + ss << "\n -- erasing " << c.route_id << ":" << c.checkpoint_id; + candidate.waypoint.arrival.erase(c_it); + continue; + } + + if (c.route_id > remove) + { + ss << "\n -- decrementing " << c.route_id << ":" << c.checkpoint_id; + --c.route_id; + } + + ++c_it; + } + } + } + + std::cout << ss.str() << std::endl; + auto plan_waypoints = find_dependencies( itinerary, candidates, validator, dependency_window, dependency_resolution); From 697bf6af76ec632cc082e43decbd560a4b31ef1a Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Sun, 26 Nov 2023 15:15:36 +0000 Subject: [PATCH 18/30] Use merge radius pairs to constrain lane widths Signed-off-by: Michael X. Grey --- rmf_traffic/src/rmf_traffic/agv/Planner.cpp | 16 +++++++++++----- 1 file changed, 11 insertions(+), 5 deletions(-) diff --git a/rmf_traffic/src/rmf_traffic/agv/Planner.cpp b/rmf_traffic/src/rmf_traffic/agv/Planner.cpp index d7cbccf5..c976a4ae 100644 --- a/rmf_traffic/src/rmf_traffic/agv/Planner.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/Planner.cpp @@ -1191,6 +1191,9 @@ std::vector compute_plan_starts( if (wp0.get_map_name() != map_name && wp1.get_map_name() != map_name) continue; + const std::optional wp0_merge_radius = wp0.merge_radius(); + const std::optional wp1_merge_radius = wp1.merge_radius(); + const Eigen::Vector2d p0 = wp0.get_location(); const Eigen::Vector2d p1 = wp1.get_location(); @@ -1199,8 +1202,8 @@ std::vector compute_plan_starts( // This "lane" is either two points stacked very close or is moving // vertically through a lift. const double merge_dist = std::max( - wp0.merge_radius().value_or(max_merge_lane_distance), - wp1.merge_radius().value_or(max_merge_lane_distance)); + wp0_merge_radius.value_or(max_merge_lane_distance), + wp1_merge_radius.value_or(max_merge_lane_distance)); if (lane_length < min_lane_length) { @@ -1231,7 +1234,7 @@ std::vector compute_plan_starts( const std::size_t entry_waypoint_index = lane.entry().waypoint_index(); const double merge_dist = - wp0.merge_radius().value_or(max_merge_lane_distance); + wp0_merge_radius.value_or(max_merge_lane_distance); if (dist_to_entry < merge_dist) { if (!raw_starts.insert(entry_waypoint_index).second) @@ -1253,7 +1256,7 @@ std::vector compute_plan_starts( const std::size_t exit_waypoint_index = lane.exit().waypoint_index(); const double merge_dist = - wp1.merge_radius().value_or(max_merge_lane_distance); + wp1_merge_radius.value_or(max_merge_lane_distance); if (dist_to_exit < merge_dist) { if (!raw_starts.insert(exit_waypoint_index).second) @@ -1273,8 +1276,11 @@ std::vector compute_plan_starts( { const double lane_dist = (p_l - p_l_projection*pn).norm(); const std::size_t exit_waypoint_index = lane.exit().waypoint_index(); + double merge_dist = max_merge_lane_distance; + if (wp0_merge_radius.has_value() && wp1_merge_radius.has_value()) + merge_dist = std::max(*wp0_merge_radius, *wp1_merge_radius); - if (lane_dist < max_merge_lane_distance) + if (lane_dist < merge_dist) { ss << "\n -- " << __LINE__ << ": " << wp0.name_or_index() << " -> " << wp1.name_or_index() From 53fe350ab92a59c9d21bae16fe0471b56c6faa2f Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Wed, 29 Nov 2023 12:33:37 +0000 Subject: [PATCH 19/30] Remove unnecessary debug output Signed-off-by: Michael X. Grey --- rmf_traffic/include/rmf_traffic/agv/Graph.hpp | 8 ++++-- rmf_traffic/src/rmf_traffic/agv/Planner.cpp | 15 ----------- .../agv/planning/DifferentialDrivePlanner.cpp | 27 ------------------- 3 files changed, 6 insertions(+), 44 deletions(-) diff --git a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp index 4f9d162b..b8707907 100644 --- a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp +++ b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp @@ -29,6 +29,7 @@ #include #include #include +#include namespace rmf_traffic { namespace agv { @@ -468,8 +469,11 @@ class Graph template DerivedExecutor& execute(DerivedExecutor& executor) const { - return static_cast(execute( - static_cast(executor))); + Executor& base_executor = static_cast(executor); + std::cout << "executing " << this << " : " << &executor + << " -> " << &base_executor << std::endl; + + return static_cast(execute(base_executor)); } /// Execute this event diff --git a/rmf_traffic/src/rmf_traffic/agv/Planner.cpp b/rmf_traffic/src/rmf_traffic/agv/Planner.cpp index c976a4ae..5bd91d46 100644 --- a/rmf_traffic/src/rmf_traffic/agv/Planner.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/Planner.cpp @@ -1158,8 +1158,6 @@ std::vector compute_plan_starts( const double max_merge_lane_distance, const double min_lane_length) { - std::stringstream ss; - ss << "Computing plan starts:"; const Eigen::Vector2d p_location = {pose[0], pose[1]}; const double start_yaw = pose[2]; @@ -1211,9 +1209,6 @@ std::vector compute_plan_starts( const double dp1 = (p_location - p1).norm(); if (dp0 < merge_dist || dp1 < merge_dist) { - ss << "\n -- " << __LINE__ << ": " << wp0.name_or_index() << " -> " - << wp1.name_or_index() - << " | " << dp0 << ", " << dp1 << " vs " << merge_dist; starts.emplace_back( Plan::Start( start_time, lane.exit().waypoint_index(), @@ -1240,9 +1235,6 @@ std::vector compute_plan_starts( if (!raw_starts.insert(entry_waypoint_index).second) continue; - ss << "\n -- " << __LINE__ << ": " << wp0.name_or_index() - << " -> " << wp1.name_or_index() - << " " << dist_to_entry << " vs " << merge_dist; starts.emplace_back( Plan::Start( start_time, entry_waypoint_index, start_yaw, p_location)); @@ -1262,9 +1254,6 @@ std::vector compute_plan_starts( if (!raw_starts.insert(exit_waypoint_index).second) continue; - ss << "\n -- " << __LINE__ << ": " << wp0.name_or_index() - << " -> " << wp1.name_or_index() - << " " << dist_to_exit << " vs " << merge_dist; starts.emplace_back( Plan::Start( start_time, exit_waypoint_index, start_yaw, p_location)); @@ -1282,16 +1271,12 @@ std::vector compute_plan_starts( if (lane_dist < merge_dist) { - ss << "\n -- " << __LINE__ << ": " << wp0.name_or_index() - << " -> " << wp1.name_or_index() - << " " << lane_dist << " vs " << max_merge_lane_distance; starts.emplace_back( Plan::Start( start_time, exit_waypoint_index, start_yaw, p_location, i)); } } } - std::cout << ss.str() << std::endl; return starts; } diff --git a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp index b6f3f8aa..36e3ec5e 100644 --- a/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp +++ b/rmf_traffic/src/rmf_traffic/agv/planning/DifferentialDrivePlanner.cpp @@ -186,21 +186,6 @@ std::vector reconstruct_nodes( const double rotational_threshold) { auto node_sequence = reconstruct_nodes(finish_node); - std::stringstream ss; - ss << " ---------- Plan Nodes"; - for (const auto& node : node_sequence) - { - ss << "\n -- line " << node->line; - if (node->waypoint.has_value()) - ss << " wp:" << *node->waypoint; - else - ss << " wp:(null)"; - ss << " " << node->position.transpose() << " approach: "; - for (const auto& l : node->approach_lanes) - ss << " " << l; - } - std::cout << ss.str() << std::endl; - std::optional start; if (!node_sequence.empty()) { @@ -312,7 +297,6 @@ std::vector find_dependencies( // There may be duplicate insertions because of event waypoints, but // that's okay. We just use the first relevant plan waypoint and allow the // insertion to fail for the rest. - std::cout << "inserting candidate " << c.route_id << ":" << c.checkpoint_id << std::endl; checkpoint_maps.at(c.route_id).insert({c.checkpoint_id, i}); } } @@ -675,7 +659,6 @@ reconstruct_waypoints( { // The last itinerary did not have enough waypoints, so we should // discard it. - std::cout << "popping back route " << itinerary.size()-1 << std::endl; const std::size_t remove = itinerary.size() - 1; itinerary.pop_back(); for (auto& candidate : candidates) @@ -728,25 +711,19 @@ reconstruct_waypoints( candidate.velocity).it->index(); candidate.waypoint.arrival.push_back({itinerary.size()-1, index}); - std::cout << __LINE__ << " push candidate " << candidate.waypoint.arrival.back().route_id - << ":" << candidate.waypoint.arrival.back().checkpoint_id << std::endl; } } candidates.back().waypoint.arrival .push_back({itinerary.size()-1, itinerary.back().trajectory().size()-1}); - std::cout << __LINE__ << " push candidate " << candidates.back().waypoint.arrival.back().route_id - << ":" << candidates.back().waypoint.arrival.back().checkpoint_id << std::endl; } } - std::stringstream ss; std::vector removals; for (std::size_t i=0; i < itinerary.size(); ++i) { if (itinerary[i].trajectory().size() < 2) { - ss << "\n -- removing " << i; removals.push_back(i); } } @@ -763,14 +740,12 @@ reconstruct_waypoints( Plan::Checkpoint& c = *c_it; if (c.route_id == remove) { - ss << "\n -- erasing " << c.route_id << ":" << c.checkpoint_id; candidate.waypoint.arrival.erase(c_it); continue; } if (c.route_id > remove) { - ss << "\n -- decrementing " << c.route_id << ":" << c.checkpoint_id; --c.route_id; } @@ -779,8 +754,6 @@ reconstruct_waypoints( } } - std::cout << ss.str() << std::endl; - auto plan_waypoints = find_dependencies( itinerary, candidates, validator, dependency_window, dependency_resolution); From dd567041ddb9bd5290ad20887dc4162fdd8be0b8 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Wed, 29 Nov 2023 15:38:16 +0000 Subject: [PATCH 20/30] Remove another unnecessary debug output Signed-off-by: Michael X. Grey --- rmf_traffic/include/rmf_traffic/agv/Graph.hpp | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp index b8707907..2b1d2b82 100644 --- a/rmf_traffic/include/rmf_traffic/agv/Graph.hpp +++ b/rmf_traffic/include/rmf_traffic/agv/Graph.hpp @@ -469,11 +469,8 @@ class Graph template DerivedExecutor& execute(DerivedExecutor& executor) const { - Executor& base_executor = static_cast(executor); - std::cout << "executing " << this << " : " << &executor - << " -> " << &base_executor << std::endl; - - return static_cast(execute(base_executor)); + return static_cast( + execute(static_cast(executor))); } /// Execute this event From c0b4d252154822f8afcd724d3d6b54be652c345f Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Mon, 4 Dec 2023 17:00:11 +0800 Subject: [PATCH 21/30] Update tests Signed-off-by: Michael X. Grey --- .../test/unit/schedule/test_Database.cpp | 6 ++-- .../test/unit/schedule/test_Mirror.cpp | 35 ++++++++++++++----- 2 files changed, 30 insertions(+), 11 deletions(-) diff --git a/rmf_traffic/test/unit/schedule/test_Database.cpp b/rmf_traffic/test/unit/schedule/test_Database.cpp index 622d73a5..64bb2cb9 100644 --- a/rmf_traffic/test/unit/schedule/test_Database.cpp +++ b/rmf_traffic/test/unit/schedule/test_Database.cpp @@ -200,7 +200,8 @@ SCENARIO("Test Database Conflicts") // query from the start changes = db.changes(query_all, rmf_utils::nullopt); - CHECK(changes.size() == 0); + // Progress will always be reported for each participant in a full update + CHECK(changes.size() == 1); CHECK_FALSE(changes.cull()); CHECK(changes.latest_version() == db.latest_version()); @@ -230,7 +231,8 @@ SCENARIO("Test Database Conflicts") // query from the start changes = db.changes(query_all, rmf_utils::nullopt); - CHECK(changes.size() == 0); + // Progress will always be reported for each participant in a full update + CHECK(changes.size() == 1); CHECK_FALSE(changes.cull()); CHECK(changes.latest_version() == db.latest_version()); diff --git a/rmf_traffic/test/unit/schedule/test_Mirror.cpp b/rmf_traffic/test/unit/schedule/test_Mirror.cpp index d0e6d712..62b3c5bc 100644 --- a/rmf_traffic/test/unit/schedule/test_Mirror.cpp +++ b/rmf_traffic/test/unit/schedule/test_Mirror.cpp @@ -327,10 +327,18 @@ SCENARIO("Testing specialized mirrors") db.changes(query, rmf_utils::nullopt); REQUIRE(changes.size() > 0); - CHECK(changes.size() == 1); - CHECK(changes.begin()->participant_id() == p1.id()); - REQUIRE(changes.begin()->additions().items().size() == 1); - CHECK(changes.begin()->additions().items().begin()->storage_id == 0); + // Progress will always be updated when a full update is asked for + CHECK(changes.size() == 2); + + rmf_traffic::schedule::Patch::const_iterator p_it = changes.begin(); + for (; p_it != changes.end(); ++p_it) + { + if (p_it->participant_id() == p1.id()) + break; + } + REQUIRE(p_it != changes.end()); + REQUIRE(p_it->additions().items().size() == 1); + CHECK(p_it->additions().items().begin()->storage_id == 0); } GIVEN("Query patch with spacetime region overlapping with t2") @@ -356,10 +364,18 @@ SCENARIO("Testing specialized mirrors") db.changes(query, rmf_utils::nullopt); REQUIRE(changes.size() > 0); - CHECK(changes.size() == 1); - CHECK(changes.begin()->participant_id() == p1.id()); - REQUIRE(changes.begin()->additions().items().size() == 1); - CHECK(changes.begin()->additions().items().begin()->storage_id == 1); + // Progress will always be updated when a full update is asked for + CHECK(changes.size() == 2); + + rmf_traffic::schedule::Patch::const_iterator p_it = changes.begin(); + for (; p_it != changes.end(); ++p_it) + { + if (p_it->participant_id() == p1.id()) + break; + } + REQUIRE(p_it != changes.end()); + REQUIRE(p_it->additions().items().size() == 1); + CHECK(p_it->additions().items().begin()->storage_id == 1); } // // COMMENTED DUE TO NON-DETERMINISTIC BEHAVIOR OF FCL @@ -413,7 +429,8 @@ SCENARIO("Testing specialized mirrors") rmf_traffic::schedule::Patch changes = db.changes(query, rmf_utils::nullopt); REQUIRE(changes.size() > 0); - CHECK(changes.size() == 1); + // Progress will always be updated when a full update is asked for + CHECK(changes.size() == 2); CHECK(changes.begin()->participant_id() == p2.id()); REQUIRE(changes.begin()->additions().items().size() == 1); CHECK(changes.begin()->additions().items().begin()->storage_id == 0); From ba3457e6ee86ec9f3d2366c7d4b191978be298a5 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Mon, 11 Dec 2023 05:12:01 +0000 Subject: [PATCH 22/30] Switch to 0.3 of action-ros-ci Signed-off-by: Michael X. Grey --- .github/workflows/asan.yaml | 2 +- .github/workflows/build.yaml | 2 +- .github/workflows/tsan.yaml | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/.github/workflows/asan.yaml b/.github/workflows/asan.yaml index 00eae666..1056d906 100644 --- a/.github/workflows/asan.yaml +++ b/.github/workflows/asan.yaml @@ -16,7 +16,7 @@ jobs: with: required-ros-distributions: foxy - name: build_and_test - uses: ros-tooling/action-ros-ci@v0.2 + uses: ros-tooling/action-ros-ci@v0.3 env: CC: clang -fsanitize-blacklist=/home/runner/work/blacklist.txt CXX: clang++ -fsanitize-blacklist=/home/runner/work/blacklist.txt diff --git a/.github/workflows/build.yaml b/.github/workflows/build.yaml index 6ae74109..804130f0 100644 --- a/.github/workflows/build.yaml +++ b/.github/workflows/build.yaml @@ -33,7 +33,7 @@ jobs: with: required-ros-distributions: ${{ matrix.ros_distribution }} - name: build - uses: ros-tooling/action-ros-ci@v0.2 + uses: ros-tooling/action-ros-ci@v0.3 with: target-ros2-distro: ${{ matrix.ros_distribution }} # build all packages listed in the meta package diff --git a/.github/workflows/tsan.yaml b/.github/workflows/tsan.yaml index 1d36e9d1..d6a67975 100644 --- a/.github/workflows/tsan.yaml +++ b/.github/workflows/tsan.yaml @@ -14,7 +14,7 @@ jobs: with: required-ros-distributions: foxy - name: tsan_build_test - uses: ros-tooling/action-ros-ci@v0.2 + uses: ros-tooling/action-ros-ci@v0.3 id: tsan_build_test env: CC: clang From b6a4ff902434f35d4600f3aaeb50c75f9ad30d94 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Mon, 11 Dec 2023 05:21:17 +0000 Subject: [PATCH 23/30] Bump ros ci versions even more Signed-off-by: Michael X. Grey --- .github/workflows/asan.yaml | 4 ++-- .github/workflows/build.yaml | 4 ++-- .github/workflows/tsan.yaml | 4 ++-- 3 files changed, 6 insertions(+), 6 deletions(-) diff --git a/.github/workflows/asan.yaml b/.github/workflows/asan.yaml index 1056d906..98aaab6b 100644 --- a/.github/workflows/asan.yaml +++ b/.github/workflows/asan.yaml @@ -12,11 +12,11 @@ jobs: - name: create_blacklist run: echo "fun:*Eigen*" > /home/runner/work/blacklist.txt - name: deps - uses: ros-tooling/setup-ros@v0.2 + uses: ros-tooling/setup-ros@0.7.1 with: required-ros-distributions: foxy - name: build_and_test - uses: ros-tooling/action-ros-ci@v0.3 + uses: ros-tooling/action-ros-ci@0.3.5 env: CC: clang -fsanitize-blacklist=/home/runner/work/blacklist.txt CXX: clang++ -fsanitize-blacklist=/home/runner/work/blacklist.txt diff --git a/.github/workflows/build.yaml b/.github/workflows/build.yaml index 804130f0..ca13bf15 100644 --- a/.github/workflows/build.yaml +++ b/.github/workflows/build.yaml @@ -29,11 +29,11 @@ jobs: - name: pwd run: pwd - name: setup ROS environment - uses: ros-tooling/setup-ros@v0.3 + uses: ros-tooling/setup-ros@0.7.1 with: required-ros-distributions: ${{ matrix.ros_distribution }} - name: build - uses: ros-tooling/action-ros-ci@v0.3 + uses: ros-tooling/action-ros-ci@0.3.5 with: target-ros2-distro: ${{ matrix.ros_distribution }} # build all packages listed in the meta package diff --git a/.github/workflows/tsan.yaml b/.github/workflows/tsan.yaml index d6a67975..5afef798 100644 --- a/.github/workflows/tsan.yaml +++ b/.github/workflows/tsan.yaml @@ -10,11 +10,11 @@ jobs: runs-on: ubuntu-20.04 steps: - name: deps - uses: ros-tooling/setup-ros@v0.2 + uses: ros-tooling/setup-ros@0.7.1 with: required-ros-distributions: foxy - name: tsan_build_test - uses: ros-tooling/action-ros-ci@v0.3 + uses: ros-tooling/action-ros-ci@0.3.5 id: tsan_build_test env: CC: clang From be6e5d089837e147567b90ed6cd06145561920b1 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Mon, 11 Dec 2023 06:25:05 +0000 Subject: [PATCH 24/30] Update ROS Distro Signed-off-by: Michael X. Grey --- .github/workflows/asan.yaml | 4 ++-- .github/workflows/build.yaml | 8 ++++---- .github/workflows/tsan.yaml | 4 ++-- rmf_traffic/QUALITY_DECLARATION.md | 2 +- 4 files changed, 9 insertions(+), 9 deletions(-) diff --git a/.github/workflows/asan.yaml b/.github/workflows/asan.yaml index 98aaab6b..14db9391 100644 --- a/.github/workflows/asan.yaml +++ b/.github/workflows/asan.yaml @@ -14,14 +14,14 @@ jobs: - name: deps uses: ros-tooling/setup-ros@0.7.1 with: - required-ros-distributions: foxy + required-ros-distributions: iron - name: build_and_test uses: ros-tooling/action-ros-ci@0.3.5 env: CC: clang -fsanitize-blacklist=/home/runner/work/blacklist.txt CXX: clang++ -fsanitize-blacklist=/home/runner/work/blacklist.txt with: - target-ros2-distro: foxy + target-ros2-distro: iron # build all packages listed in the meta package package-name: | rmf_traffic diff --git a/.github/workflows/build.yaml b/.github/workflows/build.yaml index ca13bf15..032ac91b 100644 --- a/.github/workflows/build.yaml +++ b/.github/workflows/build.yaml @@ -12,12 +12,12 @@ jobs: strategy: matrix: ros_distribution: - - galactic + - iron - rolling include: - # Galactic Geochelone (May 2021 - November 2022) - - docker_image: ubuntu:focal - ros_distribution: galactic + # Iron Irwini (May 2023 - November 2024) + - docker_image: ubuntu:jammy + ros_distribution: iron ros_version: 2 # Rolling Ridley (No End-Of-Life) - docker_image: ubuntu:jammy diff --git a/.github/workflows/tsan.yaml b/.github/workflows/tsan.yaml index 5afef798..ba44de75 100644 --- a/.github/workflows/tsan.yaml +++ b/.github/workflows/tsan.yaml @@ -12,7 +12,7 @@ jobs: - name: deps uses: ros-tooling/setup-ros@0.7.1 with: - required-ros-distributions: foxy + required-ros-distributions: iron - name: tsan_build_test uses: ros-tooling/action-ros-ci@0.3.5 id: tsan_build_test @@ -20,7 +20,7 @@ jobs: CC: clang CXX: clang++ with: - target-ros2-distro: foxy + target-ros2-distro: iron # build all packages listed in the meta package package-name: | rmf_traffic diff --git a/rmf_traffic/QUALITY_DECLARATION.md b/rmf_traffic/QUALITY_DECLARATION.md index 70234309..71139c8d 100644 --- a/rmf_traffic/QUALITY_DECLARATION.md +++ b/rmf_traffic/QUALITY_DECLARATION.md @@ -158,7 +158,7 @@ This is assumed to be **Quality Level 4** due to its provided feature documentat ### Target platforms [6.i] `rmf_traffic` does not support all of the tier 1 platforms as described in [REP-2000](https://www.ros.org/reps/rep-2000.html#support-tiers). -`rmf_traffic` supports ROS Foxy. +`rmf_traffic` supports ROS Iron. ## Security [7] From b0c6ffbdd96bda5ab7eecfd9d151a7f2a97e5734 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Mon, 11 Dec 2023 06:52:49 +0000 Subject: [PATCH 25/30] Update ubuntu version for asan and tsan tests Signed-off-by: Michael X. Grey --- .github/workflows/asan.yaml | 2 +- .github/workflows/tsan.yaml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/asan.yaml b/.github/workflows/asan.yaml index 14db9391..10d9b226 100644 --- a/.github/workflows/asan.yaml +++ b/.github/workflows/asan.yaml @@ -7,7 +7,7 @@ on: jobs: asan: name: asan - runs-on: ubuntu-20.04 + runs-on: ubuntu-22.04 steps: - name: create_blacklist run: echo "fun:*Eigen*" > /home/runner/work/blacklist.txt diff --git a/.github/workflows/tsan.yaml b/.github/workflows/tsan.yaml index ba44de75..4820ca76 100644 --- a/.github/workflows/tsan.yaml +++ b/.github/workflows/tsan.yaml @@ -7,7 +7,7 @@ on: jobs: tsan: name: tsan - runs-on: ubuntu-20.04 + runs-on: ubuntu-22.04 steps: - name: deps uses: ros-tooling/setup-ros@0.7.1 From 3ad880ae0402f552b4d3f2643a1192337944f001 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Mon, 11 Dec 2023 07:26:25 +0000 Subject: [PATCH 26/30] Add humble to the matrix Signed-off-by: Michael X. Grey --- .github/workflows/build.yaml | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/.github/workflows/build.yaml b/.github/workflows/build.yaml index 032ac91b..bf691487 100644 --- a/.github/workflows/build.yaml +++ b/.github/workflows/build.yaml @@ -12,9 +12,14 @@ jobs: strategy: matrix: ros_distribution: + - humble - iron - rolling include: + # Humble Hawksbill (May 2022 - May 2027) + - docker_image: ubuntu:jammy + ros_distribution: humble + ros_version: 2 # Iron Irwini (May 2023 - November 2024) - docker_image: ubuntu:jammy ros_distribution: iron From 22ab63fe38fab6092a0b374c2a0bc845b0dfb030 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Mon, 11 Dec 2023 08:43:34 +0000 Subject: [PATCH 27/30] Replace setup step with docker image Signed-off-by: Michael X. Grey --- .github/workflows/asan.yaml | 6 ++---- .github/workflows/build.yaml | 8 ++++---- .github/workflows/tsan.yaml | 6 ++---- 3 files changed, 8 insertions(+), 12 deletions(-) diff --git a/.github/workflows/asan.yaml b/.github/workflows/asan.yaml index 10d9b226..4d2a709f 100644 --- a/.github/workflows/asan.yaml +++ b/.github/workflows/asan.yaml @@ -8,13 +8,11 @@ jobs: asan: name: asan runs-on: ubuntu-22.04 + container: + image: osrf/ros:iron-desktop-jammy steps: - name: create_blacklist run: echo "fun:*Eigen*" > /home/runner/work/blacklist.txt - - name: deps - uses: ros-tooling/setup-ros@0.7.1 - with: - required-ros-distributions: iron - name: build_and_test uses: ros-tooling/action-ros-ci@0.3.5 env: diff --git a/.github/workflows/build.yaml b/.github/workflows/build.yaml index bf691487..cd890674 100644 --- a/.github/workflows/build.yaml +++ b/.github/workflows/build.yaml @@ -17,19 +17,19 @@ jobs: - rolling include: # Humble Hawksbill (May 2022 - May 2027) - - docker_image: ubuntu:jammy + - ubuntu_distribution: jammy ros_distribution: humble ros_version: 2 # Iron Irwini (May 2023 - November 2024) - - docker_image: ubuntu:jammy + - ubuntu_distribution: jammy ros_distribution: iron ros_version: 2 # Rolling Ridley (No End-Of-Life) - - docker_image: ubuntu:jammy + - ubuntu_distribution: jammy ros_distribution: rolling ros_version: 2 container: - image: ${{ matrix.docker_image }} + image: osrf/ros:${{ matrix.ros_distribution }}-desktop-${{ matrix.ubuntu_distribution }} steps: - name: pwd run: pwd diff --git a/.github/workflows/tsan.yaml b/.github/workflows/tsan.yaml index 4820ca76..f15b61b2 100644 --- a/.github/workflows/tsan.yaml +++ b/.github/workflows/tsan.yaml @@ -8,11 +8,9 @@ jobs: tsan: name: tsan runs-on: ubuntu-22.04 + container: + image: osrf/ros:iron-desktop-jammy steps: - - name: deps - uses: ros-tooling/setup-ros@0.7.1 - with: - required-ros-distributions: iron - name: tsan_build_test uses: ros-tooling/action-ros-ci@0.3.5 id: tsan_build_test From 912b5e05b37ce450fa5ad92768ff4c6bea66969b Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Mon, 11 Dec 2023 09:26:52 +0000 Subject: [PATCH 28/30] Tweak asan and tsan Signed-off-by: Michael X. Grey --- .github/workflows/asan.yaml | 5 ++++- .github/workflows/tsan.yaml | 3 --- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/.github/workflows/asan.yaml b/.github/workflows/asan.yaml index 4d2a709f..fbeee774 100644 --- a/.github/workflows/asan.yaml +++ b/.github/workflows/asan.yaml @@ -12,7 +12,10 @@ jobs: image: osrf/ros:iron-desktop-jammy steps: - name: create_blacklist - run: echo "fun:*Eigen*" > /home/runner/work/blacklist.txt + run: | + mkdir -p ${{ github.workspace }}/ + touch ${{ github.workspace }}/blacklist.txt + echo "fun:*Eigen*" > ${{ github.workspace }}/blacklist.txt - name: build_and_test uses: ros-tooling/action-ros-ci@0.3.5 env: diff --git a/.github/workflows/tsan.yaml b/.github/workflows/tsan.yaml index f15b61b2..ca735cc7 100644 --- a/.github/workflows/tsan.yaml +++ b/.github/workflows/tsan.yaml @@ -14,9 +14,6 @@ jobs: - name: tsan_build_test uses: ros-tooling/action-ros-ci@0.3.5 id: tsan_build_test - env: - CC: clang - CXX: clang++ with: target-ros2-distro: iron # build all packages listed in the meta package From 1947537732c7110edf0e685bf0ecf838a2084517 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Mon, 11 Dec 2023 09:50:48 +0000 Subject: [PATCH 29/30] Install dev tools for asan and tsan Signed-off-by: Michael X. Grey --- .github/workflows/asan.yaml | 3 +++ .github/workflows/tsan.yaml | 5 +++++ 2 files changed, 8 insertions(+) diff --git a/.github/workflows/asan.yaml b/.github/workflows/asan.yaml index fbeee774..a3ab6ca2 100644 --- a/.github/workflows/asan.yaml +++ b/.github/workflows/asan.yaml @@ -16,6 +16,9 @@ jobs: mkdir -p ${{ github.workspace }}/ touch ${{ github.workspace }}/blacklist.txt echo "fun:*Eigen*" > ${{ github.workspace }}/blacklist.txt + + - name: install_clang_and_tools + run: sudo apt update && sudo apt install -y clang clang-tools lld wget python3-pip python3-colcon-coveragepy-result python3-colcon-lcov-result lcov - name: build_and_test uses: ros-tooling/action-ros-ci@0.3.5 env: diff --git a/.github/workflows/tsan.yaml b/.github/workflows/tsan.yaml index ca735cc7..16356ffc 100644 --- a/.github/workflows/tsan.yaml +++ b/.github/workflows/tsan.yaml @@ -11,7 +11,12 @@ jobs: container: image: osrf/ros:iron-desktop-jammy steps: + - name: install_clang_and_tools + run: sudo apt update && sudo apt install -y clang clang-tools lld wget python3-pip python3-colcon-coveragepy-result python3-colcon-lcov-result lcov - name: tsan_build_test + env: + CC: clang + CXX: clang++ uses: ros-tooling/action-ros-ci@0.3.5 id: tsan_build_test with: From de519dbcc63885c6383cb4192c00001dff4c1bb8 Mon Sep 17 00:00:00 2001 From: "Michael X. Grey" Date: Mon, 11 Dec 2023 10:07:22 +0000 Subject: [PATCH 30/30] Fix blacklist for asan Signed-off-by: Michael X. Grey --- .github/workflows/asan.yaml | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/.github/workflows/asan.yaml b/.github/workflows/asan.yaml index a3ab6ca2..3a7d6630 100644 --- a/.github/workflows/asan.yaml +++ b/.github/workflows/asan.yaml @@ -16,14 +16,13 @@ jobs: mkdir -p ${{ github.workspace }}/ touch ${{ github.workspace }}/blacklist.txt echo "fun:*Eigen*" > ${{ github.workspace }}/blacklist.txt - - name: install_clang_and_tools run: sudo apt update && sudo apt install -y clang clang-tools lld wget python3-pip python3-colcon-coveragepy-result python3-colcon-lcov-result lcov - name: build_and_test uses: ros-tooling/action-ros-ci@0.3.5 env: - CC: clang -fsanitize-blacklist=/home/runner/work/blacklist.txt - CXX: clang++ -fsanitize-blacklist=/home/runner/work/blacklist.txt + CC: clang -fsanitize-blacklist=${{ github.workspace }}/blacklist.txt + CXX: clang++ -fsanitize-blacklist=${{ github.workspace }}/blacklist.txt with: target-ros2-distro: iron # build all packages listed in the meta package