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

Add Lie group capabilities to NavState #1613

Draft
wants to merge 16 commits into
base: develop
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from 13 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
4 changes: 2 additions & 2 deletions gtsam/geometry/Pose3.h
Original file line number Diff line number Diff line change
Expand Up @@ -135,10 +135,10 @@ class GTSAM_EXPORT Pose3: public LieGroup<Pose3, 6> {
/// @name Lie Group
/// @{

/// Exponential map at identity - create a rotation from canonical coordinates \f$ [R_x,R_y,R_z,T_x,T_y,T_z] \f$
/// Exponential map at identity - create a pose from canonical coordinates \f$ [R_x,R_y,R_z,T_x,T_y,T_z] \f$
static Pose3 Expmap(const Vector6& xi, OptionalJacobian<6, 6> Hxi = {});

/// Log map at identity - return the canonical coordinates \f$ [R_x,R_y,R_z,T_x,T_y,T_z] \f$ of this rotation
/// Log map at identity - return the canonical coordinates \f$ [R_x,R_y,R_z,T_x,T_y,T_z] \f$ of this pose
static Vector6 Logmap(const Pose3& pose, OptionalJacobian<6, 6> Hpose = {});

/**
Expand Down
2 changes: 1 addition & 1 deletion gtsam/geometry/Rot3.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -120,7 +120,7 @@ Unit3 Rot3::unrotate(const Unit3& p,
OptionalJacobian<2,3> HR, OptionalJacobian<2,2> Hp) const {
Matrix32 Dp;
Unit3 q = Unit3(unrotate(p.point3(Dp)));
if (Hp) *Hp = q.basis().transpose() * matrix().transpose () * Dp;
if (Hp) *Hp = q.basis().transpose() * matrix().transpose() * Dp;
if (HR) *HR = q.basis().transpose() * q.skew();
return q;
}
Expand Down
301 changes: 291 additions & 10 deletions gtsam/navigation/NavState.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -77,10 +77,13 @@ Vector3 NavState::bodyVelocity(OptionalJacobian<3, 9> H) const {
}

//------------------------------------------------------------------------------
Matrix7 NavState::matrix() const {
Matrix5 NavState::matrix() const {
Matrix3 R = this->R();
Matrix7 T;
T << R, Z_3x3, t(), Z_3x3, R, v(), Vector6::Zero().transpose(), 1.0;

Matrix5 T = Matrix5::Identity();
T.block<3, 3>(0, 0) = R;
T.block<3, 1>(0, 3) = t_;
T.block<3, 1>(0, 4) = v_;
return T;
}

Expand All @@ -103,9 +106,261 @@ bool NavState::equals(const NavState& other, double tol) const {
&& equal_with_abs_tol(v_, other.v_, tol);
}

//------------------------------------------------------------------------------
NavState NavState::inverse() const {
Rot3 Rt = R_.inverse();
return NavState(Rt, Rt * (-t_), Rt * -(v_));
}

//------------------------------------------------------------------------------
NavState NavState::Expmap(const Vector9& xi, OptionalJacobian<9, 9> Hxi) {
if (Hxi) *Hxi = ExpmapDerivative(xi);

// Order is rotation, position, velocity, represented by ω,ρ,ν
Vector3 omega(xi(0), xi(1), xi(2)), rho(xi(3), xi(4), xi(5)),
nu(xi(6), xi(7), xi(8));

Rot3 R = Rot3::Expmap(omega);

double omega_norm = omega.norm();

if (omega_norm < 1e-8) {
return NavState(Rot3(), Point3(rho), Point3(nu));

} else {
Matrix W = skewSymmetric(omega);
double omega_norm2 = omega_norm * omega_norm;
double omega_norm3 = omega_norm2 * omega_norm;
Matrix A = I_3x3 + ((1 - cos(omega_norm)) / omega_norm2) * W +
((omega_norm - sin(omega_norm)) / omega_norm3) * (W * W);

return NavState(Rot3::Expmap(omega), Point3(A * rho), Point3(A * nu));
}
}

//------------------------------------------------------------------------------
Vector9 NavState::Logmap(const NavState& state, OptionalJacobian<9, 9> Hstate) {
if (Hstate) *Hstate = LogmapDerivative(state);

const Vector3 phi = Rot3::Logmap(state.rotation());
const Vector3& p = state.position();
const Vector3& v = state.velocity();
const double t = phi.norm();
if (t < 1e-8) {
Vector9 log;
log << phi, p, v;
return log;

} else {
const Matrix3 W = skewSymmetric(phi / t);

const double Tan = tan(0.5 * t);
const Vector3 Wp = W * p;
const Vector3 Wv = W * v;
const Vector3 rho = p - (0.5 * t) * Wp + (1 - t / (2. * Tan)) * (W * Wp);
const Vector3 nu = v - (0.5 * t) * Wv + (1 - t / (2. * Tan)) * (W * Wv);
Vector9 log;
// Order is ω, p, v
log << phi, rho, nu;
return log;
}
}

//------------------------------------------------------------------------------
Matrix9 NavState::AdjointMap() const {
const Matrix3 R = R_.matrix();
Matrix3 A = skewSymmetric(t_.x(), t_.y(), t_.z()) * R;
Matrix3 B = skewSymmetric(v_.x(), v_.y(), v_.z()) * R;
// Eqn 2 in Barrau20icra
Matrix9 adj;
adj << R, Z_3x3, Z_3x3, A, R, Z_3x3, B, Z_3x3, R;
return adj;
}

//------------------------------------------------------------------------------
Vector9 NavState::Adjoint(const Vector9& xi_b, OptionalJacobian<9, 9> H_state,
OptionalJacobian<9, 9> H_xib) const {
const Matrix9 Ad = AdjointMap();

// Jacobians
if (H_state) *H_state = -Ad * adjointMap(xi_b);
if (H_xib) *H_xib = Ad;

return Ad * xi_b;
}

//------------------------------------------------------------------------------
/// The dual version of Adjoint
Vector9 NavState::AdjointTranspose(const Vector9& x, OptionalJacobian<9, 9> H_state,
OptionalJacobian<9, 9> H_x) const {
const Matrix9 Ad = AdjointMap();
const Vector9 AdTx = Ad.transpose() * x;

// Jacobians
if (H_state) {
//TODO(Varun)
// const auto w_T_hat = skewSymmetric(AdTx.head<3>()),
// v_T_hat = skewSymmetric(AdTx.segment<3>(3)),
// a_T_hat = skewSymmetric(AdTx.tail<3>());
// *H_state << w_T_hat, v_T_hat, //
// /* */ v_T_hat, Z_3x3;
throw std::runtime_error(
"AdjointTranpose H_state Jacobian not implemented.");
}
if (H_x) {
*H_x = Ad.transpose();
}

