diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 030e5abc..7ed8f737 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -19,7 +19,7 @@ jobs: strategy: fail-fast: false matrix: - os: [ubuntu-16.04, ubuntu-18.04, ubuntu-20.04] + os: [ubuntu-18.04, ubuntu-20.04] build_type: [Release, Debug] compiler: [{ "cc": "gcc", diff --git a/CMakeLists.txt b/CMakeLists.txt index 13f8dbbc..f86147b4 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -252,6 +252,8 @@ if(ENABLE_CPPCHECK) --suppress=unknownMacro:*test/ceres/ceres_test_utils.h --suppress=constStatement:*examples* --suppress=unreadVariable:*gtest_se2_autodiff.cpp:101 + --suppress=unreadVariable:*bundle_sam.cpp:94 + --suppress=unreadVariable:*bundle_sam.cpp:124 # Uncomment the line below to run cppcheck-html # --output-file=${CMAKE_BINARY_DIR}/cppcheck_results.xml ${CMAKE_SOURCE_DIR}/include diff --git a/README.md b/README.md index e34d9a0b..6c18c2a1 100644 --- a/README.md +++ b/README.md @@ -26,6 +26,9 @@ At the moment, it provides the groups: introduced (to the best of knowledge) in this [paper][barrau15]. NOTE: The implementation here differs slightly from the developments in the [paper][barrau15]. +- Bundle<>: allows manipulating a manifold bundle as a single Lie group. + Referred to as a *composite manifold* in Section IV of the + [reference paper](http://arxiv.org/abs/1812.01537). Other Lie groups can and will be added, contributions are welcome. diff --git a/examples/CMakeLists.txt b/examples/CMakeLists.txt index 4f0548d4..abcff5a9 100644 --- a/examples/CMakeLists.txt +++ b/examples/CMakeLists.txt @@ -36,6 +36,18 @@ set(CXX_11_EXAMPLE_TARGETS se_2_3_localization ) +if(NOT MSVC) + add_executable(bundle_sam bundle_sam.cpp) + + set(CXX_11_EXAMPLE_TARGETS + + ${CXX_11_EXAMPLE_TARGETS} + + # Bundle + bundle_sam + ) +endif() + # Link to manif foreach(target ${CXX_11_EXAMPLE_TARGETS}) target_link_libraries(${target} ${PROJECT_NAME}) @@ -45,7 +57,7 @@ foreach(target ${CXX_11_EXAMPLE_TARGETS}) target_compile_options(${target} PRIVATE -Werror=all -Werror=extra - ) + ) endif() endforeach() diff --git a/examples/bundle_sam.cpp b/examples/bundle_sam.cpp new file mode 100644 index 00000000..06c63d41 --- /dev/null +++ b/examples/bundle_sam.cpp @@ -0,0 +1,386 @@ +/** + * \file bundle_sam.cpp + * + * Created on: Feb 28, 2021 + * \author: pettni + * + * Adopted from se2_sam.cpp, see that file for explanations. + */ + + +// manif +#include "manif/Rn.h" +#include "manif/SE2.h" +#include "manif/Bundle.h" + +// Std +#include +#include +#include +#include + +// Debug +#include +#include + +// std shortcuts and namespaces +using std::cout; +using std::endl; +using std::vector; +using std::map; +using std::list; +using std::pair; + +// Eigen namespace +using namespace Eigen; + +// manif namespace and shortcuts +using manif::SE2d; +using manif::SE2Tangentd; + +static constexpr int DoF = SE2d::DoF; +static constexpr int Dim = SE2d::Dim; + +// Define many data types (Tangent refers to the tangent of SE2) +typedef Array ArrayT; // tangent-size array +typedef Matrix VectorT; // tangent-size vector +typedef Matrix MatrixT; // tangent-size square matrix +typedef Matrix VectorB; // landmark-size vector +typedef Array ArrayY; // measurement-size array +typedef Matrix VectorY; // measurement-size vector +typedef Matrix MatrixY; // measurement-size square matrix +typedef Matrix MatrixYT; // measurement x tangent size matrix +typedef Matrix MatrixYB; // measurement x landmark size matrix + +// some experiment constants +static const int NUM_POSES = 3; +static const int NUM_LMKS = 5; +static const int NUM_FACTORS = 9; +static const int NUM_MEAS = NUM_POSES * DoF + NUM_FACTORS * Dim; +static const int MAX_ITER = 20; // for the solver + + +// bundle state type to optimize over +using BundleT = manif::Bundle; + +// Insert a relative pose factor from pose XI to pose XJ +// into the residual-jacobian pair (r, J) +template +void add_pose_factor( + const BundleT & X, + const SE2Tangentd & control, + const MatrixT & W, + Eigen::Ref> r, + Eigen::Ref> J) +{ + // index start position and length in the DoF of BundleT + const int BegI = std::get(manif::internal::traits::DoFIdx); + constexpr int LenI = BundleT::Element::DoF; + const int BegJ = std::get(manif::internal::traits::DoFIdx); + constexpr int LenJ = BundleT::Element::DoF; + + MatrixT J_d_xi, J_d_xj; // Jacobian of motion wrt poses i and j + + auto d = X.element().rminus(X.element(), J_d_xj, J_d_xi); + r = W * (d - control).coeffs(); + J.setZero(); + J.block<3, LenI>(0, BegI) = W * J_d_xi; + J.block<3, LenJ>(0, BegJ) = W * J_d_xj; +} + + +// Insert a landmark measurement factor of landmark LK +// from pose XI into the residual-jacobian pair (r, J) +template +void add_beacon_factor( + const BundleT & X, + const VectorY & measurement, + const MatrixY & S, + Eigen::Ref> r, + Eigen::Ref> J) +{ + // index start position and length in the DoF of BundleT + const int BegX = std::get(manif::internal::traits::DoFIdx); + constexpr int LenX = BundleT::Element::DoF; + const int BegLMK = std::get(manif::internal::traits::DoFIdx); + constexpr int LenLMK = BundleT::Element::DoF; + + MatrixT J_ix_x; // Jacobian of inverse pose wrt pose + MatrixYT J_e_ix; // Jacobian of measurement expectation wrt inverse pose + MatrixYT J_e_x; // Jacobian of measurement expectation wrt pose + MatrixYB J_e_b; // Jacobian of measurement expectation wrt lmk + + auto e = X.element().inverse(J_ix_x).act(X.element().coeffs(), J_e_ix, J_e_b); + J_e_x = J_e_ix * J_ix_x; + r = S * (e - measurement); + J.setZero(); + J.block(0, BegX) = S * J_e_x; + J.block(0, BegLMK) = S * J_e_b; +} + + +int main() +{ + std::srand((unsigned int) time(0)); + + // DEBUG INFO + cout << endl; + cout << "2D Smoothing and Mapping. 3 poses, 5 landmarks." << endl; + cout << "-----------------------------------------------" << endl; + cout << std::fixed << std::setprecision(3) << std::showpos; + + // START CONFIGURATION + // + // + + // Define the robot pose elements + SE2d X_simu, // pose of the simulated robot + Xi, // robot pose at time i + Xj; // robot pose at time j + vector poses, // estimator poses + poses_simu;// simulator poses + Xi.setIdentity(); + X_simu.setIdentity(); + + + // Define a control vector and its noise and covariance in the tangent of SE2 + SE2Tangentd u; // control signal, generic + SE2Tangentd u_nom; // nominal control signal + ArrayT u_sigmas; // control noise std specification + VectorT u_noise; // control noise + // MatrixT Q; // Covariance + MatrixT W; // sqrt Info + vector controls; // robot controls + + u_nom << 0.1, 0.0, 0.05; + u_sigmas << 0.01, 0.01, 0.01; + // Q = (u_sigmas * u_sigmas).matrix().asDiagonal(); + W = u_sigmas.inverse() .matrix().asDiagonal(); // this is Q^(-T/2) + + // Landmarks in R^2 and map + VectorB b; // Landmark, generic + vector landmarks(NUM_LMKS), landmarks_simu; + { + // Define five landmarks (beacons) in R^2 + VectorB b0, b1, b2, b3, b4; + b0 << 3.0, 0.0; + b1 << 2.0, -1.0; + b2 << 2.0, 1.0; + b3 << 3.0, -1.0; + b4 << 3.0, 1.0; + landmarks_simu.push_back(b0); + landmarks_simu.push_back(b1); + landmarks_simu.push_back(b2); + landmarks_simu.push_back(b3); + landmarks_simu.push_back(b4); + } // destroy b0...b4 + + // Define the beacon's measurements in R^2 + VectorY y, y_noise; + ArrayY y_sigmas; + // MatrixY R; // Covariance + MatrixY S; // sqrt Info + vector> measurements(NUM_POSES); // y = measurements[pose_id][lmk_id] + + y_sigmas << 0.001, 0.001; + // R = (y_sigmas * y_sigmas).matrix().asDiagonal(); + S = y_sigmas.inverse() .matrix().asDiagonal(); // this is R^(-T/2) + + // Problem-size variables + + /* + * The factor graph of the SAM problem looks like this: + * + * ------- b1 + * b3 / | + * | / b4 | + * | / / \| + * X0 ---- X1 ---- X2 + * | \ / \ / + * | b0 b2 + * * + * + * where: + * - Xi are poses + * - bk are landmarks or beacons + * - * is a pose prior to anchor the map and make the problem observable + * + * Define pairs of nodes for all the landmark measurements + * There are 3 pose nodes [0..2] and 5 landmarks [0..4]. + * A pair pose -- lmk means that the lmk was measured from the pose + * Each pair declares a factor in the factor graph + * We declare 9 pairs, or 9 factors, as follows: + */ + vector> pairs(NUM_POSES); + pairs[0].push_back(0); // 0-0 + pairs[0].push_back(1); // 0-1 + pairs[0].push_back(3); // 0-3 + pairs[1].push_back(0); // 1-0 + pairs[1].push_back(2); // 1-2 + pairs[1].push_back(4); // 1-4 + pairs[2].push_back(1); // 2-1 + pairs[2].push_back(2); // 2-2 + pairs[2].push_back(4); // 2-4 + + // + // + // END CONFIGURATION + + + //// Simulator ################################################################### + poses_simu. push_back(X_simu); + poses. push_back(Xi + SE2Tangentd::Random()); // use very noisy priors + + // temporal loop + for (int i = 0; i < NUM_POSES; ++i) + { + // make measurements + for (const auto& k : pairs[i]) + { + // simulate measurement + b = landmarks_simu[k]; // lmk coordinates in world frame + y_noise = y_sigmas * ArrayY::Random(); // measurement noise + y = X_simu.inverse().act(b); // landmark measurement, before adding noise + + // add noise and compute prior lmk from prior pose + measurements[i][k] = y + y_noise; // store noisy measurements + b = Xi.act(y + y_noise); // mapped landmark with noise + landmarks[k] = b + VectorB::Random(); // use very noisy priors + } + + // make motions + if (i < NUM_POSES - 1) // do not make the last motion since we're done after 3rd pose + { + // move simulator, without noise + X_simu = X_simu + u_nom; + + // move prior, with noise + u_noise = u_sigmas * ArrayT::Random(); + Xi = Xi + (u_nom + u_noise); + + // store + poses_simu. push_back(X_simu); + poses. push_back(Xi + SE2Tangentd::Random()); // use very noisy priors + controls. push_back(u_nom + u_noise); + } + } + + //// Estimator ################################################################# + + // Insert priors into bundle state + BundleT X(poses[0], poses[1], poses[2], + manif::R2d(landmarks[0]), manif::R2d(landmarks[1]), + manif::R2d(landmarks[2]), manif::R2d(landmarks[3]), + manif::R2d(landmarks[4])); + + cout << "prior" << std::showpos << endl; + cout << "pose :" << X.element<0>().translation().transpose() << " " << X.element<0>().angle() << endl; + cout << "pose :" << X.element<1>().translation().transpose() << " " << X.element<1>().angle() << endl; + cout << "pose :" << X.element<2>().translation().transpose() << " " << X.element<2>().angle() << endl; + + cout << "lmk :" << X.element<3>() << endl; + cout << "lmk :" << X.element<4>() << endl; + cout << "lmk :" << X.element<5>() << endl; + cout << "lmk :" << X.element<6>() << endl; + cout << "lmk :" << X.element<7>() << endl; + cout << "-----------------------------------------------" << endl; + + // iterate + // DEBUG INFO + cout << "iterations" << std::noshowpos << endl; + for (int iteration = 0; iteration < MAX_ITER; ++iteration) + { + Matrix J; // full Jacobian + Matrix r; // full residual + + int row = 0; // keep track of row in J and r + + // first residual: prior + r.segment(row) = X.element<0>().lminus(SE2d::Identity(), J.block(row, 0)).coeffs(); + row += DoF; + + // motion residuals + add_pose_factor<0, 1>(X, controls[0], W, r.segment(row), J.block(row, 0)); + row += DoF; + + add_pose_factor<1, 2>(X, controls[1], W, r.segment(row), J.block(row, 0)); + row += DoF; + + // measurement residuals + add_beacon_factor<0, 0>(X, measurements[0][0], S, r.segment(row), J.block(row, 0)); + row += Dim; + add_beacon_factor<0, 1>(X, measurements[0][1], S, r.segment(row), J.block(row, 0)); + row += Dim; + add_beacon_factor<0, 3>(X, measurements[0][3], S, r.segment(row), J.block(row, 0)); + row += Dim; + + add_beacon_factor<1, 0>(X, measurements[1][0], S, r.segment(row), J.block(row, 0)); + row += Dim; + add_beacon_factor<1, 2>(X, measurements[1][2], S, r.segment(row), J.block(row, 0)); + row += Dim; + add_beacon_factor<1, 4>(X, measurements[1][4], S, r.segment(row), J.block(row, 0)); + row += Dim; + + add_beacon_factor<2, 1>(X, measurements[2][1], S, r.segment(row), J.block(row, 0)); + row += Dim; + add_beacon_factor<2, 2>(X, measurements[2][2], S, r.segment(row), J.block(row, 0)); + row += Dim; + add_beacon_factor<2, 4>(X, measurements[2][4], S, r.segment(row), J.block(row, 0)); + row += Dim; + + // 4. Solve ----------------------------------------------------------------- + + // compute optimal step + // ATTENTION: This is an expensive step!! + // ATTENTION: Use QR factorization and column reordering for larger problems!! + const auto dX = (-(J.transpose() * J).inverse() * J.transpose() * r).eval(); + + // update estimate + X += BundleT::Tangent(dX); + + // DEBUG INFO + cout << "residual norm: " << std::scientific << r.norm() << ", step norm: " << dX.norm() << endl; + + // conditional exit + if (dX.norm() < 1e-6) break; + } + cout << "-----------------------------------------------" << endl; + + + //// Print results #################################################################### + + cout << std::fixed; + + // solved problem + cout << "posterior" << std::showpos << endl; + cout << "pose :" << X.element<0>().translation().transpose() << " " << X.element<0>().angle() << endl; + cout << "pose :" << X.element<1>().translation().transpose() << " " << X.element<1>().angle() << endl; + cout << "pose :" << X.element<2>().translation().transpose() << " " << X.element<2>().angle() << endl; + + cout << "lmk :" << X.element<3>() << endl; + cout << "lmk :" << X.element<4>() << endl; + cout << "lmk :" << X.element<5>() << endl; + cout << "lmk :" << X.element<6>() << endl; + cout << "lmk :" << X.element<7>() << endl; + + cout << "-----------------------------------------------" << endl; + + // ground truth + cout << "ground truth" << std::showpos << endl; + for (const auto& ps : poses_simu) + cout << "pose : " << ps.translation().transpose() << " " << ps.angle() << endl; + for (const auto& ls : landmarks_simu) + cout << "lmk : " << ls.transpose() << endl; + cout << "-----------------------------------------------" << endl; + + return 0; +} diff --git a/include/manif/Bundle.h b/include/manif/Bundle.h new file mode 100644 index 00000000..8cf7873c --- /dev/null +++ b/include/manif/Bundle.h @@ -0,0 +1,17 @@ +#ifndef _MANIF_BUNDLE_H_ +#define _MANIF_BUNDLE_H_ + +#include "manif/impl/macro.h" +#include "manif/impl/utils.h" +#include "manif/impl/lie_group_base.h" +#include "manif/impl/tangent_base.h" + +#include "manif/impl/bundle/Bundle_properties.h" +#include "manif/impl/bundle/Bundle_base.h" +#include "manif/impl/bundle/Bundle_map.h" +#include "manif/impl/bundle/Bundle.h" +#include "manif/impl/bundle/BundleTangent_base.h" +#include "manif/impl/bundle/BundleTangent.h" +#include "manif/impl/bundle/BundleTangent_map.h" + +#endif // _MANIF_BUNDLE_H_ diff --git a/include/manif/impl/bundle/Bundle.h b/include/manif/impl/bundle/Bundle.h new file mode 100644 index 00000000..7edf35f2 --- /dev/null +++ b/include/manif/impl/bundle/Bundle.h @@ -0,0 +1,208 @@ +#ifndef _MANIF_MANIF_BUNDLE_H_ +#define _MANIF_MANIF_BUNDLE_H_ + +#include "manif/impl/bundle/Bundle_base.h" +#include "manif/impl/traits.h" + +namespace manif { + +// Forward declare for type traits specialization + +template class ... _T> struct Bundle; +template class ... _T> struct BundleTangent; + +namespace internal { + +//! Traits specialization +template class ... _T> +struct traits> +{ + // Bundle-specific traits + static constexpr std::size_t BundleSize = sizeof...(_T); + + using Elements = std::tuple<_T<_Scalar>...>; + + template + using Element = typename std::tuple_element<_N, Elements>::type; + + template + using MapElement = Eigen::Map>; + + template + using MapConstElement = Eigen::Map>; + + static constexpr std::array DimIdx = compute_indices<_T<_Scalar>::Dim ...>(); + static constexpr std::array DoFIdx = compute_indices<_T<_Scalar>::DoF ...>(); + static constexpr std::array RepSizeIdx = compute_indices<_T<_Scalar>::RepSize ...>(); + static constexpr std::array TraIdx = compute_indices<_T<_Scalar>::Transformation::RowsAtCompileTime ...>(); + + // Regular traits + using Scalar = _Scalar; + + using LieGroup = Bundle<_Scalar, _T ...>; + using Tangent = BundleTangent<_Scalar, _T ...>; + + using Base = BundleBase>; + + static constexpr int Dim = accumulate(int(_T<_Scalar>::Dim) ...); + static constexpr int DoF = accumulate(int(_T<_Scalar>::DoF) ...); + static constexpr int RepSize = accumulate(int(_T<_Scalar>::RepSize) ...); + + using DataType = Eigen::Matrix<_Scalar, RepSize, 1>; + using Jacobian = Eigen::Matrix<_Scalar, DoF, DoF>; + using Transformation = SquareMatrix< + _Scalar, + accumulate(int(_T<_Scalar>::Transformation::RowsAtCompileTime) ...) + >; + using Vector = Eigen::Matrix<_Scalar, Dim, 1>; +}; + +template class ... _T> +const constexpr std::array traits>::DimIdx; +template class ... _T> +const constexpr std::array traits>::DoFIdx; +template class ... _T> +const constexpr std::array traits>::RepSizeIdx; +template class ... _T> +const constexpr std::array traits>::TraIdx; +template class ... _T> +const constexpr int traits>::Dim; +template class ... _T> +const constexpr int traits>::DoF; +template class ... _T> +const constexpr int traits>::RepSize; + +} // namespace internal + +// +// Bundle LieGroup +// + +/** + * @brief Represents a Bundle (or Composite) element as + * described in Section IV of the reference paper + * (see also Example 7). + * + * A Bundle of Lie groups can be utilized as + * a single group with element-wise operations. This can be + * convenient when working with aggregate states that consist of + * multiple Lie group sub-states, like the example in Section VIIb + * of the reference paper. + * + * Example: create an element of the composite + * using double as the scalar type. + * + * > Bundle element; + */ +template class ... _T> +struct Bundle : BundleBase> +{ +private: + + static_assert(sizeof...(_T) > 0, "Must have at least one element in Bundle !"); + + using Base = BundleBase>; + using Type = Bundle<_Scalar, _T...>; + +protected: + + using Base::derived; + +public: + + template using Element = typename Base::template Element; + using Base::BundleSize; + + MANIF_MAKE_ALIGNED_OPERATOR_NEW_COND + + MANIF_COMPLETE_GROUP_TYPEDEF + MANIF_INHERIT_GROUP_API + + Bundle() = default; + ~Bundle() = default; + + MANIF_COPY_CONSTRUCTOR(Bundle); + MANIF_MOVE_CONSTRUCTOR(Bundle); + + // Copy constructor + template + Bundle(const LieGroupBase<_DerivedOther> & o); + + MANIF_GROUP_ASSIGN_OP(Bundle); + + // LieGroup common API + + /** + * @brief Get a reference to the underlying DataType. + * @param[out] a reference to the underlying Eigen vector + */ + DataType & coeffs(); + + /** + * @brief Get a const reference to the underlying DataType. + * @param[out] a const reference to the underlying Eigen vector + */ + const DataType & coeffs() const; + + + // Bundle specific API + + /** + * @brief Construct from Bundle elements + */ + Bundle(const _T<_Scalar> & ... elements); + +protected: + + // Helper for the elements constructor + template + Bundle(internal::intseq<_Idx...>, const _T<_Scalar> & ... elements); + +protected: + + //! Underlying data (Eigen) vector + DataType data_; +}; + + +template class ... _T> +template +Bundle<_Scalar, _T...>::Bundle(const LieGroupBase<_DerivedOther> & o) +: Bundle(o.coeffs()) +{} + +template class ... _T> +Bundle<_Scalar, _T...>::Bundle(const _T<_Scalar> & ... elements) +: Bundle(internal::make_intseq_t{}, elements ...) +{} + +template class ... _T> +template +Bundle<_Scalar, _T...>::Bundle( + internal::intseq<_Idx...>, const _T<_Scalar> & ... elements +) +{ + // c++11 "fold expression" + auto l = {((data_.template segment::RepSize>( + std::get<_Idx>(internal::traits::RepSizeIdx) + ) = elements.coeffs()), 0) ...}; + static_cast(l); // compiler warning +} + +template class ... _T> +typename Bundle<_Scalar, _T...>::DataType & +Bundle<_Scalar, _T...>::coeffs() +{ + return data_; +} + +template class ... _T> +const typename Bundle<_Scalar, _T...>::DataType & +Bundle<_Scalar, _T...>::coeffs() const +{ + return data_; +} + +} // namespace manif + +#endif // _MANIF_MANIF_BUNDLE_H_ diff --git a/include/manif/impl/bundle/BundleTangent.h b/include/manif/impl/bundle/BundleTangent.h new file mode 100644 index 00000000..239a5b6f --- /dev/null +++ b/include/manif/impl/bundle/BundleTangent.h @@ -0,0 +1,191 @@ +#ifndef _MANIF_MANIF_BUNDLETANGENT_H_ +#define _MANIF_MANIF_BUNDLETANGENT_H_ + +#include "manif/impl/bundle/BundleTangent_base.h" +#include "manif/impl/traits.h" + +namespace manif { + +// Forward declare for type traits specialization + +template class ... _T> struct Bundle; +template class ... _T> struct BundleTangent; + +namespace internal { + +//! Traits specialization +template class ... _T> +struct traits> +{ + // BundleTangent-specific traits + static constexpr std::size_t BundleSize = sizeof...(_T); + + using Elements = std::tuple::Tangent...>; + + template + using Element = typename std::tuple_element<_N, Elements>::type; + + template + using MapElement = Eigen::Map>; + + template + using MapConstElement = Eigen::Map>; + + static constexpr std::array DoFIdx = compute_indices<_T<_Scalar>::Tangent::DoF ...>(); + static constexpr std::array RepSizeIdx = compute_indices<_T<_Scalar>::Tangent::RepSize ...>(); + static constexpr std::array AlgIdx = compute_indices<_T<_Scalar>::Tangent::LieAlg::RowsAtCompileTime ...>(); + + // Regular traits + using Scalar = _Scalar; + + using LieGroup = Bundle<_Scalar, _T...>; + using Tangent = BundleTangent<_Scalar, _T...>; + + using Base = BundleTangentBase; + + static constexpr int Dim = accumulate(int(_T<_Scalar>::Tangent::Dim) ...); + static constexpr int DoF = accumulate(int(_T<_Scalar>::Tangent::DoF) ...); + static constexpr int RepSize = accumulate(int(_T<_Scalar>::Tangent::RepSize) ...); + + using DataType = Eigen::Matrix; + using Jacobian = Eigen::Matrix; + using LieAlg = SquareMatrix< + Scalar, + accumulate(int(_T<_Scalar>::Tangent::LieAlg::RowsAtCompileTime) ...) + >; +}; + +template class ... _T> +const constexpr std::array traits>::DoFIdx; +template class ... _T> +const constexpr std::array traits>::RepSizeIdx; +template class ... _T> +const constexpr std::array traits>::AlgIdx; +template class ... _T> +const constexpr int traits>::Dim; +template class ... _T> +const constexpr int traits>::DoF; +template class ... _T> +const constexpr int traits>::RepSize; + +} // namespace internal + +// +// BundleTangent +// + +/** + * @brief Represents a BundleTangent element. + */ +template class ... _T> +struct BundleTangent : BundleTangentBase> +{ +private: + + static_assert(sizeof...(_T) > 0, "Must have at least one element in BundleTangent !"); + + using Base = BundleTangentBase>; + using Type = BundleTangent<_Scalar, _T...>; + +protected: + + using Base::derived; + +public: + + template using Element = typename Base::template Element; + using Base::BundleSize; + + MANIF_MAKE_ALIGNED_OPERATOR_NEW_COND + + MANIF_TANGENT_TYPEDEF + MANIF_INHERIT_TANGENT_API + MANIF_INHERIT_TANGENT_OPERATOR + + BundleTangent() = default; + ~BundleTangent() = default; + + MANIF_COPY_CONSTRUCTOR(BundleTangent) + MANIF_MOVE_CONSTRUCTOR(BundleTangent) + + // Copy constructors given base + template + BundleTangent(const TangentBase<_DerivedOther> & o); + + MANIF_TANGENT_ASSIGN_OP(BundleTangent) + + // Tangent common API + + /** + * @brief Get a reference to the underlying DataType. + */ + DataType & coeffs(); + + /** + * @brief Get a const reference to the underlying DataType. + */ + const DataType & coeffs() const; + + + // BundleTangent specific API + + /** + * @brief Construct from BundleTangent elements + */ + BundleTangent(const typename _T<_Scalar>::Tangent & ... elements); + +protected: + + // Helper for the elements constructor + template + BundleTangent( + internal::intseq<_Idx...>, + const typename _T<_Scalar>::Tangent & ... elements + ); + +protected: + + DataType data_; +}; + +template class ... _T> +template +BundleTangent<_Scalar, _T...>::BundleTangent(const TangentBase<_DerivedOther> & o) +: data_(o.coeffs()) +{} + +template class ... _T> +BundleTangent<_Scalar, _T...>::BundleTangent(const typename _T<_Scalar>::Tangent & ... elements) +: BundleTangent(internal::make_intseq_t{}, elements ...) +{} + +template class ... _T> +template +BundleTangent<_Scalar, _T...>::BundleTangent( + internal::intseq<_Idx...>, + const typename _T<_Scalar>::Tangent & ... elements +) { + // c++11 "fold expression" + auto l = {((data_.template segment::RepSize>( + std::get<_Idx>(internal::traits::RepSizeIdx) + ) = elements.coeffs()), 0) ...}; + static_cast(l); // compiler warning +} + +template class ... _T> +typename BundleTangent<_Scalar, _T...>::DataType & +BundleTangent<_Scalar, _T...>::coeffs() +{ + return data_; +} + +template class ... _T> +const typename BundleTangent<_Scalar, _T...>::DataType & +BundleTangent<_Scalar, _T...>::coeffs() const +{ + return data_; +} + +} // namespace manif + +#endif // _MANIF_MANIF_BUNDLETANGENT_H_ diff --git a/include/manif/impl/bundle/BundleTangent_base.h b/include/manif/impl/bundle/BundleTangent_base.h new file mode 100644 index 00000000..ac99acb6 --- /dev/null +++ b/include/manif/impl/bundle/BundleTangent_base.h @@ -0,0 +1,414 @@ +#ifndef _MANIF_MANIF_BUNDLETANGENT_BASE_H_ +#define _MANIF_MANIF_BUNDLETANGENT_BASE_H_ + +#include "manif/impl/tangent_base.h" +#include "manif/impl/traits.h" + +namespace manif { + +/** + * @brief The base class of the Bundle tangent. + */ +template +struct BundleTangentBase : TangentBase<_Derived> +{ +private: + + using Base = TangentBase<_Derived>; + using Type = BundleTangentBase<_Derived>; + +public: + + /** + * @brief Number of elements in the BundleTangent + */ + static constexpr std::size_t BundleSize = internal::traits<_Derived>::BundleSize; + + using Elements = typename internal::traits<_Derived>::Elements; + + template + using Element = typename internal::traits<_Derived>::template Element; + + template + using MapElement = typename internal::traits<_Derived>::template MapElement; + + template + using MapConstElement = typename internal::traits<_Derived>::template MapConstElement; + + MANIF_TANGENT_TYPEDEF + MANIF_INHERIT_TANGENT_API + MANIF_INHERIT_TANGENT_OPERATOR + + using Base::data; + using Base::coeffs; + +protected: + + using Base::derived; + + MANIF_DEFAULT_CONSTRUCTOR(BundleTangentBase) + +public: + + MANIF_TANGENT_ML_ASSIGN_OP(BundleTangentBase) + + // Tangent common API + + /** + * @brief Hat operator. + * @return An element of the Lie algebra. + */ + LieAlg hat() const; + + /** + * @brief Exponential operator. + * @return An element of the Lie Group. + */ + LieGroup exp(OptJacobianRef J_m_t = {}) const; + + /** + * @brief This function is deprecated. + * Please considere using + * @ref exp instead. + */ + MANIF_DEPRECATED + LieGroup retract(OptJacobianRef J_m_t = {}) const; + + /** + * @brief Get the right Jacobian. + */ + Jacobian rjac() const; + + /** + * @brief Get the left Jacobian. + */ + Jacobian ljac() const; + + /** + * @brief Get the inverse of the right Jacobian. + */ + Jacobian rjacinv() const; + + /** + * @brief Get the inverse of the left Jacobian. + */ + Jacobian ljacinv() const; + + /** + * @brief + */ + Jacobian smallAdj() const; + + + // BundleTangent specific API + + /** + * @brief Access BundleTangent element as Map + * @tparam _Idx element index + */ + template + MapElement<_Idx> element(); + + /** + * @brief Access BundleTangent element as Map to const + * @tparam _Idx element index + */ + template + MapConstElement<_Idx> element() const; + +protected: + + template + LieAlg hat_impl(internal::intseq<_Idx...>) const; + + template + LieGroup exp_impl(OptJacobianRef J_m_t, internal::intseq<_Idx...>) const; + + template + Jacobian rjac_impl(internal::intseq<_Idx...>) const; + + template + Jacobian ljac_impl(internal::intseq<_Idx...>) const; + + template + Jacobian rjacinv_impl(internal::intseq<_Idx...>) const; + + template + Jacobian ljacinv_impl(internal::intseq<_Idx...>) const; + + template + Jacobian smallAdj_impl(internal::intseq<_Idx...>) const; +}; + + +template +typename BundleTangentBase<_Derived>::LieAlg +BundleTangentBase<_Derived>::hat() const +{ + return hat_impl(internal::make_intseq_t{}); +} + +template +template +typename BundleTangentBase<_Derived>::LieAlg +BundleTangentBase<_Derived>::hat_impl(internal::intseq<_Idx...>) const +{ + LieAlg ret = LieAlg::Zero(); + // c++11 "fold expression" + auto l = {((ret.template block< + Element<_Idx>::LieAlg::RowsAtCompileTime, + Element<_Idx>::LieAlg::RowsAtCompileTime + >( + std::get<_Idx>(internal::traits<_Derived>::AlgIdx), + std::get<_Idx>(internal::traits<_Derived>::AlgIdx) + ) = element<_Idx>().hat()), 0) ...}; + static_cast(l); // compiler warning + return ret; +} + +template +typename BundleTangentBase<_Derived>::LieGroup +BundleTangentBase<_Derived>::exp(OptJacobianRef J_m_t) const +{ + if (J_m_t) { + J_m_t->setZero(); + } + return exp_impl(J_m_t, internal::make_intseq_t{}); +} + +template +template +typename BundleTangentBase<_Derived>::LieGroup +BundleTangentBase<_Derived>::exp_impl( + OptJacobianRef J_m_t, internal::intseq<_Idx...> +) const +{ + if (J_m_t) { + return LieGroup( + element<_Idx>().exp( + J_m_t->template block< + Element<_Idx>::DoF, Element<_Idx>::DoF + >( + std::get<_Idx>(internal::traits<_Derived>::DoFIdx), + std::get<_Idx>(internal::traits<_Derived>::DoFIdx) + ) + ) ... + ); + } + return LieGroup(element<_Idx>().exp() ...); +} + +template +typename BundleTangentBase<_Derived>::LieGroup +BundleTangentBase<_Derived>::retract(OptJacobianRef J_m_t) const +{ + return exp(J_m_t); +} + +template +typename BundleTangentBase<_Derived>::Jacobian +BundleTangentBase<_Derived>::rjac() const +{ + return rjac_impl(internal::make_intseq_t{}); +} + +template +typename BundleTangentBase<_Derived>::Jacobian +BundleTangentBase<_Derived>::ljac() const +{ + return ljac_impl(internal::make_intseq_t{}); +} + +template +typename BundleTangentBase<_Derived>::Jacobian +BundleTangentBase<_Derived>::rjacinv() const +{ + return rjacinv_impl(internal::make_intseq_t{}); +} + +template +typename BundleTangentBase<_Derived>::Jacobian +BundleTangentBase<_Derived>::ljacinv() const +{ + return ljacinv_impl(internal::make_intseq_t{}); +} + +template +typename BundleTangentBase<_Derived>::Jacobian +BundleTangentBase<_Derived>::smallAdj() const +{ + return smallAdj_impl(internal::make_intseq_t{}); +} + +template +template +typename BundleTangentBase<_Derived>::Jacobian +BundleTangentBase<_Derived>::rjac_impl(internal::intseq<_Idx...>) const +{ + Jacobian Jr = Jacobian::Zero(); + // c++11 "fold expression" + auto l = {((Jr.template block::DoF, Element<_Idx>::DoF>( + std::get<_Idx>(internal::traits<_Derived>::DoFIdx), + std::get<_Idx>(internal::traits<_Derived>::DoFIdx) + ) = element<_Idx>().rjac() ), 0) ...}; + static_cast(l); // compiler warning + return Jr; +} + +template +template +typename BundleTangentBase<_Derived>::Jacobian +BundleTangentBase<_Derived>::ljac_impl(internal::intseq<_Idx...>) const +{ + Jacobian Jr = Jacobian::Zero(); + // c++11 "fold expression" + auto l = {((Jr.template block::DoF, Element<_Idx>::DoF>( + std::get<_Idx>(internal::traits<_Derived>::DoFIdx), + std::get<_Idx>(internal::traits<_Derived>::DoFIdx) + ) = element<_Idx>().ljac()), 0) ...}; + static_cast(l); // compiler warning + return Jr; +} + +template +template +typename BundleTangentBase<_Derived>::Jacobian +BundleTangentBase<_Derived>::rjacinv_impl(internal::intseq<_Idx...>) const +{ + Jacobian Jr = Jacobian::Zero(); + // c++11 "fold expression" + auto l = { + ((Jr.template block::DoF, Element<_Idx>::DoF>( + std::get<_Idx>(internal::traits<_Derived>::DoFIdx), + std::get<_Idx>(internal::traits<_Derived>::DoFIdx) + ) = element<_Idx>().rjacinv()), 0) ... + }; + static_cast(l); // compiler warning + return Jr; +} + +template +template +typename BundleTangentBase<_Derived>::Jacobian +BundleTangentBase<_Derived>::ljacinv_impl(internal::intseq<_Idx...>) const +{ + Jacobian Jr = Jacobian::Zero(); + // c++11 "fold expression" + auto l = { + ((Jr.template block::DoF, Element<_Idx>::DoF>( + std::get<_Idx>(internal::traits<_Derived>::DoFIdx), + std::get<_Idx>(internal::traits<_Derived>::DoFIdx) + ) = element<_Idx>().ljacinv()), 0) ... + }; + static_cast(l); // compiler warning + return Jr; +} + +template +template +typename BundleTangentBase<_Derived>::Jacobian +BundleTangentBase<_Derived>::smallAdj_impl(internal::intseq<_Idx...>) const +{ + Jacobian Jr = Jacobian::Zero(); + // c++11 "fold expression" + auto l = { + ((Jr.template block::DoF, Element<_Idx>::DoF>( + std::get<_Idx>(internal::traits<_Derived>::DoFIdx), + std::get<_Idx>(internal::traits<_Derived>::DoFIdx) + ) = element<_Idx>().smallAdj()), 0) ... + }; + static_cast(l); // compiler warning + return Jr; +} + +template +template +auto BundleTangentBase<_Derived>::element() -> MapElement<_Idx> +{ + return MapElement<_Idx>( + static_cast<_Derived &>(*this).coeffs().data() + + std::get<_Idx>(internal::traits<_Derived>::RepSizeIdx) + ); +} + +template +template +auto BundleTangentBase<_Derived>::element() const -> MapConstElement<_Idx> +{ + return MapConstElement<_Idx>( + static_cast(*this).coeffs().data() + + std::get<_Idx>(internal::traits<_Derived>::RepSizeIdx) + ); +} + +namespace internal { + +/** + * @brief Generator specialization for BundleTangentBase objects. + */ +template +struct GeneratorEvaluator> +{ + static typename BundleTangentBase::LieAlg + run(const unsigned int i) + { + MANIF_CHECK( + i < BundleTangentBase::DoF, + "Index i must less than DoF!", + invalid_argument + ); + + return run(i, make_intseq_t{}); + } + + template + static typename BundleTangentBase::LieAlg + run(const unsigned int i, intseq<_Idx...>) + { + using LieAlg = typename BundleTangentBase::LieAlg; + LieAlg Ei = LieAlg::Constant(0); + // c++11 "fold expression" + auto l = {((Ei.template block< + Derived::template Element<_Idx>::LieAlg::RowsAtCompileTime, + Derived::template Element<_Idx>::LieAlg::RowsAtCompileTime + >( + std::get<_Idx>(internal::traits::AlgIdx), + std::get<_Idx>(internal::traits::AlgIdx) + ) = ( + static_cast(i) >= std::get<_Idx>(internal::traits::DoFIdx) && + static_cast(i) < std::get<_Idx>(internal::traits::DoFIdx) + Derived::template Element<_Idx>::DoF + ) ? + Derived::template Element<_Idx>::Generator( + static_cast(i) - std::get<_Idx>(internal::traits::DoFIdx) + ) : + Derived::template Element<_Idx>::LieAlg::Zero() + ), 0) ...}; + static_cast(l); // compiler warning + return Ei; + } +}; + +/** + * @brief Random specialization for BundleTangent objects. + */ +template +struct RandomEvaluatorImpl> +{ + static void run(BundleTangentBase & m) + { + run(m, make_intseq_t{}); + } + + template + static void run(BundleTangentBase & m, intseq<_Idx...>) + { + m = typename BundleTangentBase::Tangent( + Derived::template Element<_Idx>::Random() ... + ); + } +}; + +} // namespace internal +} // namespace manif + +#endif // _MANIF_MANIF_BUNDLETANGENT_BASE_H_ diff --git a/include/manif/impl/bundle/BundleTangent_map.h b/include/manif/impl/bundle/BundleTangent_map.h new file mode 100644 index 00000000..0b91f46c --- /dev/null +++ b/include/manif/impl/bundle/BundleTangent_map.h @@ -0,0 +1,99 @@ +#ifndef _MANIF_MANIF_BUNDLETANGENT_MAP_H_ +#define _MANIF_MANIF_BUNDLETANGENT_MAP_H_ + +#include "manif/impl/bundle/BundleTangent.h" + +namespace manif { +namespace internal { + +/** + * @brief traits specialization for Eigen Map + */ +template class ... T> +struct traits, 0>> + : public traits> +{ + using typename traits>::Scalar; + using traits>::DoF; + using Base = BundleTangentBase, 0>>; + using DataType = Eigen::Map, 0>; +}; + +/** + * @brief traits specialization for Eigen const Map + */ +template class ... T> +struct traits, 0>> + : public traits> +{ + using typename traits>::Scalar; + using traits>::DoF; + using Base = BundleTangentBase, 0>>; + using DataType = Eigen::Map, 0>; +}; + +} // namespace internal +} // namespace manif + + +namespace Eigen { + +/** + * @brief Specialization of Map for manif::Bundle + */ +template class ... T> +class Map, 0> + : public manif::BundleTangentBase, 0>> +{ + using Base = manif::BundleTangentBase, 0>>; + +public: + + MANIF_TANGENT_TYPEDEF + MANIF_INHERIT_TANGENT_API + MANIF_INHERIT_TANGENT_OPERATOR + + using Base::BundleSize; + + Map(Scalar * coeffs) : data_(coeffs) { } + + MANIF_TANGENT_MAP_ASSIGN_OP(BundleTangent) + + DataType & coeffs() {return data_;} + + const DataType & coeffs() const {return data_;} + +protected: + + DataType data_; +}; + +/** + * @brief Specialization of Map for const manif::BundleTangent + */ +template class ... T> +class Map, 0> + : public manif::BundleTangentBase, 0>> +{ + using Base = manif::BundleTangentBase, 0>>; + +public: + + MANIF_TANGENT_TYPEDEF + MANIF_INHERIT_TANGENT_API + MANIF_INHERIT_TANGENT_OPERATOR + + using Base::BundleSize; + + Map(const Scalar * coeffs) : data_(coeffs) { } + + const DataType & coeffs() const {return data_;} + +protected: + + const DataType data_; +}; + +} // namespace Eigen + +#endif // _MANIF_MANIF_BUNDLETANGENT_MAP_H_ diff --git a/include/manif/impl/bundle/Bundle_base.h b/include/manif/impl/bundle/Bundle_base.h new file mode 100644 index 00000000..afc48bd3 --- /dev/null +++ b/include/manif/impl/bundle/Bundle_base.h @@ -0,0 +1,442 @@ +#ifndef _MANIF_MANIF_BUNDLE_BASE_H_ +#define _MANIF_MANIF_BUNDLE_BASE_H_ + +#include "manif/impl/lie_group_base.h" +#include "manif/impl/traits.h" + +namespace manif { + +/** + * @brief The base class of the Bundle group. + */ +template +struct BundleBase : LieGroupBase<_Derived> +{ +private: + + using Base = LieGroupBase<_Derived>; + using Type = BundleBase<_Derived>; + +public: + + /** + * @brief Number of elements in bundle + */ + static constexpr std::size_t BundleSize = internal::traits<_Derived>::BundleSize; + + using Elements = typename internal::traits<_Derived>::Elements; + + template + using Element = typename internal::traits<_Derived>::template Element; + + template + using MapElement = typename internal::traits<_Derived>::template MapElement; + + template + using MapConstElement = typename internal::traits<_Derived>::template MapConstElement; + + MANIF_GROUP_TYPEDEF + MANIF_INHERIT_GROUP_AUTO_API + MANIF_INHERIT_GROUP_OPERATOR + + using Base::coeffs; + + using Transformation = typename internal::traits<_Derived>::Transformation; + + // LieGroup common API + +protected: + + using Base::derived; + + MANIF_DEFAULT_CONSTRUCTOR(BundleBase) + +public: + + MANIF_GROUP_ML_ASSIGN_OP(BundleBase) + + /** + * @brief Get the inverse of this. + * @param[out] -optional- J_minv_m Jacobian of the inverse wrt this. + */ + LieGroup inverse(OptJacobianRef J_minv_m = {}) const; + + /** + * @brief Get the corresponding Lie algebra element. + * @param[out] -optional- J_t_m Jacobian of the tangent wrt to this. + * @return The tangent of this. + */ + Tangent log(OptJacobianRef J_t_m = {}) const; + + /** + * @brief This function is deprecated. + * Please consider using + * @ref log instead. + */ + MANIF_DEPRECATED + Tangent lift(OptJacobianRef J_t_m = {}) const; + + /** + * @brief Composition of this and another Bundle element. + * @param[in] m Another Bundle element. + * @param[out] -optional- J_mc_ma Jacobian of the composition wrt this. + * @param[out] -optional- J_mc_mb Jacobian of the composition wrt m. + * @return The composition of 'this . m'. + */ + template + LieGroup compose( + const LieGroupBase<_DerivedOther> & m, + OptJacobianRef J_mc_ma = {}, + OptJacobianRef J_mc_mb = {} + ) const; + + /** + * @brief Bundle group action + * @param v vector. + * @param[out] -optional- J_vout_m The Jacobian of the new object wrt this. + * @param[out] -optional- J_vout_v The Jacobian of the new object wrt input object. + * @return The translated vector. + */ + Vector act( + const Vector & v, + tl::optional>> J_vout_m = {}, + tl::optional>> J_vout_v = {} + ) const; + + /** + * @brief Get the adjoint matrix at this. + */ + Jacobian adj() const; + + + // Bundle-specific API + + /** + * @brief Get the element-diagonal transformation matrix + */ + Transformation transform() const; + + /** + * @brief Access Bundle element as Map + * @tparam _Idx element index + */ + template + MapElement<_Idx> element(); + + /** + * @brief Access Bundle element as Map to const + * @tparam _Idx element index + */ + template + MapConstElement<_Idx> element() const; + +protected: + + template + LieGroup inverse_impl(OptJacobianRef, internal::intseq<_Idx...>) const; + + template + Tangent log_impl(OptJacobianRef, internal::intseq<_Idx...>) const; + + template + LieGroup compose_impl( + const LieGroupBase<_DerivedOther> & m, + OptJacobianRef J_mc_ma, + OptJacobianRef J_mc_mb, + internal::intseq<_Idx...> + ) const; + + template + Vector act_impl( + const Vector & v, + tl::optional>> J_vout_m, + tl::optional>> J_vout_v, + internal::intseq<_Idx...> + ) const; + + template + Jacobian adj_impl(internal::intseq<_Idx...>) const; + + template + Transformation + transform_impl(internal::intseq<_Idx...>) const; +}; + + +template +typename BundleBase<_Derived>::Transformation +BundleBase<_Derived>::transform() const +{ + return transform_impl(internal::make_intseq_t{}); +} + +template +template +typename BundleBase<_Derived>::Transformation +BundleBase<_Derived>::transform_impl( + internal::intseq<_Idx...> +) const { + Transformation ret = Transformation::Zero(); + // cxx11 "fold expression" + auto l = + {((ret.template element< + Element<_Idx>::Dim+1, Element<_Idx>::Dim+1 + >( + std::get<_Idx>(internal::traits<_Derived>::TraIdx), + std::get<_Idx>(internal::traits<_Derived>::TraIdx) + ) = element<_Idx>().transform()), 0) ...}; + static_cast(l); // compiler warning + return ret; +} + +template +typename BundleBase<_Derived>::LieGroup +BundleBase<_Derived>::inverse(OptJacobianRef J_minv_m) const +{ + if (J_minv_m) { + J_minv_m->setZero(); + } + return inverse_impl(J_minv_m, internal::make_intseq_t{}); +} + +template +template +typename BundleBase<_Derived>::LieGroup +BundleBase<_Derived>::inverse_impl( + OptJacobianRef J_minv_m, internal::intseq<_Idx...> +) const { + if (J_minv_m) { + return LieGroup( + element<_Idx>().inverse( + J_minv_m->template block< + Element<_Idx>::DoF, + Element<_Idx>::DoF + >( + std::get<_Idx>(internal::traits<_Derived>::DoFIdx), + std::get<_Idx>(internal::traits<_Derived>::DoFIdx) + ) + ) ... + ); + } + return LieGroup(element<_Idx>().inverse() ...); +} + +template +typename BundleBase<_Derived>::Tangent +BundleBase<_Derived>::log(OptJacobianRef J_t_m) const +{ + if (J_t_m) { + J_t_m->setZero(); + } + return log_impl(J_t_m, internal::make_intseq_t{}); +} + +template +template +typename BundleBase<_Derived>::Tangent +BundleBase<_Derived>::log_impl( + OptJacobianRef J_minv_m, + internal::intseq<_Idx...> +) const { + if (J_minv_m) { + return Tangent( + element<_Idx>().log( + J_minv_m->template block< + Element<_Idx>::DoF, + Element<_Idx>::DoF + >( + std::get<_Idx>(internal::traits<_Derived>::DoFIdx), + std::get<_Idx>(internal::traits<_Derived>::DoFIdx) + ) + )... + ); + } + return Tangent(element<_Idx>().log() ...); +} + +template +typename BundleBase<_Derived>::Tangent +BundleBase<_Derived>::lift(OptJacobianRef J_t_m) const +{ + return log(J_t_m); +} + +template +template +typename BundleBase<_Derived>::LieGroup +BundleBase<_Derived>::compose( + const LieGroupBase<_DerivedOther> & m, + OptJacobianRef J_mc_ma, + OptJacobianRef J_mc_mb +) const { + if (J_mc_ma) { + J_mc_ma->setZero(); + } + if (J_mc_mb) { + J_mc_mb->setZero(); + } + return compose_impl(m, J_mc_ma, J_mc_mb, internal::make_intseq_t{}); +} + +template +template +typename BundleBase<_Derived>::LieGroup +BundleBase<_Derived>::compose_impl( + const LieGroupBase<_DerivedOther> & m, + OptJacobianRef J_mc_ma, + OptJacobianRef J_mc_mb, + internal::intseq<_Idx...> +) const { + return LieGroup( + element<_Idx>().compose( + static_cast(m).template element<_Idx>(), + J_mc_ma ? + J_mc_ma->template block< + Element<_Idx>::DoF, Element<_Idx>::DoF + >( + std::get<_Idx>(internal::traits<_Derived>::DoFIdx), + std::get<_Idx>(internal::traits<_Derived>::DoFIdx) + ) : + tl::optional< + Eigen::Ref::DoF, Element<_Idx>::DoF>> + >{}, + J_mc_mb ? + J_mc_mb->template block< + Element<_Idx>::DoF, Element<_Idx>::DoF + >( + std::get<_Idx>(internal::traits<_Derived>::DoFIdx), + std::get<_Idx>(internal::traits<_Derived>::DoFIdx) + ) : + tl::optional< + Eigen::Ref::DoF, Element<_Idx>::DoF>> + >{} + ) ... + ); +} + +template +typename BundleBase<_Derived>::Vector +BundleBase<_Derived>::act( + const typename BundleBase<_Derived>::Vector & v, + tl::optional>> J_vout_m, + tl::optional>> J_vout_v) const +{ + if (J_vout_m) { + J_vout_m->setZero(); + } + if (J_vout_v) { + J_vout_v->setZero(); + } + + return act_impl(v, J_vout_m, J_vout_v, internal::make_intseq_t{}); +} + +template +template +typename BundleBase<_Derived>::Vector +BundleBase<_Derived>::act_impl( + const typename BundleBase<_Derived>::Vector & v, + tl::optional>> J_vout_m, + tl::optional>> J_vout_v, + internal::intseq<_Idx...> +) const { + Vector ret; + // cxx11 "fold expression" + auto l = {((ret.template segment::Dim>( + std::get<_Idx>(internal::traits<_Derived>::DimIdx) + ) = element<_Idx>().act( + v.template segment::Dim>( + std::get<_Idx>(internal::traits<_Derived>::DimIdx) + ), + J_vout_m ? + J_vout_m->template block::Dim, Element<_Idx>::DoF>( + std::get<_Idx>(internal::traits<_Derived>::DimIdx), + std::get<_Idx>(internal::traits<_Derived>::DoFIdx) + ) : + tl::optional< + Eigen::Ref::Dim, Element<_Idx>::DoF>> + >{}, + J_vout_v ? + J_vout_v->template block::Dim, Element<_Idx>::Dim>( + std::get<_Idx>(internal::traits<_Derived>::DimIdx), + std::get<_Idx>(internal::traits<_Derived>::DimIdx) + ) : + tl::optional< + Eigen::Ref::Dim, Element<_Idx>::Dim>> + >{} + ) + ), 0) ...}; + static_cast(l); // compiler warning + return ret; +} + +template +typename BundleBase<_Derived>::Jacobian +BundleBase<_Derived>::adj() const +{ + return adj_impl(internal::make_intseq_t{}); +} + +template +template +typename BundleBase<_Derived>::Jacobian +BundleBase<_Derived>::adj_impl(internal::intseq<_Idx...>) const +{ + Jacobian adj = Jacobian::Zero(); + // cxx11 "fold expression" + auto l = {((adj.template block< + Element<_Idx>::DoF, Element<_Idx>::DoF + >( + std::get<_Idx>(internal::traits<_Derived>::DoFIdx), + std::get<_Idx>(internal::traits<_Derived>::DoFIdx) + ) = element<_Idx>().adj()), 0) ...}; + static_cast(l); // compiler warning + return adj; +} + +template +template +auto BundleBase<_Derived>::element() -> MapElement<_Idx> +{ + return MapElement<_Idx>( + static_cast<_Derived &>(*this).coeffs().data() + + std::get<_Idx>(internal::traits<_Derived>::RepSizeIdx) + ); +} + +template +template +auto BundleBase<_Derived>::element() const -> MapConstElement<_Idx> +{ + return MapConstElement<_Idx>( + static_cast(*this).coeffs().data() + + std::get<_Idx>(internal::traits<_Derived>::RepSizeIdx) + ); +} + +namespace internal { + +/** + * @brief Random specialization for Bundle objects. + */ +template +struct RandomEvaluatorImpl> +{ + static void run(BundleBase & m) + { + run(m, internal::make_intseq_t{}); + } + + template + static void run(BundleBase & m, internal::intseq<_Idx...>) + { + m = typename BundleBase::LieGroup( + BundleBase::template Element<_Idx>::Random() ... + ); + } +}; + +} // namespace internal +} // namespace manif + +#endif // _MANIF_MANIF_BUNDLE_BASE_H_ diff --git a/include/manif/impl/bundle/Bundle_map.h b/include/manif/impl/bundle/Bundle_map.h new file mode 100644 index 00000000..8a1c7a19 --- /dev/null +++ b/include/manif/impl/bundle/Bundle_map.h @@ -0,0 +1,97 @@ +#ifndef _MANIF_MANIF_BUNDLE_MAP_H_ +#define _MANIF_MANIF_BUNDLE_MAP_H_ + +#include "manif/impl/bundle/Bundle.h" + +namespace manif { +namespace internal { + +/** + * @brief traits specialization for Eigen Map + */ +template class ... T> +struct traits, 0>> + : public traits> +{ + using typename traits>::Scalar; + using traits>::RepSize; + using Base = BundleBase, 0>>; + using DataType = Eigen::Map, 0>; +}; + +/** + * @brief traits specialization for Eigen const Map + */ +template class ... T> +struct traits, 0>> + : public traits> +{ + using typename traits>::Scalar; + using traits>::RepSize; + using Base = BundleBase, 0>>; + using DataType = Eigen::Map, 0>; +}; + +} // namespace internal +} // namespace manif + + +namespace Eigen { + +/** + * @brief Specialization of Map for manif::Bundle + */ +template class ... T> +class Map, 0> + : public manif::BundleBase, 0>> +{ + using Base = manif::BundleBase, 0>>; + +public: + + MANIF_COMPLETE_GROUP_TYPEDEF + MANIF_INHERIT_GROUP_API + + using Base::BundleSize; + + Map(Scalar * coeffs) : data_(coeffs) { } + + MANIF_GROUP_MAP_ASSIGN_OP(Bundle) + + DataType & coeffs() {return data_;} + + const DataType & coeffs() const {return data_;} + +protected: + + DataType data_; +}; + +/** + * @brief Specialization of Map for const manif::Bundle + */ +template class ... T> +class Map, 0> + : public manif::BundleBase, 0>> +{ + using Base = manif::BundleBase, 0>>; + +public: + + MANIF_COMPLETE_GROUP_TYPEDEF + MANIF_INHERIT_GROUP_API + + using Base::BundleSize; + + Map(const Scalar * coeffs) : data_(coeffs) { } + + const DataType & coeffs() const {return data_;} + +protected: + + const DataType data_; +}; + +} // namespace Eigen + +#endif // _MANIF_MANIF_BUNDLE_MAP_H_ diff --git a/include/manif/impl/bundle/Bundle_properties.h b/include/manif/impl/bundle/Bundle_properties.h new file mode 100644 index 00000000..10e71490 --- /dev/null +++ b/include/manif/impl/bundle/Bundle_properties.h @@ -0,0 +1,33 @@ +#ifndef _MANIF_MANIF_BUNDLE_PROPERTIES_H_ +#define _MANIF_MANIF_BUNDLE_PROPERTIES_H_ + +#include "manif/impl/traits.h" + +namespace manif { + +// Forward declaration for type traits specialization +template struct BundleBase; +template struct BundleTangentBase; + +namespace internal { + +//! traits specialization +template +struct LieGroupProperties> +{ + static constexpr int Dim = traits<_Derived>::Dim; /// @brief Space dimension + static constexpr int DoF = traits<_Derived>::DoF; /// @brief Degrees of freedom +}; + +//! traits specialization +template +struct LieGroupProperties> +{ + static constexpr int Dim = traits<_Derived>::Dim; /// @brief Space dimension + static constexpr int DoF = traits<_Derived>::DoF; /// @brief Degrees of freedom +}; + +} // namespace internal +} // namespace manif + +#endif // _MANIF_MANIF_BUNDLE_PROPERTIES_H_ diff --git a/include/manif/impl/eigen.h b/include/manif/impl/eigen.h index 56d42150..ed97d9fb 100644 --- a/include/manif/impl/eigen.h +++ b/include/manif/impl/eigen.h @@ -108,6 +108,10 @@ assert_cols_dim(x, dim); namespace manif { + +template +using SquareMatrix = Eigen::Matrix; + namespace internal { template< class Base, class Derived > diff --git a/include/manif/impl/macro.h b/include/manif/impl/macro.h index cb4992a9..87f2913b 100644 --- a/include/manif/impl/macro.h +++ b/include/manif/impl/macro.h @@ -156,6 +156,7 @@ raise(Args&&... args) X& operator =(Eigen::MatrixBase<_EigenDerived>&& o) { coeffs() = std::move(o); return derived(); } #define MANIF_GROUP_MAP_ASSIGN_OP(X) \ + Map(const Map & o) : Base(), data_(o.coeffs()) { }\ Map& operator=(const Map& o) { coeffs() = o.coeffs(); return *this; }\ template \ Map& operator =(const manif::X##Base<_DerivedOther>& o) { coeffs() = o.coeffs(); return *this; }\ @@ -200,6 +201,7 @@ raise(Args&&... args) X& operator =(Eigen::MatrixBase<_EigenDerived>&& o) MANIF_MOVE_NOEXCEPT { coeffs() = std::move(o); return derived(); } #define MANIF_TANGENT_MAP_ASSIGN_OP(X) \ + Map(const Map & o) : Base(), data_(o.coeffs()) { }\ Map& operator=(const Map& o) { coeffs() = o.coeffs(); return *this; }\ template \ Map& operator =(const manif::X##Base<_DerivedOther>& o) { coeffs() = o.coeffs(); return *this; }\ diff --git a/include/manif/impl/rn/RnTangent.h b/include/manif/impl/rn/RnTangent.h index 094ac921..09cf5a39 100644 --- a/include/manif/impl/rn/RnTangent.h +++ b/include/manif/impl/rn/RnTangent.h @@ -93,15 +93,15 @@ template using R7Tangent = RnTangent<_Scalar, 7>; template using R8Tangent = RnTangent<_Scalar, 8>; template using R9Tangent = RnTangent<_Scalar, 9>; -MANIF_EXTRA_GROUP_TYPEDEF(R1Tangent) -MANIF_EXTRA_GROUP_TYPEDEF(R2Tangent) -MANIF_EXTRA_GROUP_TYPEDEF(R3Tangent) -MANIF_EXTRA_GROUP_TYPEDEF(R4Tangent) -MANIF_EXTRA_GROUP_TYPEDEF(R5Tangent) -MANIF_EXTRA_GROUP_TYPEDEF(R6Tangent) -MANIF_EXTRA_GROUP_TYPEDEF(R7Tangent) -MANIF_EXTRA_GROUP_TYPEDEF(R8Tangent) -MANIF_EXTRA_GROUP_TYPEDEF(R9Tangent) +MANIF_EXTRA_TANGENT_TYPEDEF(R1Tangent) +MANIF_EXTRA_TANGENT_TYPEDEF(R2Tangent) +MANIF_EXTRA_TANGENT_TYPEDEF(R3Tangent) +MANIF_EXTRA_TANGENT_TYPEDEF(R4Tangent) +MANIF_EXTRA_TANGENT_TYPEDEF(R5Tangent) +MANIF_EXTRA_TANGENT_TYPEDEF(R6Tangent) +MANIF_EXTRA_TANGENT_TYPEDEF(R7Tangent) +MANIF_EXTRA_TANGENT_TYPEDEF(R8Tangent) +MANIF_EXTRA_TANGENT_TYPEDEF(R9Tangent) template template diff --git a/include/manif/impl/rn/RnTangent_base.h b/include/manif/impl/rn/RnTangent_base.h index 325bd886..7502a48c 100644 --- a/include/manif/impl/rn/RnTangent_base.h +++ b/include/manif/impl/rn/RnTangent_base.h @@ -84,7 +84,7 @@ struct RnTangentBase : TangentBase<_Derived> Jacobian rjacinv() const; /** - * @brief Get the inverse of the right Jacobian of Rn. + * @brief Get the inverse of the left Jacobian of Rn. * @note See Eq. (191). * @see ljac. */ diff --git a/include/manif/impl/rn/Rn_base.h b/include/manif/impl/rn/Rn_base.h index 7843d1c0..ab9be11e 100644 --- a/include/manif/impl/rn/Rn_base.h +++ b/include/manif/impl/rn/Rn_base.h @@ -156,8 +156,6 @@ RnBase<_Derived>::compose( OptJacobianRef J_mc_ma, OptJacobianRef J_mc_mb) const { - using std::abs; - static_assert( std::is_base_of, _DerivedOther>::value, "Argument does not inherit from RnBase !"); diff --git a/include/manif/impl/se_2_3/SE_2_3_base.h b/include/manif/impl/se_2_3/SE_2_3_base.h index 00a23701..01920c50 100644 --- a/include/manif/impl/se_2_3/SE_2_3_base.h +++ b/include/manif/impl/se_2_3/SE_2_3_base.h @@ -47,6 +47,7 @@ struct SE_2_3Base : LieGroupBase<_Derived> using Rotation = typename internal::traits<_Derived>::Rotation; using Translation = typename internal::traits<_Derived>::Translation; using LinearVelocity = typename internal::traits<_Derived>::LinearVelocity; + using Transformation = Eigen::Matrix; using Isometry = Eigen::Matrix; /**< Double direct spatial isometry*/ using QuaternionDataType = Eigen::Quaternion; @@ -118,6 +119,14 @@ struct SE_2_3Base : LieGroupBase<_Derived> // SE_2_3 specific functions + /** + * Get the isometry object (double direct isometry). + * @note T = | R t v| + * | 1 | + * | 1| + */ + Transformation transform() const; + /** * Get the isometry object (double direct isometry). * @note T = | R t v| @@ -195,8 +204,8 @@ struct SE_2_3Base : LieGroupBase<_Derived> }; template -typename SE_2_3Base<_Derived>::Isometry -SE_2_3Base<_Derived>::isometry() const +typename SE_2_3Base<_Derived>::Transformation +SE_2_3Base<_Derived>::transform() const { Eigen::Matrix T; T.template topLeftCorner<3,3>() = rotation(); @@ -207,6 +216,13 @@ SE_2_3Base<_Derived>::isometry() const return T; } +template +typename SE_2_3Base<_Derived>::Isometry +SE_2_3Base<_Derived>::isometry() const +{ + return Isometry(transform()); +} + template typename SE_2_3Base<_Derived>::Rotation SE_2_3Base<_Derived>::rotation() const diff --git a/include/manif/impl/traits.h b/include/manif/impl/traits.h index 93124a82..45c88171 100644 --- a/include/manif/impl/traits.h +++ b/include/manif/impl/traits.h @@ -1,6 +1,7 @@ #ifndef _MANIF_MANIF_TRAITS_H_ #define _MANIF_MANIF_TRAITS_H_ +#include #include namespace manif { @@ -39,13 +40,17 @@ struct traitscast; /** * @brief Traits to change the scalar type of a template class - * @note given using FooDouble = Foo + * @note given using FooDouble = Foo * using FooFloat = typename traitscast::cast; */ -template