17#include <rclcpp/rclcpp.hpp>
22#include <boost/uuid/uuid_generators.hpp>
23#include <boost/uuid/uuid_io.hpp>
24#include <lanelet2_core/geometry/Point.h>
25#include <trajectory_utils/trajectory_utils.hpp>
26#include <trajectory_utils/conversions/conversions.hpp>
28#include <carma_ros2_utils/carma_lifecycle_node.hpp>
30#include <Eigen/Geometry>
34#include <carma_v2x_msgs/msg/location_ecef.hpp>
35#include <carma_v2x_msgs/msg/trajectory.hpp>
36#include <carma_v2x_msgs/msg/plan_type.hpp>
41using oss = std::ostringstream;
49 : nh_(nh), wm_(wm), config_(config),mobility_response_publisher_(mobility_response_publisher), lc_status_publisher_(lc_status_publisher)
56 return rclcpp::Time(trajectory.trajectory_points.back().target_time).seconds();
61 return rclcpp::Time(trajectory.trajectory_points.front().target_time).seconds();
71 return (rclcpp::Time(trajectory.back().header.stamp) - rclcpp::Time(trajectory.front().header.stamp)).seconds();
76 std::vector<std::pair<int, lanelet::BasicPoint2d>> intersection_points;
77 boost::geometry::model::linestring<lanelet::BasicPoint2d> self_traj;
78 for (
auto tpp:self_trajectory)
80 boost::geometry::append(self_traj, tpp);
83 for (
size_t i=0;
i<incoming_trajectory.size();
i++)
85 double res = boost::geometry::distance(incoming_trajectory.at(
i), self_traj);
89 intersection_points.push_back(std::make_pair(
i, incoming_trajectory.at(
i)));
92 return intersection_points;
97 carma_planning_msgs::msg::TrajectoryPlan trajectory_plan;
98 std::vector<lanelet::BasicPoint2d> map_points;
103 auto curr_point = ecef_trajectory.location;
105 for (
size_t i = 0;
i<ecef_trajectory.offsets.size();
i++)
107 lanelet::BasicPoint2d offset_point;
108 curr_point.ecef_x += ecef_trajectory.offsets.at(
i).offset_x;
109 curr_point.ecef_y += ecef_trajectory.offsets.at(
i).offset_y;
110 curr_point.ecef_z += ecef_trajectory.offsets.at(
i).offset_z;
114 map_points.push_back(offset_point);
124 throw std::invalid_argument(
"No map projector available for ecef conversion");
127 lanelet::BasicPoint3d map_point =
map_projector_->projectECEF( {
static_cast<double>(ecef_point.ecef_x)/100.0,
static_cast<double>(ecef_point.ecef_y)/100.0,
static_cast<double>(ecef_point.ecef_z)/100.0 } , 1);
129 return lanelet::traits::to2D(map_point);
136 carma_v2x_msgs::msg::MobilityResponse out_mobility_response;
138 out_mobility_response.m_header.recipient_id = resp_recipient_id;
139 out_mobility_response.m_header.sender_bsm_id =
host_bsm_id_;
140 out_mobility_response.m_header.plan_id = req_plan_id;
141 out_mobility_response.m_header.timestamp =
nh_->now().seconds()*1000;
146 out_mobility_response.is_accepted =
true;
148 else out_mobility_response.is_accepted =
false;
150 return out_mobility_response;
156 carma_v2x_msgs::msg::MobilityRequest incoming_request = *msg;
157 carma_planning_msgs::msg::LaneChangeStatus lc_status_msg;
158 if (incoming_request.strategy ==
"carma/cooperative-lane-change")
161 RCLCPP_ERROR(
nh_->get_logger(),
"Cannot process mobility request as map projection is not yet set!");
164 if (incoming_request.plan_type.type == carma_v2x_msgs::msg::PlanType::CHANGE_LANE_LEFT || incoming_request.plan_type.type == carma_v2x_msgs::msg::PlanType::CHANGE_LANE_RIGHT)
166 RCLCPP_DEBUG(
nh_->get_logger(),
"Cooperative Lane Change Request Received");
167 lc_status_msg.status = carma_planning_msgs::msg::LaneChangeStatus::REQUEST_RECEIVED;
168 lc_status_msg.description =
"Received lane merge request";
172 RCLCPP_DEBUG(
nh_->get_logger(),
"CLC Request correctly received");
176 std::string req_sender_id = incoming_request.m_header.sender_id;
177 std::string req_plan_id = incoming_request.m_header.plan_id;
179 carma_v2x_msgs::msg::LocationECEF ecef_location = incoming_request.location;
180 carma_v2x_msgs::msg::Trajectory incoming_trajectory = incoming_request.trajectory;
181 std::string req_strategy_params = incoming_request.strategy_params;
183 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"received urgency: " <<
clc_urgency_);
186 using boost::property_tree::ptree;
188 std::istringstream strstream(req_strategy_params);
189 boost::property_tree::json_parser::read_json(strstream, pt);
190 int req_traj_speed_full = pt.get<
int>(
"s");
191 int req_traj_fractional = pt.get<
int>(
"f");
192 int start_lanelet_id = pt.get<
int>(
"sl");
193 int end_lanelet_id = pt.get<
int>(
"el");
194 double req_traj_speed =
static_cast<double>(req_traj_speed_full) +
static_cast<double>(req_traj_fractional)/10.0;
195 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"req_traj_speed" << req_traj_speed);
196 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"start_lanelet_id" << start_lanelet_id);
197 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"end_lanelet_id" << end_lanelet_id);
199 std::vector<lanelet::BasicPoint2d> req_traj_plan = {};
203 double req_expiration_sec =
static_cast<double>(incoming_request.expiration);
204 double current_time_sec =
nh_->now().seconds();
206 bool response_to_clc_req =
false;
208 double req_plan_time = req_expiration_sec - current_time_sec;
209 double req_timestamp =
static_cast<double>(incoming_request.m_header.timestamp) / 1000.0 - current_time_sec;
216 lc_status_msg.status = carma_planning_msgs::msg::LaneChangeStatus::REQUEST_ACCEPTED;
217 lc_status_msg.description =
"Accepted lane merge request";
218 response_to_clc_req =
true;
219 RCLCPP_DEBUG(
nh_->get_logger(),
"CLC accepted");
223 lc_status_msg.status = carma_planning_msgs::msg::LaneChangeStatus::REQUEST_REJECTED;
224 lc_status_msg.description =
"Rejected lane merge request";
225 response_to_clc_req =
false;
226 RCLCPP_DEBUG(
nh_->get_logger(),
"CLC rejected");
228 carma_v2x_msgs::msg::MobilityResponse outgoing_response =
compose_mobility_response(req_sender_id, req_plan_id, response_to_clc_req);
230 lc_status_msg.status = carma_planning_msgs::msg::LaneChangeStatus::RESPONSE_SENT;
231 RCLCPP_DEBUG(
nh_->get_logger(),
"response sent");
249 carma_v2x_msgs::msg::BSMCoreData bsm_core_ = msg->core_data;
254 carma_planning_msgs::srv::PlanTrajectory::Request::SharedPtr req,
255 carma_planning_msgs::srv::PlanTrajectory::Response::SharedPtr resp)
257 RCLCPP_DEBUG(
nh_->get_logger(),
"Yield_plugin was called!");
258 if (req->initial_trajectory_plan.trajectory_points.size() < 2){
259 throw std::invalid_argument(
"Empty Trajectory received by Yield");
261 rclcpp::Clock system_clock(RCL_SYSTEM_TIME);
262 rclcpp::Time start_time = system_clock.now();
264 carma_planning_msgs::msg::TrajectoryPlan original_trajectory = req->initial_trajectory_plan;
265 carma_planning_msgs::msg::TrajectoryPlan yield_trajectory;
268 if (req->vehicle_state.longitudinal_vel >
EPSILON &&
271 RCLCPP_DEBUG(
nh_->get_logger(),
"Using last committed trajectory to stopping");
272 lanelet::BasicPoint2d veh_pos(req->vehicle_state.x_pos_global,
273 req->vehicle_state.y_pos_global);
277 size_t idx_to_start_new_traj =
279 updated_trajectory.trajectory_points,
283 if (!updated_trajectory.trajectory_points.empty()) {
284 updated_trajectory.trajectory_points =
285 std::vector<carma_planning_msgs::msg::TrajectoryPlanPoint>
286 (updated_trajectory.trajectory_points.begin() + idx_to_start_new_traj,
287 updated_trajectory.trajectory_points.end());
292 rclcpp::Time end_time = system_clock.now();
294 auto duration = end_time - start_time;
295 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
307 double initial_velocity = req->vehicle_state.longitudinal_vel;
310 if (initial_velocity <
EPSILON)
312 initial_velocity = original_trajectory.initial_longitudinal_velocity;
318 RCLCPP_DEBUG(
nh_->get_logger(),
"First time stopped to prevent collision: %f",
326 RCLCPP_DEBUG(
nh_->get_logger(),
"Only consider high urgency clc");
329 RCLCPP_DEBUG(
nh_->get_logger(),
"Yield for CLC. We haven't received an updated negotiation this timestep");
335 RCLCPP_DEBUG(
nh_->get_logger(),
"unreliable CLC communication, switching to object avoidance");
341 RCLCPP_DEBUG(
nh_->get_logger(),
"Yield for object avoidance");
350 resp->trajectory_plan = original_trajectory;
358 yield_trajectory.header.frame_id =
"map";
359 yield_trajectory.header.stamp =
nh_->now();
360 yield_trajectory.trajectory_id = original_trajectory.trajectory_id;
361 resp->trajectory_plan = yield_trajectory;
364 catch(
const std::runtime_error& e) {
365 RCLCPP_WARN_STREAM(
nh_->get_logger(),
"Yield Plugin failed to plan trajectory due to known negative time issue: " << e.what());
366 RCLCPP_WARN_STREAM(
nh_->get_logger(),
"Returning the original trajectory, and retrying at the next call.");
367 resp->trajectory_plan = original_trajectory;
370 rclcpp::Time end_time = system_clock.now();
372 auto duration = end_time - start_time;
373 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
379 carma_planning_msgs::msg::TrajectoryPlan cooperative_trajectory;
381 double initial_pos = 0;
383 double initial_velocity = current_speed;
387 std::vector<lanelet::BasicPoint2d> host_traj_points = {};
388 for (
size_t i=0;
i<original_tp.trajectory_points.size();
i++)
390 lanelet::BasicPoint2d traj_point;
391 traj_point.x() = original_tp.trajectory_points.at(
i).x;
392 traj_point.y() = original_tp.trajectory_points.at(
i).y;
393 host_traj_points.push_back(traj_point);
397 if (!intersection_points.empty())
399 lanelet::BasicPoint2d intersection_point = intersection_points[0].second;
400 double dx = original_tp.trajectory_points[0].x - intersection_point.x();
401 double dy = original_tp.trajectory_points[0].y - intersection_point.y();
404 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"digital_gap: " << digital_gap);
406 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"Goal position (goal_pos): " << goal_pos);
409 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"Collision time: " << collision_time);
410 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"intersection num: " << intersection_points[0].first);
411 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"Planning time: " << planning_time);
416 double incoming_trajectory_speed = sqrt(dx2*dx2 + dy2*dy2)/(intersection_points[0].first *
ecef_traj_timestep_);
418 goal_velocity = std::min(goal_velocity, incoming_trajectory_speed);
421 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"goal_velocity: " << goal_velocity);
422 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"incoming_trajectory_speed: " << incoming_trajectory_speed);
424 if (planning_time > min_time)
428 cooperative_trajectory =
generate_JMT_trajectory(original_tp, initial_pos, goal_pos, initial_velocity, goal_velocity, planning_time, original_max_speed);
433 RCLCPP_DEBUG(
nh_->get_logger(),
"The incoming requested trajectory is rejected, due to insufficient gap");
434 cooperative_trajectory = original_tp;
441 RCLCPP_DEBUG(
nh_->get_logger(),
"The incoming requested trajectory does not overlap with host vehicle's trajectory");
442 cooperative_trajectory = original_tp;
445 return cooperative_trajectory;
450 double smallest_time_step = std::numeric_limits<double>::infinity();
451 for (
size_t i = 0;
i < original_tp.trajectory_points.size() - 1;
i ++)
453 smallest_time_step = std::min(smallest_time_step,
454 (rclcpp::Time(original_tp.trajectory_points.at(
i + 1).target_time)
455 - rclcpp::Time(original_tp.trajectory_points.at(
i).target_time)).seconds());
457 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
"smallest_time_step: " << smallest_time_step);
459 return smallest_time_step;
463 double initial_velocity,
double goal_velocity,
double planning_time,
double original_max_speed)
465 carma_planning_msgs::msg::TrajectoryPlan jmt_trajectory;
466 std::vector<carma_planning_msgs::msg::TrajectoryPlanPoint> jmt_trajectory_points;
467 jmt_trajectory_points.push_back(original_tp.trajectory_points[0]);
470 std::vector<double> calculated_speeds = {};
471 std::vector<double> new_relative_downtracks = {};
472 new_relative_downtracks.push_back(0.0);
473 calculated_speeds.push_back(initial_velocity);
474 double new_traj_accumulated_downtrack = 0.0;
475 double original_traj_accumulated_downtrack = original_traj_relative_downtracks.at(1);
479 const double initial_time = 0;
480 double initial_accel = 0;
483 initial_accel = (initial_velocity -
last_speed_.value()) /
486 if (!std::isnormal(initial_accel))
488 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
"Detecting nan initial_accel set to 0");
492 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
"Detecting initial_accel: " << initial_accel
493 <<
", initial_velocity:" << initial_velocity
495 <<
", nh_->now(): " <<
nh_->now().seconds()
499 const double goal_accel = 0;
500 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
"Following parameters used for JMT: "
501 "\ninitial_pos: " << initial_pos <<
502 "\ngoal_pos: " << goal_pos <<
503 "\ninitial_velocity: " << initial_velocity <<
504 "\ngoal_velocity: " << goal_velocity <<
505 "\ninitial_accel: " << initial_accel <<
506 "\ngoal_accel: " << goal_accel <<
507 "\nplanning_time: " << planning_time <<
508 "\noriginal_max_speed: " << original_max_speed);
511 std::vector<double> polynomial_coefficients = quintic_coefficient_calculator::quintic_coefficient_calculator(initial_pos,
519 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
"Used original_max_speed: " << original_max_speed);
520 for (
size_t i = 0;
i < polynomial_coefficients.size();
i++) {
521 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
"Coefficient " <<
i <<
": " << polynomial_coefficients[
i]);
524 int new_traj_idx = 1;
525 int original_traj_idx = 1;
526 while (new_traj_accumulated_downtrack < goal_pos -
EPSILON && original_traj_idx < original_traj_relative_downtracks.size())
528 const double target_time = new_traj_idx * smallest_time_step;
529 const double downtrack_at_target_time =
polynomial_calc(polynomial_coefficients, target_time);
530 double velocity_at_target_time =
polynomial_calc_d(polynomial_coefficients, target_time);
532 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
"Calculated speed velocity_at_target_time: " << velocity_at_target_time
533 <<
", downtrack_at_target_time: "<< downtrack_at_target_time <<
", target_time: " << target_time);
537 if (velocity_at_target_time < 0.0)
543 velocity_at_target_time = std::clamp(velocity_at_target_time, 0.0, original_max_speed);
546 if (downtrack_at_target_time >= original_traj_accumulated_downtrack)
548 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
"Picked calculated speed velocity_at_target_time: " << velocity_at_target_time
549 <<
", downtrack_at_target_time: "<< downtrack_at_target_time <<
", target_time: " << target_time);
552 calculated_speeds.push_back(velocity_at_target_time);
553 original_traj_accumulated_downtrack += original_traj_relative_downtracks.at(original_traj_idx);
554 original_traj_idx ++;
556 new_traj_accumulated_downtrack = downtrack_at_target_time;
563 std::fill_n(std::back_inserter(calculated_speeds),
564 std::size(original_traj_relative_downtracks) - std::size(calculated_speeds),
569 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
"filtered_speeds size: " << filtered_speeds.size());
570 for (
const auto& speed : filtered_speeds)
572 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
"filtered speed: " << speed);
575 double prev_speed = filtered_speeds.at(0);
578 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
"start speed: " << prev_speed <<
", target_time: " <<
std::to_string(rclcpp::Time(original_tp.trajectory_points[0].target_time).seconds()));
580 for(
size_t i = 1;
i < original_tp.trajectory_points.size();
i++)
582 carma_planning_msgs::msg::TrajectoryPlanPoint jmt_tpp = original_tp.trajectory_points.at(
i);
586 double current_speed = goal_velocity;
588 if (
i < filtered_speeds.size())
590 current_speed = filtered_speeds.at(
i);
601 const double dt = (2 * original_traj_relative_downtracks.at(
i)) / (current_speed + prev_speed);
602 jmt_tpp.target_time = rclcpp::Time(jmt_trajectory_points.back().target_time) + rclcpp::Duration::from_nanoseconds(dt*1e9);
611 jmt_tpp.target_time = rclcpp::Time(jmt_trajectory_points.back().target_time) + rclcpp::Duration::from_nanoseconds(6000 * 1e9);
612 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
"Zero speed = x: " << jmt_tpp.x <<
", y:" << jmt_tpp.y
613 <<
", t:" <<
std::to_string(rclcpp::Time(jmt_tpp.target_time).seconds())
614 <<
", prev_speed: " << prev_speed <<
", current_speed: " << current_speed);
618 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
"Non-zero speed = x: " << jmt_tpp.x <<
", y:" << jmt_tpp.y
619 <<
", t:" <<
std::to_string(rclcpp::Time(jmt_tpp.target_time).seconds())
620 <<
", prev_speed: " << prev_speed <<
", current_speed: " << current_speed);
623 jmt_trajectory_points.push_back(jmt_tpp);
624 double insta_decel = (current_speed - prev_speed) / (rclcpp::Time(jmt_trajectory_points.at(
i).target_time).seconds() - rclcpp::Time(jmt_trajectory_points.at(
i - 1).target_time).seconds());
625 RCLCPP_DEBUG_STREAM(rclcpp::get_logger(
"yield_plugin"),
"insta_decel: " << insta_decel );
626 prev_speed = current_speed;
629 jmt_trajectory.header = original_tp.header;
630 jmt_trajectory.trajectory_id = original_tp.trajectory_id;
631 jmt_trajectory.trajectory_points = jmt_trajectory_points;
632 jmt_trajectory.initial_longitudinal_velocity = initial_velocity;
633 return jmt_trajectory;
637 const std::vector<carma_perception_msgs::msg::PredictedState>& trajectory2,
double collision_radius,
double trajectory1_max_speed)
641 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"Starting a new collision detection, trajectory size: "
642 << trajectory1.trajectory_points.size() <<
". prediction size: " << trajectory2.size());
645 bool on_route =
false;
646 int on_route_idx = 0;
649 const auto traj2_speed{std::hypot(trajectory2.front().predicted_velocity.linear.x,
650 trajectory2.front().predicted_velocity.linear.y)};
653 if (trajectory2.size() < 2)
655 throw std::invalid_argument(
"Object on ther road doesn't have enough predicted states! Please check motion_computation is correctly applying predicted states");
657 const double predict_step_duration = (rclcpp::Time(trajectory2.at(1).header.stamp) - rclcpp::Time(trajectory2.front().header.stamp)).seconds();
660 if (predict_step_duration < 0.0)
662 throw std::invalid_argument(
"Predicted states of the object is malformed. Detected trajectory going backwards in time!");
672 int iteration_stride = std::max(1,
static_cast<int>(iteration_stride_max_time_s / predict_step_duration));
674 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"Determined iteration_stride: " << iteration_stride
675 <<
", with traj2_speed: " << traj2_speed
676 <<
", with trajectory1_max_speed: " << trajectory1_max_speed
677 <<
", with predict_step_duration: " << predict_step_duration
678 <<
", iteration_stride_max_time_s: " << iteration_stride_max_time_s);
680 for (
size_t j = 0; j < trajectory2.size(); j += iteration_stride)
682 lanelet::BasicPoint2d curr_point;
683 curr_point.x() = trajectory2.at(j).predicted_position.position.x;
684 curr_point.y() = trajectory2.at(j).predicted_position.position.y;
686 auto corresponding_lanelets =
wm_->getLaneletsFromPoint(curr_point, 8);
688 for (
const auto& llt: corresponding_lanelets)
690 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"Checking llt: " << llt.id());
699 if (on_route || traj2_has_zero_speed)
705 RCLCPP_DEBUG(
nh_->get_logger(),
"Predicted states are not on the route! ignoring");
709 double smallest_dist = std::numeric_limits<double>::infinity();
710 for (
size_t i = 0;
i < trajectory1.trajectory_points.size() - 1; ++
i)
712 auto p1a = trajectory1.trajectory_points.at(
i);
713 auto p1b = trajectory1.trajectory_points.at(
i + 1);
714 double previous_distance_between_predictions = std::numeric_limits<double>::infinity();
715 for (
size_t j = on_route_idx; j < trajectory2.size() - 1; j += iteration_stride)
717 auto p2a = trajectory2.at(j);
718 auto p2b = trajectory2.at(j + 1);
719 double p1a_t = rclcpp::Time(p1a.target_time).seconds();
720 double p1b_t = rclcpp::Time(p1b.target_time).seconds();
721 double p2a_t = rclcpp::Time(p2a.header.stamp).seconds();
722 double p2b_t = rclcpp::Time(p2b.header.stamp).seconds();
726 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"p1a.x: " << p1a.x <<
", p1a.y: " << p1a.y);
727 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"p1b.x: " << p1b.x <<
", p1b.y: " << p1b.y);
729 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"p2a.x: " << p2a.predicted_position.position.x <<
", p2a.y: " << p2a.predicted_position.position.y);
730 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"p2b.x: " << p2b.predicted_position.position.x <<
", p2b.y: " << p2b.predicted_position.position.y);
733 double dt = (p2a_t - p1a_t) / (p1b_t - p1a_t);
739 double x1 = p1a.x + dt * (p1b.x - p1a.x);
740 double y1 = p1a.y + dt * (p1b.y - p1a.y);
741 double x2 = p2a.predicted_position.position.x;
742 double y2 = p2a.predicted_position.position.y;
745 const auto distance{std::hypot(x1 - x2, y1 - y2)};
747 smallest_dist = std::min(distance, smallest_dist);
753 if (previous_distance_between_predictions < distance)
755 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"Stopping search here because the distance between predictions started to increase");
758 previous_distance_between_predictions = distance;
762 RCLCPP_DEBUG(
nh_->get_logger(),
"Too far away" );
766 if (distance > collision_radius)
773 collision_result.
point1 = lanelet::BasicPoint2d(x1,y1);
774 collision_result.
point2 = lanelet::BasicPoint2d(x2,y2);
776 return collision_result;
780 rclcpp::get_logger(
"yield_plugin"),
781 "Was not able to find collision: smallest_dist: " << smallest_dist);
796 RCLCPP_INFO_STREAM(
nh_->get_logger(),
"Detected an object nearby might be behind the vehicle at timestamp: " <<
std::to_string(collision_time.seconds()) <<
815 const carma_perception_msgs::msg::ExternalObject& curr_obstacle,
double original_tp_max_speed)
819 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"Object's back time: " <<
std::to_string(rclcpp::Time(curr_obstacle.predictions.back().header.stamp).seconds())
823 if (rclcpp::Time(curr_obstacle.predictions.back().header.stamp).seconds() <= plan_start_time)
828 std::vector<carma_perception_msgs::msg::PredictedState> new_list;
829 carma_perception_msgs::msg::PredictedState curr_state;
831 curr_state.header.stamp = curr_obstacle.header.stamp;
832 curr_state.predicted_position.position.x = curr_obstacle.pose.pose.position.x;
833 curr_state.predicted_position.position.y = curr_obstacle.pose.pose.position.y;
835 curr_state.predicted_velocity.linear.x = curr_obstacle.velocity.twist.linear.x;
836 curr_state.predicted_velocity.linear.y = curr_obstacle.velocity.twist.linear.y;
837 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"Object: " << curr_obstacle.id <<
", type: " <<
static_cast<int>(curr_obstacle.object_type)
838 <<
", speed_x: " << curr_obstacle.velocity.twist.linear.x <<
", speed_y: " << curr_obstacle.velocity.twist.linear.y);
839 new_list.push_back(curr_state);
840 new_list.insert(new_list.end(), curr_obstacle.predictions.cbegin(), curr_obstacle.predictions.cend());
844 if (!collision_result)
852 const double vehicle_downtrack =
wm_->routeTrackPos(collision_result.value().point1).downtrack;
853 const double object_downtrack =
wm_->routeTrackPos(collision_result.value().point2).downtrack;
855 if (
is_object_behind_vehicle(curr_obstacle.id, collision_result.value().collision_time, vehicle_downtrack, object_downtrack))
857 RCLCPP_INFO_STREAM(
nh_->get_logger(),
"Confirmed that the object: " << curr_obstacle.id <<
" is behind the vehicle at timestamp " <<
std::to_string(collision_result.value().collision_time.seconds()));
861 const auto distance{std::hypot(
862 collision_result.value().point1.x() - collision_result.value().point2.x(),
863 collision_result.value().point1.y() - collision_result.value().point2.y()
866 RCLCPP_WARN_STREAM(
nh_->get_logger(),
"Collision detected for object: " << curr_obstacle.id <<
", at timestamp " <<
std::to_string(collision_result.value().collision_time.seconds()) <<
867 ", x: " << collision_result.value().point1.x() <<
", y: " << collision_result.value().point1.y() <<
868 ", within actual downtrack distance: " << object_downtrack - vehicle_downtrack <<
869 ", and collision distance: " << distance);
871 return collision_result.value().collision_time;
875 const std::vector<carma_perception_msgs::msg::ExternalObject>& external_objects,
double original_tp_max_speed)
878 std::unordered_map<uint32_t, std::future<std::optional<rclcpp::Time>>> futures;
879 std::unordered_map<uint32_t, rclcpp::Time> collision_times;
882 for (
const auto&
object : external_objects) {
883 futures[
object.id] = std::async(std::launch::async,[
this, &original_tp, &
object, &original_tp_max_speed]{
889 for (
const auto&
object : external_objects) {
890 if (
const auto collision_time{futures.at(
object.
id).get()}) {
891 collision_times[
object.id] = collision_time.value();
895 return collision_times;
899 const std::vector<carma_perception_msgs::msg::ExternalObject>& external_objects)
901 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"ExternalObjects size: " << external_objects.size());
903 if (!
wm_->getRoute())
905 RCLCPP_WARN(
nh_->get_logger(),
"Yield plugin was not able to analyze collision since route is not available! Please check if route is set");
910 for (
const auto& llt:
wm_->getRoute()->shortestPath())
916 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"External Object List (external_objects) size: " << external_objects.size());
920 if (collision_times.empty()) {
return std::nullopt; }
922 const auto earliest_colliding_object_id{std::min_element(
923 std::cbegin(collision_times), std::cend(collision_times),
924 [](
const auto & a,
const auto & b){
return a.second < b.second; })->first};
926 const auto earliest_colliding_object{std::find_if(
927 std::cbegin(external_objects), std::cend(external_objects),
928 [&earliest_colliding_object_id](
const auto &
object) {
return object.id == earliest_colliding_object_id; })};
930 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"earliest object x: " << earliest_colliding_object->velocity.twist.linear.x
931 <<
", y: " << earliest_colliding_object->velocity.twist.linear.y);
932 return std::make_pair(*earliest_colliding_object, collision_times.at(earliest_colliding_object_id).seconds());
937 const carma_planning_msgs::msg::TrajectoryPlan& original_tp,
double timestamp_in_sec_to_predict)
939 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"timestamp_in_sec_to_predict: " <<
std::to_string(timestamp_in_sec_to_predict) <<
942 double point_b_time = 0.0;
943 carma_planning_msgs::msg::TrajectoryPlanPoint point_a;
944 carma_planning_msgs::msg::TrajectoryPlanPoint point_b;
948 for (
size_t i = 0;
i < original_tp.trajectory_points.size() - 1; ++
i)
950 point_a = original_tp.trajectory_points.at(
i);
951 point_b = original_tp.trajectory_points.at(
i + 1);
952 point_b_time = rclcpp::Time(point_b.target_time).seconds();
953 if (point_b_time >= timestamp_in_sec_to_predict)
959 auto dx = point_b.x - point_a.x;
960 auto dy = point_b.y - point_a.y;
961 const tf2::Vector3 trajectory_direction(
dx,
dy, 0);
963 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"timestamp_in_sec_to_predict: " <<
std::to_string(timestamp_in_sec_to_predict)
965 <<
", dx: " <<
dx <<
", dy: " <<
dy <<
", "
966 <<
", object_velocity_in_map_frame.x: " << object_velocity_in_map_frame.linear.x
967 <<
", object_velocity_in_map_frame.y: " << object_velocity_in_map_frame.linear.y);
969 if (trajectory_direction.length() < 0.001)
974 const tf2::Vector3 object_direction(object_velocity_in_map_frame.linear.x, object_velocity_in_map_frame.linear.y, 0);
976 return tf2::tf2Dot(object_direction, trajectory_direction) / trajectory_direction.length();
980 const std::vector<carma_perception_msgs::msg::ExternalObject>& external_objects,
double initial_velocity)
982 if (original_tp.trajectory_points.size() < 2)
984 RCLCPP_WARN(
nh_->get_logger(),
"Yield plugin received less than 2 points in update_traj_for_object, returning unchanged...");
991 if (!earliest_collision_obj_pair)
993 RCLCPP_DEBUG(
nh_->get_logger(),
"No collision detected, so trajectory not modified.");
997 carma_perception_msgs::msg::ExternalObject earliest_collision_obj = earliest_collision_obj_pair.value().first;
998 double earliest_collision_time_in_seconds = earliest_collision_obj_pair.value().second;
1003 const lanelet::BasicPoint2d vehicle_point(original_tp.trajectory_points[0].x,original_tp.trajectory_points[0].y);
1004 const double vehicle_downtrack =
wm_->routeTrackPos(vehicle_point).downtrack;
1006 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"vehicle_downtrack: " << vehicle_downtrack);
1008 RCLCPP_WARN_STREAM(
nh_->get_logger(),
"Collision Detected!");
1010 const lanelet::BasicPoint2d object_point(earliest_collision_obj.pose.pose.position.x, earliest_collision_obj.pose.pose.position.y);
1011 const double object_downtrack =
wm_->routeTrackPos(object_point).downtrack;
1013 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"object_downtrack: " << object_downtrack);
1015 const double object_downtrack_lead = std::max(0.0, object_downtrack - vehicle_downtrack);
1016 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"object_downtrack_lead: " << object_downtrack_lead);
1020 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"object's speed along trajectory at collision: " << goal_velocity);
1026 RCLCPP_WARN_STREAM(
nh_->get_logger(),
"The obstacle is not moving, goal velocity is set to 0 from: " << goal_velocity);
1027 goal_velocity = 0.0;
1032 if (!std::isnormal(safety_gap))
1034 RCLCPP_WARN_STREAM(rclcpp::get_logger(
"yield_plugin"),
"Detected non-normal (nan, inf, etc.) safety_gap."
1043 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"externally_commanded_safety_gap: " << externally_commanded_safety_gap);
1045 safety_gap = std::max(safety_gap, externally_commanded_safety_gap);
1049 const double initial_pos = 0.0;
1050 const double original_max_speed =
max_trajectory_speed(original_tp.trajectory_points, earliest_collision_time_in_seconds);
1051 const double delta_v_max = fabs(goal_velocity - original_max_speed);
1052 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"delta_v_max: " << delta_v_max <<
", safety_gap: " << safety_gap);
1059 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"time_required_for_comfortable_decel_in_s: " << time_required_for_comfortable_decel_in_s <<
", min_time_required_for_comfortable_decel_in_s: " << min_time_required_for_comfortable_decel_in_s);
1061 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"Object avoidance planning time: " << planning_time_in_s);
1064 initial_pos, goal_pos, initial_velocity, goal_velocity,
1065 planning_time_in_s, original_max_speed);
1070 earliest_collision_time_in_seconds -
nh_->now().seconds()
1075 return jmt_trajectory;
1081 std::vector<double> downtracks;
1082 downtracks.reserve(trajectory_plan.trajectory_points.size());
1084 downtracks.push_back(0.0);
1085 for (
size_t i=1;
i < trajectory_plan.trajectory_points.size();
i++){
1087 double dx = trajectory_plan.trajectory_points.at(
i).x - trajectory_plan.trajectory_points.at(
i-1).x;
1088 double dy = trajectory_plan.trajectory_points.at(
i).y - trajectory_plan.trajectory_points.at(
i-1).y;
1089 downtracks.push_back(sqrt(
dx*
dx +
dy*
dy));
1097 for (
size_t i = 0;
i < coeff.size();
i++)
1099 double value = coeff.at(
i) * pow(
x,
static_cast<int>(coeff.size() - 1 -
i));
1100 result = result + value;
1108 for (
size_t i = 0;
i < coeff.size()-1;
i++)
1110 double value =
static_cast<int>(coeff.size() - 1 -
i) * coeff.at(
i) * pow(
x,
static_cast<int>(coeff.size() - 2 -
i));
1111 result = result + value;
1118 double max_speed = 0;
1119 for(
size_t i = 0;
i < trajectory_points.size() - 2;
i++ )
1121 double dx = trajectory_points.at(
i + 1).x - trajectory_points.at(
i).x;
1122 double dy = trajectory_points.at(
i + 1).y - trajectory_points.at(
i).y;
1124 double t = (rclcpp::Time(trajectory_points.at(
i + 1).target_time).seconds() - rclcpp::Time(trajectory_points.at(
i).target_time).seconds());
1130 if (rclcpp::Time(trajectory_points.at(
i + 1).target_time).seconds() >= timestamp_in_sec_to_search_until)
1141 double desired_gap = 0;
1143 for (
size_t i = 0;
i < original_tp.trajectory_points.size();
i++)
1145 lanelet::BasicPoint2d veh_pos(original_tp.trajectory_points.at(
i).x, original_tp.trajectory_points.at(
i).y);
1146 auto llts =
wm_->getLaneletsFromPoint(veh_pos, 1);
1151 RCLCPP_WARN_STREAM(
nh_->get_logger(),
"Trajectory point: x= " << original_tp.trajectory_points.at(
i).x <<
"y="<< original_tp.trajectory_points.at(
i).y);
1152 RCLCPP_WARN_STREAM(
nh_->get_logger(),
"Trajectory is not on the road, so was unable to get the digital minimum gap. Returning default minimum_safety_gap_in_meters: " <<
config_.
minimum_safety_gap_in_meters);
1155 auto digital_min_gap = llts[0].regulatoryElementsAs<lanelet::DigitalMinimumGap>();
1156 if (!digital_min_gap.empty())
1158 double digital_gap = digital_min_gap[0]->getMinimumGap();
1159 RCLCPP_DEBUG_STREAM(
nh_->get_logger(),
"Digital Gap found with value: " << digital_gap);
1160 desired_gap = std::max(desired_gap, digital_gap);
1171 map_projector_ = std::make_shared<lanelet::projection::LocalFrameProjector>(georeference.c_str());
std::string georeference_
std::set< lanelet::Id > route_llt_ids_
void plan_trajectory_callback(carma_planning_msgs::srv::PlanTrajectory::Request::SharedPtr req, carma_planning_msgs::srv::PlanTrajectory::Response::SharedPtr resp)
Service callback for trajectory planning.
std::vector< double > get_relative_downtracks(const carma_planning_msgs::msg::TrajectoryPlan &trajectory_plan) const
calculates distance between trajectory points in a plan
LaneChangeStatusCB lc_status_publisher_
void set_incoming_request_info(std::vector< lanelet::BasicPoint2d > req_trajectory, double req_speed, double req_planning_time, double req_timestamp)
set values for member variables related to cooperative behavior
std::vector< lanelet::BasicPoint2d > req_trajectory_points_
void mobilityrequest_cb(const carma_v2x_msgs::msg::MobilityRequest::UniquePtr msg)
callback for mobility request
double check_traj_for_digital_min_gap(const carma_planning_msgs::msg::TrajectoryPlan &original_tp) const
checks trajectory for minimum gap associated with it from the road
double get_predicted_velocity_at_time(const geometry_msgs::msg::Twist &object_velocity_in_map_frame, const carma_planning_msgs::msg::TrajectoryPlan &original_tp, double timestamp_in_sec_to_predict)
Given the object velocity in map frame with x,y components, this function returns the projected veloc...
std::shared_ptr< lanelet::projection::LocalFrameProjector > map_projector_
std::vector< lanelet::BasicPoint2d > convert_eceftrajectory_to_mappoints(const carma_v2x_msgs::msg::Trajectory &ecef_trajectory) const
convert a carma trajectory from ecef frame to map frame ecef trajectory consists of the point and a s...
std::optional< std::pair< carma_perception_msgs::msg::ExternalObject, double > > get_earliest_collision_object_and_time(const carma_planning_msgs::msg::TrajectoryPlan &original_tp, const std::vector< carma_perception_msgs::msg::ExternalObject > &external_objects)
Return the earliest collision object and time of collision pair from the given trajectory and list of...
std::optional< rclcpp::Time > get_collision_time(const carma_planning_msgs::msg::TrajectoryPlan &original_tp, const carma_perception_msgs::msg::ExternalObject &curr_obstacle, double original_tp_max_speed)
Return collision time given two trajectories with one being external object with predicted steps.
std::optional< rclcpp::Time > last_speed_time_
void bsm_cb(const carma_v2x_msgs::msg::BSM::UniquePtr msg)
callback for bsm message
double ecef_traj_timestep_
MobilityResponseCB mobility_response_publisher_
std::string bsmIDtoString(carma_v2x_msgs::msg::BSMCoreData bsm_core)
void set_external_objects(const std::vector< carma_perception_msgs::msg::ExternalObject > &object_list)
Setter for external objects with predictions in the environment.
YieldPlugin(std::shared_ptr< carma_ros2_utils::CarmaLifecycleNode > nh, carma_wm::WorldModelConstPtr wm, YieldPluginConfig config, MobilityResponseCB mobility_response_publisher, LaneChangeStatusCB lc_status_publisher)
Constructor.
std::shared_ptr< carma_ros2_utils::CarmaLifecycleNode > nh_
void set_georeference_string(const std::string &georeference)
Setter for map projection string to define lat/lon -> map conversion.
std::unordered_map< uint32_t, rclcpp::Time > get_collision_times_concurrently(const carma_planning_msgs::msg::TrajectoryPlan &original_tp, const std::vector< carma_perception_msgs::msg::ExternalObject > &external_objects, double original_tp_max_speed)
Given the list of objects with predicted states, get all collision times concurrently using multi-thr...
YieldPluginConfig config_
std::optional< rclcpp::Time > first_time_stopped_to_prevent_collision_
std::optional< GetCollisionResult > get_collision(const carma_planning_msgs::msg::TrajectoryPlan &trajectory1, const std::vector< carma_perception_msgs::msg::PredictedState > &trajectory2, double collision_radius, double trajectory1_max_speed)
Return naive collision time and locations based on collision radius given two trajectories with one b...
std::vector< std::pair< int, lanelet::BasicPoint2d > > detect_trajectories_intersection(std::vector< lanelet::BasicPoint2d > self_trajectory, std::vector< lanelet::BasicPoint2d > incoming_trajectory) const
detect intersection point(s) of two trajectories
std::optional< carma_planning_msgs::msg::TrajectoryPlan > last_traj_plan_committed_to_stopping_
lanelet::BasicPoint2d ecef_to_map_point(const carma_v2x_msgs::msg::LocationECEF &ecef_point) const
convert a point in ecef frame (in cm) into map frame (in meters)
carma_planning_msgs::msg::TrajectoryPlan update_traj_for_object(const carma_planning_msgs::msg::TrajectoryPlan &original_tp, const std::vector< carma_perception_msgs::msg::ExternalObject > &external_objects, double initial_velocity)
trajectory is modified to safely avoid obstacles on the road
carma_planning_msgs::msg::TrajectoryPlan generate_JMT_trajectory(const carma_planning_msgs::msg::TrajectoryPlan &original_tp, double initial_pos, double goal_pos, double initial_velocity, double goal_velocity, double planning_time, double original_max_speed)
generate a Jerk Minimizing Trajectory(JMT) with the provided start and end conditions
carma_v2x_msgs::msg::MobilityResponse compose_mobility_response(const std::string &resp_recipient_id, const std::string &req_plan_id, bool response) const
compose a mobility response message
double req_target_plan_time_
bool cooperative_request_acceptable_
carma_planning_msgs::msg::TrajectoryPlan update_traj_for_cooperative_behavior(const carma_planning_msgs::msg::TrajectoryPlan &original_tp, double current_speed)
update trajectory for yielding to an incoming cooperative behavior
std::unordered_map< uint32_t, int > consecutive_clearance_count_for_obstacles_
std::vector< carma_perception_msgs::msg::ExternalObject > external_objects_
bool is_object_behind_vehicle(uint32_t object_id, const rclcpp::Time &collision_time, double vehicle_point, double object_downtrack)
Check if object location is behind the vehicle using estimates of the vehicle's length and route down...
int timesteps_since_last_req_
double max_trajectory_speed(const std::vector< carma_planning_msgs::msg::TrajectoryPlanPoint > &trajectory_points, double timestamp_in_sec_to_search_until) const
calculates the maximum speed in a set of tajectory points
double polynomial_calc(std::vector< double > coeff, double x) const
calculate quintic polynomial equation for a given x
std::optional< double > last_speed_
double polynomial_calc_d(std::vector< double > coeff, double x) const
calculate derivative of quintic polynomial equation for a given x
carma_wm::WorldModelConstPtr wm_
std::vector< double > moving_average_filter(const std::vector< double > input, int window_size, bool ignore_first_point=true)
Extremely simplie moving average filter.
int get_nearest_point_index(const std::vector< lanelet::BasicPoint2d > &points, const carma_planning_msgs::msg::VehicleState &state)
Returns the nearest point (in terms of cartesian 2d distance) to the provided vehicle pose in the pro...
auto to_string(const UtmZone &zone) -> std::string
std::shared_ptr< const WorldModel > WorldModelConstPtr
std::function< void(const carma_planning_msgs::msg::LaneChangeStatus &)> LaneChangeStatusCB
double get_smallest_time_step_of_traj(const carma_planning_msgs::msg::TrajectoryPlan &original_tp)
double get_trajectory_start_time(const carma_planning_msgs::msg::TrajectoryPlan &trajectory)
double get_trajectory_end_time(const carma_planning_msgs::msg::TrajectoryPlan &trajectory)
std::function< void(const carma_v2x_msgs::msg::MobilityResponse &)> MobilityResponseCB
double get_trajectory_duration(const carma_planning_msgs::msg::TrajectoryPlan &trajectory)
Stuct containing the algorithm configuration values for the YieldPluginConfig.
bool always_accept_mobility_request
double max_stop_speed_in_ms
double collision_check_radius_in_m
bool enable_adjustable_gap
double speed_moving_average_window_size
int consecutive_clearance_count_for_passed_obstacles_threshold
double acceleration_adjustment_factor
double yield_max_deceleration_in_ms2
double time_horizon_until_collision_to_commit_to_stop_in_s
double intervehicle_collision_distance_in_m
double obstacle_zero_speed_threshold_in_ms
bool enable_cooperative_behavior
double min_obj_avoidance_plan_time_in_s
double minimum_safety_gap_in_meters
int acceptable_passed_timesteps
double safety_collision_time_gap_in_s
Convenience class for saving collision results.
lanelet::BasicPoint2d point1
rclcpp::Time collision_time
lanelet::BasicPoint2d point2