return AdTx;
}

//------------------------------------------------------------------------------
Matrix9 NavState::adjointMap(const Vector9& xi) {
Matrix3 w_hat = skewSymmetric(xi(0), xi(1), xi(2));
Matrix3 v_hat = skewSymmetric(xi(3), xi(4), xi(5));
Matrix3 a_hat = skewSymmetric(xi(6), xi(7), xi(8));
Matrix9 adj;
adj << w_hat, Z_3x3, Z_3x3, v_hat, w_hat, Z_3x3, a_hat, Z_3x3, w_hat;
return adj;
}

//------------------------------------------------------------------------------
Vector9 NavState::adjoint(const Vector9& xi, const Vector9& y,
OptionalJacobian<9, 9> Hxi,
OptionalJacobian<9, 9> H_y) {
if (Hxi) {
Hxi->setZero();
for (int i = 0; i < 9; ++i) {
Vector9 dxi;
dxi.setZero();
dxi(i) = 1.0;
Matrix9 Gi = adjointMap(dxi);
Hxi->col(i) = Gi * y;
}
}

const Matrix9& ad_xi = adjointMap(xi);
if (H_y) *H_y = ad_xi;

return ad_xi * y;
}

//------------------------------------------------------------------------------
Vector9 NavState::adjointTranspose(const Vector9& xi, const Vector9& y,
OptionalJacobian<9, 9> Hxi,
OptionalJacobian<9, 9> H_y) {
if (Hxi) {
Hxi->setZero();
for (int i = 0; i < 9; ++i) {
Vector9 dxi;
dxi.setZero();
dxi(i) = 1.0;
Matrix9 GTi = adjointMap(dxi).transpose();
Hxi->col(i) = GTi * y;
}
}
const Matrix9& adT_xi = adjointMap(xi).transpose();
if (H_y) *H_y = adT_xi;
return adT_xi * y;
}

