Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix(out_of_lane): cherry-pick hotfix #1211

Merged
merged 2 commits into from
Mar 27, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@
#include <lanelet2_core/geometry/Polygon.h>
#include <tf2/utils.h>

#include <algorithm>
#include <vector>

namespace behavior_velocity_planner::out_of_lane
Expand Down Expand Up @@ -58,14 +59,19 @@ std::vector<lanelet::BasicPolygon2d> calculate_path_footprints(
const auto base_footprint = make_base_footprint(params);
std::vector<lanelet::BasicPolygon2d> path_footprints;
path_footprints.reserve(ego_data.path.points.size());
for (auto i = ego_data.first_path_idx; i < ego_data.path.points.size(); ++i) {
double length = 0.0;
const auto range = std::max(params.slow_dist_threshold, params.stop_dist_threshold) +
params.front_offset + params.extra_front_offset;
for (auto i = ego_data.first_path_idx; i < ego_data.path.points.size() && length < range; ++i) {
const auto & path_pose = ego_data.path.points[i].point.pose;
const auto angle = tf2::getYaw(path_pose.orientation);
const auto rotated_footprint = tier4_autoware_utils::rotatePolygon(base_footprint, angle);
lanelet::BasicPolygon2d footprint;
for (const auto & p : rotated_footprint.outer())
footprint.emplace_back(p.x() + path_pose.position.x, p.y() + path_pose.position.y);
path_footprints.push_back(footprint);
if (i + 1 < ego_data.path.points.size())
length += tier4_autoware_utils::calcDistance2d(path_pose, ego_data.path.points[i + 1].point);
}
return path_footprints;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,15 +24,65 @@

namespace behavior_velocity_planner::out_of_lane
{

lanelet::ConstLanelets consecutive_lanelets(
const route_handler::RouteHandler & route_handler, const lanelet::ConstLanelet & lanelet)
{
lanelet::ConstLanelets consecutives = route_handler.getRoutingGraphPtr()->following(lanelet);
const auto previous = route_handler.getRoutingGraphPtr()->previous(lanelet);
consecutives.insert(consecutives.end(), previous.begin(), previous.end());
return consecutives;
}

lanelet::ConstLanelets get_missing_lane_change_lanelets(
lanelet::ConstLanelets & path_lanelets, const route_handler::RouteHandler & route_handler)
{
lanelet::ConstLanelets missing_lane_change_lanelets;
const auto & routing_graph = *route_handler.getRoutingGraphPtr();
lanelet::ConstLanelets adjacents;
lanelet::ConstLanelets consecutives;
for (const auto & ll : path_lanelets) {
const auto consecutives_of_ll = consecutive_lanelets(route_handler, ll);
std::copy_if(
consecutives_of_ll.begin(), consecutives_of_ll.end(), std::back_inserter(consecutives),
[&](const auto & l) { return !contains_lanelet(consecutives, l.id()); });
const auto adjacents_of_ll = routing_graph.besides(ll);
std::copy_if(
adjacents_of_ll.begin(), adjacents_of_ll.end(), std::back_inserter(adjacents),
[&](const auto & l) { return !contains_lanelet(adjacents, l.id()); });
}
std::copy_if(
adjacents.begin(), adjacents.end(), std::back_inserter(missing_lane_change_lanelets),
[&](const auto & l) {
return !contains_lanelet(missing_lane_change_lanelets, l.id()) &&
!contains_lanelet(path_lanelets, l.id()) && contains_lanelet(consecutives, l.id());
});
return missing_lane_change_lanelets;
}

lanelet::ConstLanelets calculate_path_lanelets(
const EgoData & ego_data, const route_handler::RouteHandler & route_handler)
{
const auto lanelet_map_ptr = route_handler.getLaneletMapPtr();
lanelet::ConstLanelets path_lanelets;
lanelet::BasicLineString2d path_ls;
for (const auto & p : ego_data.path.points)
path_ls.emplace_back(p.point.pose.position.x, p.point.pose.position.y);
for (const auto & dist_lanelet :
lanelet::geometry::findWithin2d(lanelet_map_ptr->laneletLayer, path_ls)) {
if (!contains_lanelet(path_lanelets, dist_lanelet.second.id()))
path_lanelets.push_back(dist_lanelet.second);
}
const auto missing_lanelets = get_missing_lane_change_lanelets(path_lanelets, route_handler);
path_lanelets.insert(path_lanelets.end(), missing_lanelets.begin(), missing_lanelets.end());
return path_lanelets;
}

lanelet::ConstLanelets calculate_ignored_lanelets(
const EgoData & ego_data, const lanelet::ConstLanelets & path_lanelets,
const route_handler::RouteHandler & route_handler, const PlannerParam & params)
{
lanelet::ConstLanelets ignored_lanelets;
// ignore lanelets that follows path lanelets
for (const auto & path_lanelet : path_lanelets)
for (const auto & following : route_handler.getRoutingGraphPtr()->following(path_lanelet))
if (!contains_lanelet(path_lanelets, following.id())) ignored_lanelets.push_back(following);
// ignore lanelets directly behind ego
const auto behind =
planning_utils::calculateOffsetPoint2d(ego_data.pose, params.rear_offset, 0.0);
Expand All @@ -55,26 +105,14 @@ lanelet::ConstLanelets calculate_other_lanelets(
const lanelet::BasicPoint2d ego_point(ego_data.pose.position.x, ego_data.pose.position.y);
const auto lanelets_within_range = lanelet::geometry::findWithin2d(
route_handler.getLaneletMapPtr()->laneletLayer, ego_point,
std::max(params.slow_dist_threshold, params.stop_dist_threshold));
const auto is_overlapped_by_path_lanelets = [&](const auto & l) {
for (const auto & path_ll : path_lanelets) {
if (
boost::geometry::overlaps(
path_ll.polygon2d().basicPolygon(), l.polygon2d().basicPolygon()) ||
boost::geometry::within(path_ll.polygon2d().basicPolygon(), l.polygon2d().basicPolygon()) ||
boost::geometry::within(l.polygon2d().basicPolygon(), path_ll.polygon2d().basicPolygon())) {
return true;
}
}
return false;
};
std::max(params.slow_dist_threshold, params.stop_dist_threshold) + params.front_offset +
params.extra_front_offset);
for (const auto & ll : lanelets_within_range) {
if (std::string(ll.second.attributeOr(lanelet::AttributeName::Subtype, "none")) != "road")
continue;
const auto is_path_lanelet = contains_lanelet(path_lanelets, ll.second.id());
const auto is_ignored_lanelet = contains_lanelet(ignored_lanelets, ll.second.id());
if (!is_path_lanelet && !is_ignored_lanelet && !is_overlapped_by_path_lanelets(ll.second))
other_lanelets.push_back(ll.second);
if (!is_path_lanelet && !is_ignored_lanelet) other_lanelets.push_back(ll.second);
}
return other_lanelets;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,21 @@ inline bool contains_lanelet(const lanelet::ConstLanelets & lanelets, const lane
return l.id() == id;
}) != lanelets.end();
};

/// @brief calculate lanelets crossed by the ego path
/// @details calculated from the ids of the path msg, the lanelets containing path points
/// @param [in] ego_data data about the ego vehicle
/// @param [in] route_handler route handler
/// @return lanelets crossed by the ego vehicle
lanelet::ConstLanelets calculate_path_lanelets(
const EgoData & ego_data, const route_handler::RouteHandler & route_handler);
/// @brief calculate lanelets that may not be crossed by the path but may be overlapped during a
/// lane change
/// @param [in] path_lanelets lanelets driven by the ego vehicle
/// @param [in] route_handler route handler
/// @return lanelets that may be overlapped by a lane change (and are not already in path_lanelets)
lanelet::ConstLanelets get_missing_lane_change_lanelets(
lanelet::ConstLanelets & path_lanelets, const route_handler::RouteHandler & route_handler);
/// @brief calculate lanelets that should be ignored
/// @param [in] ego_data data about the ego vehicle
/// @param [in] path_lanelets lanelets driven by the ego vehicle
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ Overlap calculate_overlap(
Overlap overlap;
const auto & left_bound = lanelet.leftBound2d().basicLineString();
const auto & right_bound = lanelet.rightBound2d().basicLineString();
// TODO(Maxime): these intersects (for each path footprint, for each lanelet) are too expensive
const auto overlap_left = boost::geometry::intersects(path_footprint, left_bound);
const auto overlap_right = boost::geometry::intersects(path_footprint, right_bound);

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -66,7 +66,7 @@ bool OutOfLaneModule::modifyPathVelocity(PathWithLaneId * path, StopReason * sto
ego_data.pose = planner_data_->current_odometry->pose;
ego_data.path.points = path->points;
ego_data.first_path_idx =
motion_utils::findNearestSegmentIndex(ego_data.path.points, ego_data.pose.position);
motion_utils::findNearestSegmentIndex(path->points, ego_data.pose.position);
motion_utils::removeOverlapPoints(ego_data.path.points);
ego_data.velocity = planner_data_->current_velocity->twist.linear.x;
ego_data.max_decel = -planner_data_->max_stop_acceleration_threshold;
Expand All @@ -75,13 +75,13 @@ bool OutOfLaneModule::modifyPathVelocity(PathWithLaneId * path, StopReason * sto
const auto path_footprints = calculate_path_footprints(ego_data, params_);
const auto calculate_path_footprints_us = stopwatch.toc("calculate_path_footprints");
// Calculate lanelets to ignore and consider
const auto path_lanelets = planning_utils::getLaneletsOnPath(
ego_data.path, planner_data_->route_handler_->getLaneletMapPtr(),
planner_data_->current_odometry->pose);
stopwatch.tic("calculate_lanelets");
const auto path_lanelets = calculate_path_lanelets(ego_data, *planner_data_->route_handler_);
const auto ignored_lanelets =
calculate_ignored_lanelets(ego_data, path_lanelets, *planner_data_->route_handler_, params_);
const auto other_lanelets = calculate_other_lanelets(
ego_data, path_lanelets, ignored_lanelets, *planner_data_->route_handler_, params_);
const auto calculate_lanelets_us = stopwatch.toc("calculate_lanelets");

debug_data_.footprints = path_footprints;
debug_data_.path_lanelets = path_lanelets;
Expand All @@ -98,6 +98,7 @@ bool OutOfLaneModule::modifyPathVelocity(PathWithLaneId * path, StopReason * sto
});
if (overlapped_lanelet_it != other_lanelets.end()) {
debug_data_.current_overlapped_lanelets.push_back(*overlapped_lanelet_it);
// TODO(Maxime): we may want to just add the overlapped lane to the ignored lanelets
RCLCPP_DEBUG(logger_, "Ego is already overlapping a lane, skipping the module ()\n");
return true;
}
Expand Down Expand Up @@ -183,13 +184,15 @@ bool OutOfLaneModule::modifyPathVelocity(PathWithLaneId * path, StopReason * sto
RCLCPP_DEBUG(
logger_,
"Total time = %2.2fus\n"
"\tcalculate_lanelets = %2.0fus\n"
"\tcalculate_path_footprints = %2.0fus\n"
"\tcalculate_overlapping_ranges = %2.0fus\n"
"\tcalculate_decisions = %2.0fus\n"
"\tcalc_slowdown_points = %2.0fus\n"
"\tinsert_slowdown_points = %2.0fus\n",
total_time_us, calculate_path_footprints_us, calculate_overlapping_ranges_us,
calculate_decisions_us, calc_slowdown_points_us, insert_slowdown_points_us);
total_time_us, calculate_lanelets_us, calculate_path_footprints_us,
calculate_overlapping_ranges_us, calculate_decisions_us, calc_slowdown_points_us,
insert_slowdown_points_us);
return true;
}

Expand Down
Loading