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

Feature/set utm service #856

Merged
merged 3 commits into from
Jan 8, 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
1 change: 1 addition & 0 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ rosidl_generate_interfaces(${PROJECT_NAME}
"srv/GetState.srv"
"srv/SetDatum.srv"
"srv/SetPose.srv"
"srv/SetUTMZone.srv"
"srv/ToggleFilterProcessing.srv"
"srv/ToLL.srv"
DEPENDENCIES
Expand Down
6 changes: 3 additions & 3 deletions include/robot_localization/navsat_conversions.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -50,12 +50,12 @@
#include <stdio.h>
#include <stdlib.h>

#include <GeographicLib/MGRS.hpp>
#include <GeographicLib/UTMUPS.hpp>

#include <cmath>
#include <string>

#include <GeographicLib/MGRS.hpp>
#include <GeographicLib/UTMUPS.hpp>

namespace robot_localization
{
namespace navsat_conversions
Expand Down
29 changes: 26 additions & 3 deletions include/robot_localization/navsat_transform.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,7 @@
#include "rclcpp/timer.hpp"
#include "robot_localization/srv/from_ll.hpp"
#include "robot_localization/srv/set_datum.hpp"
#include "robot_localization/srv/set_utm_zone.hpp"
#include "robot_localization/srv/to_ll.hpp"
#include "sensor_msgs/msg/imu.hpp"
#include "sensor_msgs/msg/nav_sat_fix.hpp"
Expand Down Expand Up @@ -99,6 +100,13 @@ class NavSatTransform : public rclcpp::Node
const std::shared_ptr<robot_localization::srv::FromLL::Request> request,
std::shared_ptr<robot_localization::srv::FromLL::Response> response);

/**
* @brief Callback for the UTM zone service
*/
bool setUTMZoneCallback(
const std::shared_ptr<robot_localization::srv::SetUTMZone::Request> request,
std::shared_ptr<robot_localization::srv::SetUTMZone::Response>);

/**
* @brief Given the pose of the navsat sensor in the Cartesian frame, removes the
* offset from the vehicle's centroid and returns the Cartesian-frame pose of said
Expand Down Expand Up @@ -216,6 +224,11 @@ class NavSatTransform : public rclcpp::Node
*/
rclcpp::Service<robot_localization::srv::FromLL>::SharedPtr from_ll_srv_;

/**
* @brief Service for set UTM zone
*/
rclcpp::Service<robot_localization::srv::SetUTMZone>::SharedPtr set_utm_zone_srv_;

/**
* @brief Navsatfix publisher
*/
Expand Down Expand Up @@ -370,6 +383,11 @@ class NavSatTransform : public rclcpp::Node
*/
bool use_local_cartesian_;

/**
* @brief Whether we want to force the user's UTM zone and not rely on current GPS data for determining it
*/
bool force_user_utm_;

//! @brief Local Cartesian projection around gps origin
//!
GeographicLib::LocalCartesian gps_local_cartesian_;
Expand Down Expand Up @@ -411,9 +429,14 @@ class NavSatTransform : public rclcpp::Node
tf2::Transform cartesian_world_trans_inverse_;

/**
* @brief Cartesian zone as determined after transforming GPS message
* @brief @brief the UTM zone (zero means UPS)
*/
std::string utm_zone_;
int utm_zone_;

/**
* @brief hemisphere (true means north, false means south)
*/
bool northp_;

/**
* @brief Frame ID of the GPS odometry output
Expand Down Expand Up @@ -446,7 +469,7 @@ class NavSatTransform : public rclcpp::Node
* here until the odom message is received, and the manual datum pose can be
* set.
*/
geographic_msgs::msg::GeoPose manual_datum_geopose_;
geographic_msgs::msg::GeoPose manual_datum_geopose_;
};

} // namespace robot_localization
Expand Down
86 changes: 57 additions & 29 deletions src/navsat_transform.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -80,11 +80,13 @@ NavSatTransform::NavSatTransform(const rclcpp::NodeOptions & options)
transform_good_(false),
transform_timeout_(0ns),
use_local_cartesian_(false),
force_user_utm_(false),
use_manual_datum_(false),
use_odometry_yaw_(false),
cartesian_broadcaster_(*this),
utm_meridian_convergence_(0.0),
utm_zone_(""),
utm_zone_(0),
northp_(true),
world_frame_id_("odom"),
yaw_offset_(0.0),
zero_altitude_(false)
Expand Down Expand Up @@ -149,6 +151,9 @@ NavSatTransform::NavSatTransform(const rclcpp::NodeOptions & options)
from_ll_srv_ = this->create_service<robot_localization::srv::FromLL>(
"fromLL", std::bind(&NavSatTransform::fromLLCallback, this, _1, _2));

set_utm_zone_srv_ = this->create_service<robot_localization::srv::SetUTMZone>(
"setUTMZone", std::bind(&NavSatTransform::setUTMZoneCallback, this, _1, _2));