/* ************************************************************************* */
Matrix63 NavState::ComputeQforExpmapDerivative(const Vector9& xi,
double nearZeroThreshold) {
const auto omega = xi.head<3>();
const auto nu = xi.segment<3>(3);
const auto rho = xi.tail<3>();
const Matrix3 V = skewSymmetric(nu);
const Matrix3 P = skewSymmetric(rho);
const Matrix3 W = skewSymmetric(omega);

Matrix3 Qv, Qp;
Matrix63 Q;

// The closed-form formula in Barfoot14tro eq. (102)
double phi = omega.norm();
if (std::abs(phi) > 1e-5) {
const double sinPhi = sin(phi), cosPhi = cos(phi);
const double phi2 = phi * phi, phi3 = phi2 * phi, phi4 = phi3 * phi,
phi5 = phi4 * phi;
// Invert the sign of odd-order terms to have the right Jacobian
Qv = -0.5 * V + (phi - sinPhi) / phi3 * (W * V + V * W - W * V * W) +
(1 - phi2 / 2 - cosPhi) / phi4 *
(W * W * V + V * W * W - 3 * W * V * W) -
0.5 *
((1 - phi2 / 2 - cosPhi) / phi4 -
3 * (phi - sinPhi - phi3 / 6.) / phi5) *
(W * V * W * W + W * W * V * W);
Qp = -0.5 * P + (phi - sinPhi) / phi3 * (W * P + P * W - W * P * W) +
(1 - phi2 / 2 - cosPhi) / phi4 *
(W * W * P + P * W * W - 3 * W * P * W) -
0.5 *
((1 - phi2 / 2 - cosPhi) / phi4 -
3 * (phi - sinPhi - phi3 / 6.) / phi5) *
(W * P * W * W + W * W * P * W);
} else {
Qv = -0.5 * V + 1. / 6. * (W * V + V * W - W * V * W) +
1. / 24. * (W * W * V + V * W * W - 3 * W * V * W) -
0.5 * (1. / 24. + 3. / 120.) * (W * V * W * W + W * W * V * W);
Qp = -0.5 * P + 1. / 6. * (W * P + P * W - W * P * W) +
1. / 24. * (W * W * P + P * W * W - 3 * W * P * W) -
0.5 * (1. / 24. + 3. / 120.) * (W * P * W * W + W * W * P * W);
}

Q << Qv, Qp;
return Q;
}

//------------------------------------------------------------------------------
Matrix9 NavState::ExpmapDerivative(const Vector9& xi) {
const Vector3 w = xi.head<3>();
const Matrix3 Jw = Rot3::ExpmapDerivative(w);
const Matrix63 Q = ComputeQforExpmapDerivative(xi);
const Matrix3 Qv = Q.topRows<3>();
const Matrix3 Qp = Q.bottomRows<3>();

Matrix9 J;
J << Jw, Z_3x3, Z_3x3, Qv, Jw, Z_3x3, Qp, Z_3x3, Jw;

return J;
}

//------------------------------------------------------------------------------
Matrix9 NavState::LogmapDerivative(const NavState& state) {
const Matrix3 R = state.rotation().matrix();

const Vector9 xi = Logmap(state);
const Vector3 w = xi.head<3>();
const Matrix3 Jw = Rot3::LogmapDerivative(w);
const Matrix63 Q = ComputeQforExpmapDerivative(xi);
const Matrix3 Qv = Q.topRows<3>();
const Matrix3 Qp = Q.bottomRows<3>();
const Matrix3 Qv2 = -Jw * Qv * Jw;
const Matrix3 Qp2 = -Jw * Qp * Jw;

Matrix9 J;
J << Jw, Z_3x3, Z_3x3, Qv2, Jw, Z_3x3, Qp2, Z_3x3, Jw;
return J;
}


//------------------------------------------------------------------------------
NavState NavState::ChartAtOrigin::Retract(const Vector9& xi,
ChartJacobian Hxi) {
return Expmap(xi, Hxi);
}

