Skip to content

Conversation

@wittenator
Copy link

@wittenator wittenator commented Mar 1, 2025

This PR adds the option to use steering angle and linear velocity for controllers that inherit from the steering library.

To send us a pull request, please:

  • Fork the repository.
  • Modify the source; please focus on the specific change you are contributing. If you also reformat all the code, it will be hard for us to focus on your change.
  • Ensure local tests pass. (colcon test and pre-commit run (requires you to install pre-commit by pip3 install pre-commit)
  • Commit to your fork using clear commit messages.
  • Send a pull request, answering any default questions in the pull request interface.
  • Pay attention to any automated CI failures reported in the pull request, and stay involved in the conversation.

@wittenator wittenator changed the title Add AckermannDriveStamped control to steering library WIP: Add AckermannDriveStamped control to steering library Mar 1, 2025
@wittenator
Copy link
Author

wittenator commented Mar 1, 2025

The tests on HEAD are currently broken, but I wanted to get this PR out in order to track progress.

Copy link
Contributor

@christophfroehlich christophfroehlich left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why WIP?
Please fix the failing jobs, start with the pre-commit and clang job.
The tests on the master branch are green.

@mergify
Copy link
Contributor

mergify bot commented May 24, 2025

This pull request is in conflict. Could you fix it @wittenator?

Copy link
Contributor

@christophfroehlich christophfroehlich left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I fixed the CMakeLists according to #1697. @wittenator do you have time to discuss the open conversations in this PR?

@wittenator
Copy link
Author

@Juliaj How relevant are additional test cases for the inheriting controllers? Since I only added passthrough for the class variable in the controllers, tests for each inherited controller would essentially only test the functionality of the steering controller that is already covered and add a huge amount of boilerplate for the other controllers.

Copy link
Contributor

@christophfroehlich christophfroehlich left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'll wait for fixing the open conversations before the final review

edit: and please take care of the pre-commit errors (have you installed it?)

@Juliaj
Copy link
Contributor

Juliaj commented Jun 12, 2025

@Juliaj How relevant are additional test cases for the inheriting controllers? Since I only added passthrough for the class variable in the controllers, tests for each inherited controller would essentially only test the functionality of the steering controller that is already covered and add a huge amount of boilerplate for the other controllers.

@wittenator, sorry for missing your comment. I noticed that you have updated the tests. Will take another look shortly.

----------------------------------

The controller uses velocity input, i.e., stamped `twist messages <twist_msg_>`_ where linear ``x`` and angular ``z`` components are used.
The controller uses velocity input, i.e., stamped `twist messages <twist_msg_>`_ where linear ``x`` and angular ``z`` components are used if ``twist_input == true``. If ``twist_input == false``, the controller uses `control_msgs/msg/SteeringControllerStatus <steering_controller_status_msg_>`_ where linear ``speed`` and angular ``steering_angle`` components are used.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could we add more context for the inputs, for example

The controller uses velocity input, i.e., stamped `twist messages <twist_msg_>`_ where:
- **Linear velocity (`linear x`)**: Represents the forward/backward motion of the robot along its x-axis (in meters per second, m/s).
- **Angular velocity (`angular z`)**: Represents the rotational motion of the robot around its z-axis (in radians per second, rad/s).

These components are used when ``twist_input == true``. 

If ``twist_input == false``, the controller uses `control_msgs/msg/SteeringControllerStatus <steering_controller_status_msg_>`_ where:
- **Speed (`speed`)**: Represents the linear speed of the robot (in meters per second, m/s).
- **Steering angle (`steering_angle`)**: Represents the angle of the steering joints (in radians, rad).

Values in other components are ignored.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sure, I can add this. One thing that comes up then is: What is a good way to describe the steering angle? The exact positions of the joints and wheels depends on both the steering angle as an abstract value and the exact kinematic setup of the robot. For an Ackermann steering geometry, the positions of the joints are different from each other and from the steering angle, so I had trouble finding a fitting description of it without mentioning something like a steering wheel or some common element that links the steering joints.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

one could define the steering angle with the ICR being perpendicular to the imaginary single steering wheel.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've added a small overview for both cases that describe the inputs and their physical equivalent in the docs.

@Juliaj
Copy link
Contributor

Juliaj commented Jun 13, 2025

@Juliaj How relevant are additional test cases for the inheriting controllers? Since I only added passthrough for the class variable in the controllers, tests for each inherited controller would essentially only test the functionality of the steering controller that is already covered and add a huge amount of boilerplate for the other controllers.

@wittenator, great work on the test coverage added in test_steering_controllers_library_steering_input.cpp!

I'd suggest adding one more test to cover the new use_twist_input parameter. This could be added to the existing on_configure test or in a dedicated test case to verify the parameter is properly handled during configuration.

Overall, these changes look solid!

@mergify
Copy link
Contributor

mergify bot commented Jul 24, 2025

This pull request is in conflict. Could you fix it @wittenator?

Copy link
Contributor

@christophfroehlich christophfroehlich left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I responded to your questions, could you please address them and fix the conflicts from the latest refactoring?

@christophfroehlich
Copy link
Contributor

I'd suggest adding one more test to cover the new use_twist_input parameter. This could be added to the existing on_configure test or in a dedicated test case to verify the parameter is properly handled during configuration.

Isn't this implicitly done by the new tests using the yaml parmeterfile?

@wittenator
Copy link
Author

I responded to your questions, could you please address them and fix the conflicts from the latest refactoring?

I'm on it and going to fix the conflicts over the coming days. Thanks for answering the remaining few questions!

Copy link
Contributor

@christophfroehlich christophfroehlich left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please also address the old open conversation about the argument and variable renaming.

right_traction_wheel_vel, left_traction_wheel_vel, steer_pos_);
const double angular_velocity =
convert_steering_angle_to_angular_velocity(linear_velocity, steer_pos_);
const double angular_velocity = std::tan(steer_pos_) * linear_velocity / wheel_base_;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

shouldnt it use convert_steering_angle_to_angular_velocity here?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, most definitely, I used it now instead.

msg.twist.angular.z = std::numeric_limits<double>::quiet_NaN();
}

void reset_controller_reference_msg(
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

do we need this specialization?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You're right, this is an unnecessary specialization. I removed it

Comment on lines 656 to 658
// store (for open loop odometry) and set commands
last_linear_velocity_ = timeout ? 0.0 : reference_interfaces_[0];
last_angular_velocity_ = timeout ? 0.0 : reference_interfaces_[1];
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why do we need this here, and haven't needed it before?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry, this slipped in from some playing around in the code. I removed it now

std::string steering_interface_name_ = "position";
// defined in setup
std::string traction_interface_name_ = "";
std::string preceeding_prefix_ = "pid_controller";
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
std::string preceeding_prefix_ = "pid_controller";
std::string preceding_prefix_ = "pid_controller";

we already discussed this, haven't this changed?
However, this is not used at all. Please don't copy dead code.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

Controller handling different input types

3 participants