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

feat(external_cmd_converter): replace polling takeData function with the callback function #7263

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 @@ -15,6 +15,8 @@
#ifndef EXTERNAL_CMD_CONVERTER__NODE_HPP_
#define EXTERNAL_CMD_CONVERTER__NODE_HPP_

#include "tier4_autoware_utils/ros/polling_subscriber.hpp"

#include <diagnostic_updater/diagnostic_updater.hpp>
#include <raw_vehicle_cmd_converter/accel_map.hpp>
#include <raw_vehicle_cmd_converter/brake_map.hpp>
Expand All @@ -33,12 +35,12 @@
namespace external_cmd_converter
{
using GearCommand = autoware_vehicle_msgs::msg::GearCommand;
using Control = autoware_control_msgs::msg::Control;
using autoware_control_msgs::msg::Control;
using ExternalControlCommand = tier4_external_api_msgs::msg::ControlCommandStamped;
using Odometry = nav_msgs::msg::Odometry;
using nav_msgs::msg::Odometry;
using raw_vehicle_cmd_converter::AccelMap;
using raw_vehicle_cmd_converter::BrakeMap;
using Odometry = nav_msgs::msg::Odometry;
using tier4_control_msgs::msg::GateMode;

class ExternalCmdConverterNode : public rclcpp::Node
{
Expand All @@ -47,33 +49,36 @@ class ExternalCmdConverterNode : public rclcpp::Node

private:
// Publisher
rclcpp::Publisher<Control>::SharedPtr pub_cmd_;
rclcpp::Publisher<Control>::SharedPtr cmd_pub_;
rclcpp::Publisher<tier4_external_api_msgs::msg::ControlCommandStamped>::SharedPtr
pub_current_cmd_;
current_cmd_pub_;

// Subscriber
rclcpp::Subscription<Odometry>::SharedPtr sub_velocity_;
rclcpp::Subscription<tier4_external_api_msgs::msg::ControlCommandStamped>::SharedPtr
sub_control_cmd_;
rclcpp::Subscription<GearCommand>::SharedPtr sub_shift_cmd_;
rclcpp::Subscription<tier4_control_msgs::msg::GateMode>::SharedPtr sub_gate_mode_;
control_cmd_sub_;
rclcpp::Subscription<tier4_external_api_msgs::msg::Heartbeat>::SharedPtr
sub_emergency_stop_heartbeat_;
emergency_stop_heartbeat_sub_;

// Polling Subscriber
tier4_autoware_utils::InterProcessPollingSubscriber<Odometry> velocity_sub_{this, "in/odometry"};
tier4_autoware_utils::InterProcessPollingSubscriber<GearCommand> shift_cmd_sub_{
this, "in/shift_cmd"};
tier4_autoware_utils::InterProcessPollingSubscriber<GateMode> gate_mode_sub_{
this, "in/current_gate_mode"};

shmpwk marked this conversation as resolved.
Show resolved Hide resolved
void on_external_cmd(const ExternalControlCommand::ConstSharedPtr cmd_ptr);
void on_emergency_stop_heartbeat(
const tier4_external_api_msgs::msg::Heartbeat::ConstSharedPtr msg);

void onVelocity(const Odometry::ConstSharedPtr msg);
void onExternalCmd(const ExternalControlCommand::ConstSharedPtr cmd_ptr);
void onGearCommand(const GearCommand::ConstSharedPtr msg);
void onGateMode(const tier4_control_msgs::msg::GateMode::ConstSharedPtr msg);
void onEmergencyStopHeartbeat(const tier4_external_api_msgs::msg::Heartbeat::ConstSharedPtr msg);
Odometry::ConstSharedPtr current_velocity_ptr_{nullptr}; // [m/s]
GearCommand::ConstSharedPtr current_shift_cmd_{nullptr};
GateMode::ConstSharedPtr current_gate_mode_{nullptr};

std::shared_ptr<double> current_velocity_ptr_; // [m/s]
std::shared_ptr<rclcpp::Time> latest_emergency_stop_heartbeat_received_time_;
std::shared_ptr<rclcpp::Time> latest_cmd_received_time_;
GearCommand::ConstSharedPtr current_shift_cmd_;
tier4_control_msgs::msg::GateMode::ConstSharedPtr current_gate_mode_;

// Timer
void onTimer();
void on_timer();
rclcpp::TimerBase::SharedPtr rate_check_timer_;

// Parameter
Expand All @@ -85,18 +90,18 @@ class ExternalCmdConverterNode : public rclcpp::Node
// Diagnostics
diagnostic_updater::Updater updater_{this};

void checkTopicStatus(diagnostic_updater::DiagnosticStatusWrapper & stat);
void checkEmergencyStop(diagnostic_updater::DiagnosticStatusWrapper & stat);
bool checkEmergencyStopTopicTimeout();
bool checkRemoteTopicRate();
void check_topic_status(diagnostic_updater::DiagnosticStatusWrapper & stat);
void check_emergency_stop(diagnostic_updater::DiagnosticStatusWrapper & stat);
bool check_emergency_stop_topic_timeout();
bool check_remote_topic_rate();

// Algorithm
AccelMap accel_map_;
BrakeMap brake_map_;
bool acc_map_initialized_;

double calculateAcc(const ExternalControlCommand & cmd, const double vel);
double getShiftVelocitySign(const GearCommand & cmd);
double calculate_acc(const ExternalControlCommand & cmd, const double vel);
double get_shift_velocity_sign(const GearCommand & cmd);
};

} // namespace external_cmd_converter
Expand Down
1 change: 1 addition & 0 deletions vehicle/external_cmd_converter/package.xml
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
<version>0.1.0</version>
<description>The external_cmd_converter package</description>
<maintainer email="[email protected]">Takamasa Horibe</maintainer>
<maintainer email="[email protected]">Eiki Nagata</maintainer>
<license>Apache License 2.0</license>