//------------------------------------------------------------------------------
Vector9 NavState::ChartAtOrigin::Local(const NavState& state,
ChartJacobian Hstate) {
return Logmap(state, Hstate);
}

//------------------------------------------------------------------------------
NavState NavState::retract(const Vector9& xi, //
OptionalJacobian<9, 9> H1, OptionalJacobian<9, 9> H2) const {
// return LieGroup<NavState, 9>::retract(xi, H1, H2);

Rot3 nRb = R_;
Point3 n_t = t_, n_v = v_;
Matrix3 D_bRc_xi, D_R_nRb, D_t_nRb, D_v_nRb;
Expand Down Expand Up @@ -133,6 +388,30 @@ NavState NavState::retract(const Vector9& xi, //
//------------------------------------------------------------------------------
Vector9 NavState::localCoordinates(const NavState& g, //
OptionalJacobian<9, 9> H1, OptionalJacobian<9, 9> H2) const {
// return LieGroup<NavState, 9>::localCoordinates(g, H1, H2);
Copy link
Member

Choose a reason for hiding this comment

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

Why was retract unchanged but local coordinates changed? And, what’s the difference with example and logmap?

Copy link
Member

Choose a reason for hiding this comment

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


//TODO(Varun) Fix so that test on L680 passes

// Matrix3 D_dR_R, D_dt_R, D_dv_R;
// const Rot3 dR = R_.between(g.R_, H1 ? &D_dR_R : 0);
// const Point3 dP = R_.unrotate(g.t_ - t_, H1 ? &D_dt_R : 0);
// const Vector dV = R_.unrotate(g.v_ - v_, H1 ? &D_dv_R : 0);

// Vector9 xi;
// Matrix3 D_xi_R;
// xi << Rot3::Logmap(dR, (H1 || H2) ? &D_xi_R : 0), dP, dV;
// if (H1) {
// *H1 << D_xi_R * D_dR_R, Z_3x3, Z_3x3, //
// D_dt_R, -I_3x3, Z_3x3, //
// D_dv_R, Z_3x3, -I_3x3;
// }
// if (H2) {
// *H2 << D_xi_R, Z_3x3, Z_3x3, //
// Z_3x3, dR.matrix(), Z_3x3, //
// Z_3x3, Z_3x3, dR.matrix();
// }
// return xi;

Matrix3 D_dR_R, D_dt_R, D_dv_R;
const Rot3 dR = R_.between(g.R_, H1 ? &D_dR_R : 0);
const Point3 dP = R_.unrotate(g.t_ - t_, H1 ? &D_dt_R : 0);
Expand All @@ -142,15 +421,16 @@ Vector9 NavState::localCoordinates(const NavState& g, //
Matrix3 D_xi_R;
xi << Rot3::Logmap(dR, (H1 || H2) ? &D_xi_R : 0), dP, dV;
if (H1) {
*H1 << D_xi_R * D_dR_R, Z_3x3, Z_3x3, //
D_dt_R, -I_3x3, Z_3x3, //
D_dv_R, Z_3x3, -I_3x3;
*H1 << D_xi_R * D_dR_R, Z_3x3, Z_3x3, //
D_dt_R, -I_3x3, Z_3x3, //
D_dv_R, Z_3x3, -I_3x3;
}
if (H2) {
*H2 << D_xi_R, Z_3x3, Z_3x3, //
Z_3x3, dR.matrix(), Z_3x3, //
Z_3x3, Z_3x3, dR.matrix();
*H2 << D_xi_R, Z_3x3, Z_3x3, //
Z_3x3, dR.matrix(), Z_3x3, //
Z_3x3, Z_3x3, dR.matrix();
}

return xi;
}

Expand Down Expand Up @@ -213,7 +493,8 @@ NavState NavState::update(const Vector3& b_acceleration, const Vector3& b_omega,
//------------------------------------------------------------------------------
Vector9 NavState::coriolis(double dt, const Vector3& omega, bool secondOrder,
OptionalJacobian<9, 9> H) const {
auto [nRb, n_t, n_v] = (*this);
Rot3 nRb = R_;
Point3 n_t = t_, n_v = v_;

const double dt2 = dt * dt;
const Vector3 omega_cross_vel = omega.cross(n_v);
Expand Down
Loading
Loading