std::vector<double> datum_vals;
if (use_manual_datum_) {
datum_vals = this->declare_parameter("datum", datum_vals);
Expand Down Expand Up @@ -451,13 +456,12 @@ bool NavSatTransform::fromLLCallback(
cartesian_y,
cartesian_z);
} else {
std::string utm_zone_tmp;
navsat_conversions::LLtoUTM(
latitude,
longitude,
cartesian_y,
cartesian_x,
utm_zone_tmp);
// Transform to UTM using the fixed utm_zone_
int zone_tmp;
bool northp_tmp;
GeographicLib::UTMUPS::Forward(
latitude, longitude,
zone_tmp, northp_tmp, cartesian_x, cartesian_y, utm_zone_);
}

cartesian_pose.setOrigin(tf2::Vector3(cartesian_x, cartesian_y, altitude));
Expand All @@ -473,6 +477,25 @@ bool NavSatTransform::fromLLCallback(
return true;
}

bool NavSatTransform::setUTMZoneCallback(
const std::shared_ptr<robot_localization::srv::SetUTMZone::Request> request,
std::shared_ptr<robot_localization::srv::SetUTMZone::Response>)
{
double x_unused;
double y_unused;
int prec_unused;
GeographicLib::MGRS::Reverse(
request->utm_zone, utm_zone_, northp_, x_unused, y_unused,
prec_unused, true);
// Toggle flags such that transforms get updated to user utm zone
force_user_utm_ = true;
use_manual_datum_ = false;
transform_good_ = false;
has_transform_gps_ = false;
RCLCPP_INFO(this->get_logger(), "UTM zone set to %d %s", utm_zone_, northp_ ? "north" : "south");
return true;
}

nav_msgs::msg::Odometry NavSatTransform::cartesianToMap(
const tf2::Transform & cartesian_pose) const
{
Expand Down Expand Up @@ -512,10 +535,11 @@ void NavSatTransform::mapToLL(
odom_as_cartesian.setRotation(tf2::Quaternion::getIdentity());

// Now convert the data back to lat/long and place into the message
navsat_conversions::UTMtoLL(
odom_as_cartesian.getOrigin().getY(),
odom_as_cartesian.getOrigin().getX(),
GeographicLib::UTMUPS::Reverse(
utm_zone_,
northp_,
odom_as_cartesian.getOrigin().getX(),
odom_as_cartesian.getOrigin().getY(),
latitude,
longitude);
altitude = odom_as_cartesian.getOrigin().getZ();
Expand Down Expand Up @@ -637,13 +661,11 @@ void NavSatTransform::gpsFixCallback(

double cartesian_x = 0;
double cartesian_y = 0;
std::string cartesian_zone_tmp;
navsat_conversions::LLtoUTM(
msg->latitude,
msg->longitude,
cartesian_y,
cartesian_x,
cartesian_zone_tmp);
int zone_tmp;
bool northp_tmp;
GeographicLib::UTMUPS::Forward(
msg->latitude, msg->longitude,
zone_tmp, northp_tmp, cartesian_x, cartesian_y);
latest_cartesian_pose_.setOrigin(tf2::Vector3(cartesian_x, cartesian_y, msg->altitude));
latest_cartesian_covariance_.setZero();

Expand Down Expand Up @@ -861,23 +883,29 @@ void NavSatTransform::setTransformGps(
// UTM meridian convergence is not meaningful when using local cartesian, so set it to 0.0
utm_meridian_convergence_ = 0.0;
} else {
navsat_conversions::LLtoUTM(
msg->latitude,
msg->longitude,
cartesian_y,
cartesian_x,
utm_zone_,
utm_meridian_convergence_);
utm_meridian_convergence_ *= navsat_conversions::RADIANS_PER_DEGREE;
double k_tmp;
double utm_meridian_convergence_degrees;
try {
// If we're using a fixed UTM zone, then we want to use the zone that the user gave us.
int set_zone = force_user_utm_ ? utm_zone_ : -1;
GeographicLib::UTMUPS::Forward(
msg->latitude, msg->longitude, utm_zone_, northp_,
cartesian_x, cartesian_y, utm_meridian_convergence_degrees, k_tmp, set_zone);
} catch (const GeographicLib::GeographicErr & e) {
RCLCPP_ERROR_STREAM(this->get_logger(), e.what());
return;
}
utm_meridian_convergence_ = utm_meridian_convergence_degrees *
navsat_conversions::RADIANS_PER_DEGREE;
}

RCLCPP_INFO(
this->get_logger(), "Datum (latitude, longitude, altitude) is (%0.2f, %0.2f, %0.2f)",
msg->latitude, msg->longitude, msg->altitude);
RCLCPP_INFO(
this->get_logger(), "Datum %s coordinate is (%s, %0.2f, %0.2f)",
((use_local_cartesian_) ? "Local Cartesian" : "UTM"), utm_zone_.c_str(), cartesian_x,
cartesian_y);
this->get_logger(), "Datum %s coordinate is (%d %s, %0.2f, %0.2f)",
((use_local_cartesian_) ? "Local Cartesian" : "UTM"),
utm_zone_, (northp_ ? "north" : "south"), cartesian_x, cartesian_y);

transform_cartesian_pose_.setOrigin(tf2::Vector3(cartesian_x, cartesian_y, msg->altitude));
transform_cartesian_pose_.setRotation(tf2::Quaternion::getIdentity());
Expand Down
4 changes: 4 additions & 0 deletions srv/SetUTMZone.srv
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
# Set an UTM zone navsat_transform should stick to.
# Example: 31U
string utm_zone
---