<author email="[email protected]">Takamasa Horibe</author>
Expand Down
107 changes: 41 additions & 66 deletions vehicle/external_cmd_converter/src/node.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -26,20 +26,14 @@
{
using std::placeholders::_1;

pub_cmd_ = create_publisher<Control>("out/control_cmd", rclcpp::QoS{1});
pub_current_cmd_ =
cmd_pub_ = create_publisher<Control>("out/control_cmd", rclcpp::QoS{1});
current_cmd_pub_ =
create_publisher<ExternalControlCommand>("out/latest_external_control_cmd", rclcpp::QoS{1});
sub_velocity_ = create_subscription<Odometry>(
"in/odometry", 1, std::bind(&ExternalCmdConverterNode::onVelocity, this, _1));
sub_control_cmd_ = create_subscription<ExternalControlCommand>(
"in/external_control_cmd", 1, std::bind(&ExternalCmdConverterNode::onExternalCmd, this, _1));
sub_shift_cmd_ = create_subscription<GearCommand>(
"in/shift_cmd", 1, std::bind(&ExternalCmdConverterNode::onGearCommand, this, _1));
sub_gate_mode_ = create_subscription<tier4_control_msgs::msg::GateMode>(
"in/current_gate_mode", 1, std::bind(&ExternalCmdConverterNode::onGateMode, this, _1));
sub_emergency_stop_heartbeat_ = create_subscription<tier4_external_api_msgs::msg::Heartbeat>(
control_cmd_sub_ = create_subscription<ExternalControlCommand>(
"in/external_control_cmd", 1, std::bind(&ExternalCmdConverterNode::on_external_cmd, this, _1));
emergency_stop_heartbeat_sub_ = create_subscription<tier4_external_api_msgs::msg::Heartbeat>(
"in/emergency_stop", 1,
std::bind(&ExternalCmdConverterNode::onEmergencyStopHeartbeat, this, _1));
std::bind(&ExternalCmdConverterNode::on_emergency_stop_heartbeat, this, _1));

// Parameter
ref_vel_gain_ = declare_parameter<double>("ref_vel_gain");
Expand All @@ -52,7 +46,7 @@

const auto period_ns = rclcpp::Rate(timer_rate).period();
rate_check_timer_ = rclcpp::create_timer(
this, get_clock(), period_ns, std::bind(&ExternalCmdConverterNode::onTimer, this));
this, get_clock(), period_ns, std::bind(&ExternalCmdConverterNode::on_timer, this));

// Parameter for accel/brake map
const std::string csv_path_accel_map = declare_parameter<std::string>("csv_path_accel_map");
Expand All @@ -73,54 +67,49 @@

// Diagnostics
updater_.setHardwareID("external_cmd_converter");
updater_.add("remote_control_topic_status", this, &ExternalCmdConverterNode::checkTopicStatus);
updater_.add("remote_control_topic_status", this, &ExternalCmdConverterNode::check_topic_status);

// Set default values
current_shift_cmd_ = std::make_shared<GearCommand>();
}

void ExternalCmdConverterNode::onTimer()
void ExternalCmdConverterNode::on_timer()
{
updater_.force_update();
}

void ExternalCmdConverterNode::onVelocity(const Odometry::ConstSharedPtr msg)
{
current_velocity_ptr_ = std::make_shared<double>(msg->twist.twist.linear.x);
}

void ExternalCmdConverterNode::onGearCommand(const GearCommand::ConstSharedPtr msg)
{
current_shift_cmd_ = msg;
}

void ExternalCmdConverterNode::onEmergencyStopHeartbeat(
void ExternalCmdConverterNode::on_emergency_stop_heartbeat(
[[maybe_unused]] const tier4_external_api_msgs::msg::Heartbeat::ConstSharedPtr msg)
{
latest_emergency_stop_heartbeat_received_time_ = std::make_shared<rclcpp::Time>(this->now());
updater_.force_update();
}

void ExternalCmdConverterNode::onExternalCmd(const ExternalControlCommand::ConstSharedPtr cmd_ptr)
void ExternalCmdConverterNode::on_external_cmd(const ExternalControlCommand::ConstSharedPtr cmd_ptr)
{
// Echo back received command
{
auto current_cmd = *cmd_ptr;
current_cmd.stamp = this->now();
pub_current_cmd_->publish(current_cmd);
current_cmd_pub_->publish(current_cmd);
}

// Save received time for rate check
latest_cmd_received_time_ = std::make_shared<rclcpp::Time>(this->now());

// take data from subscribers
current_velocity_ptr_ = velocity_sub_.takeData();
current_shift_cmd_ = shift_cmd_sub_.takeData();

// Wait for input data
if (!current_velocity_ptr_ || !acc_map_initialized_) {
if (!current_velocity_ptr_ || !acc_map_initialized_ || !current_shift_cmd_) {

Check warning on line 105 in vehicle/external_cmd_converter/src/node.cpp

View check run for this annotation

CodeScene Delta Analysis / CodeScene Cloud Delta Analysis (main)

❌ New issue: Complex Conditional

ExternalCmdConverterNode::on_external_cmd has 1 complex conditionals with 2 branches, threshold = 2. A complex conditional is an expression inside a branch (e.g. if, for, while) which consists of multiple, logical operators such as AND/OR. The more logical operators in an expression, the more severe the code smell.
return;
}

// Calculate reference velocity and acceleration
const double sign = getShiftVelocitySign(*current_shift_cmd_);
const double ref_acceleration = calculateAcc(*cmd_ptr, std::fabs(*current_velocity_ptr_));
const double sign = get_shift_velocity_sign(*current_shift_cmd_);
const double ref_acceleration =
calculate_acc(*cmd_ptr, std::fabs(current_velocity_ptr_->twist.twist.linear.x));

if (ref_acceleration > 0.0 && sign == 0.0) {
RCLCPP_WARN_THROTTLE(
Expand All @@ -129,7 +118,8 @@
ref_acceleration, current_shift_cmd_->command);
}

double ref_velocity = *current_velocity_ptr_ + ref_acceleration * ref_vel_gain_ * sign;
double ref_velocity =
current_velocity_ptr_->twist.twist.linear.x + ref_acceleration * ref_vel_gain_ * sign;
if (current_shift_cmd_->command == GearCommand::REVERSE) {
ref_velocity = std::min(0.0, ref_velocity);
} else if (
Expand All @@ -144,15 +134,16 @@
// Publish ControlCommand
autoware_control_msgs::msg::Control output;
output.stamp = cmd_ptr->stamp;
output.lateral.steering_tire_angle = cmd_ptr->control.steering_angle;
output.lateral.steering_tire_rotation_rate = cmd_ptr->control.steering_angle_velocity;
output.longitudinal.velocity = ref_velocity;
output.longitudinal.acceleration = ref_acceleration;
output.lateral.steering_tire_angle = static_cast<float>(cmd_ptr->control.steering_angle);
output.lateral.steering_tire_rotation_rate =
static_cast<float>(cmd_ptr->control.steering_angle_velocity);
output.longitudinal.velocity = static_cast<float>(ref_velocity);
output.longitudinal.acceleration = static_cast<float>(ref_acceleration);

pub_cmd_->publish(output);
cmd_pub_->publish(output);

Check warning on line 143 in vehicle/external_cmd_converter/src/node.cpp

View check run for this annotation

CodeScene Delta Analysis / CodeScene Cloud Delta Analysis (main)

❌ New issue: Complex Method

ExternalCmdConverterNode::on_external_cmd has a cyclomatic complexity of 9, threshold = 9. This function has many conditional statements (e.g. if, for, while), leading to lower code health. Avoid adding more conditionals and code to it without refactoring.
}

double ExternalCmdConverterNode::calculateAcc(const ExternalControlCommand & cmd, const double vel)
double ExternalCmdConverterNode::calculate_acc(const ExternalControlCommand & cmd, const double vel)
{
const double desired_throttle = cmd.control.throttle;
const double desired_brake = cmd.control.brake;
Expand All @@ -168,7 +159,7 @@
return ref_acceleration;
}

double ExternalCmdConverterNode::getShiftVelocitySign(const GearCommand & cmd)
double ExternalCmdConverterNode::get_shift_velocity_sign(const GearCommand & cmd)
{
if (cmd.command == GearCommand::DRIVE) {
return 1.0;
Expand All @@ -183,15 +174,15 @@
return 0.0;
}

void ExternalCmdConverterNode::checkTopicStatus(diagnostic_updater::DiagnosticStatusWrapper & stat)
void ExternalCmdConverterNode::check_topic_status(
diagnostic_updater::DiagnosticStatusWrapper & stat)
{
using diagnostic_msgs::msg::DiagnosticStatus;

DiagnosticStatus status;
if (!checkEmergencyStopTopicTimeout()) {
if (!check_emergency_stop_topic_timeout()) {
status.level = DiagnosticStatus::ERROR;
status.message = "emergency stop topic is timeout";
} else if (!checkRemoteTopicRate()) {
} else if (!check_remote_topic_rate()) {
status.level = DiagnosticStatus::ERROR;
status.message = "low topic rate for remote vehicle_cmd";
} else {
Expand All @@ -202,42 +193,26 @@
stat.summary(status.level, status.message);
}

void ExternalCmdConverterNode::onGateMode(
const tier4_control_msgs::msg::GateMode::ConstSharedPtr msg)
{
current_gate_mode_ = msg;
}

bool ExternalCmdConverterNode::checkEmergencyStopTopicTimeout()
bool ExternalCmdConverterNode::check_emergency_stop_topic_timeout()
{
if (!latest_emergency_stop_heartbeat_received_time_) {
if (wait_for_first_topic_) {
return true;
} else {
return false;
}
return wait_for_first_topic_;
}

const auto duration = (this->now() - *latest_emergency_stop_heartbeat_received_time_);
if (duration.seconds() > emergency_stop_timeout_) {
return false;
}

return true;
return duration.seconds() <= emergency_stop_timeout_;
}

bool ExternalCmdConverterNode::checkRemoteTopicRate()
bool ExternalCmdConverterNode::check_remote_topic_rate()
{
current_gate_mode_ = gate_mode_sub_.takeData();

if (!current_gate_mode_) {
return true;
}

if (!latest_cmd_received_time_) {
if (wait_for_first_topic_) {
return true;
} else {
return false;
}
return wait_for_first_topic_;

Check notice on line 215 in vehicle/external_cmd_converter/src/node.cpp

View check run for this annotation

CodeScene Delta Analysis / CodeScene Cloud Delta Analysis (main)

✅ No longer an issue: Bumpy Road Ahead

ExternalCmdConverterNode::checkRemoteTopicRate is no longer above the threshold for logical blocks with deeply nested code
shmpwk marked this conversation as resolved.
Show resolved Hide resolved
}

if (current_gate_mode_->data == tier4_control_msgs::msg::GateMode::EXTERNAL) {
Expand Down
Loading