From 19d1ac42b95664530b276744f2ad19da34207793 Mon Sep 17 00:00:00 2001 From: JIanzhu Huai Date: Thu, 3 Sep 2020 15:21:15 +0800 Subject: [PATCH 01/26] correct coefficients of approximated SE3 Q_r --- gtsam/geometry/Pose3.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/gtsam/geometry/Pose3.cpp b/gtsam/geometry/Pose3.cpp index 0a56e2ef5..ecc0598eb 100644 --- a/gtsam/geometry/Pose3.cpp +++ b/gtsam/geometry/Pose3.cpp @@ -230,8 +230,8 @@ static Matrix3 computeQforExpmapDerivative(const Vector6& xi) { } else { Q = -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); + - 1./24.*(W*W*V + V*W*W - 3*W*V*W) + + 1./120.*(W*V*W*W + W*W*V*W); } #endif From 7d91540865bb9a7db74d5800bf81c3204105b178 Mon Sep 17 00:00:00 2001 From: JIanzhu Huai Date: Fri, 4 Sep 2020 17:19:58 +0800 Subject: [PATCH 02/26] test computeQforExpmapDerivative --- gtsam/geometry/Pose3.cpp | 4 ++-- gtsam/geometry/Pose3.h | 3 +++ gtsam/geometry/tests/testPose3.cpp | 9 +++++++++ 3 files changed, 14 insertions(+), 2 deletions(-) diff --git a/gtsam/geometry/Pose3.cpp b/gtsam/geometry/Pose3.cpp index 0a56e2ef5..e505e8dff 100644 --- a/gtsam/geometry/Pose3.cpp +++ b/gtsam/geometry/Pose3.cpp @@ -198,7 +198,7 @@ Vector6 Pose3::ChartAtOrigin::Local(const Pose3& pose, ChartJacobian Hpose) { * (see Chirikjian11book2, pg 44, eq 10.95. * The closed-form formula is similar to formula 102 in Barfoot14tro) */ -static Matrix3 computeQforExpmapDerivative(const Vector6& xi) { +Matrix3 Pose3::computeQforExpmapDerivative(const Vector6& xi, double nearZeroThreshold) { const auto w = xi.head<3>(); const auto v = xi.tail<3>(); const Matrix3 V = skewSymmetric(v); @@ -220,7 +220,7 @@ static Matrix3 computeQforExpmapDerivative(const Vector6& xi) { #else // The closed-form formula in Barfoot14tro eq. (102) double phi = w.norm(); - if (std::abs(phi)>1e-5) { + if (std::abs(phi)>nearZeroThreshold) { 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 diff --git a/gtsam/geometry/Pose3.h b/gtsam/geometry/Pose3.h index 159fd2927..575fbaa5c 100644 --- a/gtsam/geometry/Pose3.h +++ b/gtsam/geometry/Pose3.h @@ -181,6 +181,9 @@ public: static Vector6 Local(const Pose3& pose, ChartJacobian Hpose = boost::none); }; + static Matrix3 computeQforExpmapDerivative( + const Vector6& xi, double nearZeroThreshold = 1e-5); + using LieGroup::inverse; // version with derivative /** diff --git a/gtsam/geometry/tests/testPose3.cpp b/gtsam/geometry/tests/testPose3.cpp index 8586f35a0..55720abca 100644 --- a/gtsam/geometry/tests/testPose3.cpp +++ b/gtsam/geometry/tests/testPose3.cpp @@ -818,6 +818,15 @@ TEST( Pose3, LogmapDerivative) { EXPECT(assert_equal(expectedH, actualH)); } +TEST( Pose3, computeQforExpmapDerivative) { + Vector6 w = Vector6::Random(); + w.head<3>().normalize(); + w.head<3>() = w.head<3>() * 0.09; + Matrix3 Qexact = Pose3::computeQforExpmapDerivative(w); + Matrix3 Qapprox = Pose3::computeQforExpmapDerivative(w, 0.1); + EXPECT(assert_equal(Qapprox, Qexact, 1e-5)); +} + /* ************************************************************************* */ Vector6 testDerivAdjoint(const Vector6& xi, const Vector6& v) { return Pose3::adjointMap(xi) * v; From 859c157a0fdd475304fb2549ebe81f73341735d3 Mon Sep 17 00:00:00 2001 From: Varun Agrawal Date: Thu, 17 Sep 2020 21:26:11 -0400 Subject: [PATCH 03/26] Use cmake to identify best Python version if Default requested --- CMakeLists.txt | 17 +++++++++++++++++ cmake/FindNumPy.cmake | 12 ++---------- cmake/GtsamMatlabWrap.cmake | 22 +++++++++------------- 3 files changed, 28 insertions(+), 23 deletions(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index aa6082cb3..82adbc99f 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -14,6 +14,11 @@ set (GTSAM_VERSION_PATCH 0) math (EXPR GTSAM_VERSION_NUMERIC "10000 * ${GTSAM_VERSION_MAJOR} + 100 * ${GTSAM_VERSION_MINOR} + ${GTSAM_VERSION_PATCH}") set (GTSAM_VERSION_STRING "${GTSAM_VERSION_MAJOR}.${GTSAM_VERSION_MINOR}.${GTSAM_VERSION_PATCH}") +set (CMAKE_PROJECT_VERSION ${GTSAM_VERSION_STRING}) +set (CMAKE_PROJECT_VERSION_MAJOR ${GTSAM_VERSION_MAJOR}) +set (CMAKE_PROJECT_VERSION_MINOR ${GTSAM_VERSION_MINOR}) +set (CMAKE_PROJECT_VERSION_PATCH ${GTSAM_VERSION_PATCH}) + ############################################################################### # Gather information, perform checks, set defaults @@ -113,6 +118,18 @@ if(GTSAM_INSTALL_MATLAB_TOOLBOX AND NOT BUILD_SHARED_LIBS) endif() if(GTSAM_BUILD_PYTHON) + # Get info about the Python3 interpreter + # https://cmake.org/cmake/help/latest/module/FindPython3.html#module:FindPython3 + find_package(Python3 COMPONENTS Interpreter Development) + + if(NOT ${Python3_FOUND}) + message(FATAL_ERROR "Cannot find Python3 interpreter. Please install Python >= 3.6.") + endif() + + if(${GTSAM_PYTHON_VERSION} STREQUAL "Default") + set(GTSAM_PYTHON_VERSION "${Python3_VERSION_MAJOR}.${Python3_VERSION_MINOR}" CACHE STRING "The version of Python to build the wrappers against." FORCE) + endif() + if(GTSAM_UNSTABLE_BUILD_PYTHON) if (NOT GTSAM_BUILD_UNSTABLE) message(WARNING "GTSAM_UNSTABLE_BUILD_PYTHON requires the unstable module to be enabled.") diff --git a/cmake/FindNumPy.cmake b/cmake/FindNumPy.cmake index 4f5743aa6..d55a760c6 100644 --- a/cmake/FindNumPy.cmake +++ b/cmake/FindNumPy.cmake @@ -40,17 +40,9 @@ # Finding NumPy involves calling the Python interpreter if(NumPy_FIND_REQUIRED) - if(GTSAM_PYTHON_VERSION STREQUAL "Default") - find_package(PythonInterp REQUIRED) - else() - find_package(PythonInterp ${GTSAM_PYTHON_VERSION} EXACT REQUIRED) - endif() + find_package(PythonInterp ${GTSAM_PYTHON_VERSION} EXACT REQUIRED) else() - if(GTSAM_PYTHON_VERSION STREQUAL "Default") - find_package(PythonInterp) - else() - find_package(PythonInterp ${GTSAM_PYTHON_VERSION} EXACT) - endif() + find_package(PythonInterp ${GTSAM_PYTHON_VERSION} EXACT) endif() if(NOT PYTHONINTERP_FOUND) diff --git a/cmake/GtsamMatlabWrap.cmake b/cmake/GtsamMatlabWrap.cmake index 111114a7b..4c44d2cb3 100644 --- a/cmake/GtsamMatlabWrap.cmake +++ b/cmake/GtsamMatlabWrap.cmake @@ -215,19 +215,15 @@ function(wrap_library_internal interfaceHeader linkLibraries extraIncludeDirs ex # Set up generation of module source file file(MAKE_DIRECTORY "${generated_files_path}") - if(GTSAM_PYTHON_VERSION STREQUAL "Default") - find_package(PythonInterp REQUIRED) - find_package(PythonLibs REQUIRED) - else() - find_package(PythonInterp - ${GTSAM_PYTHON_VERSION} - EXACT - REQUIRED) - find_package(PythonLibs - ${GTSAM_PYTHON_VERSION} - EXACT - REQUIRED) - endif() + find_package(PythonInterp + ${GTSAM_PYTHON_VERSION} + EXACT + REQUIRED) + find_package(PythonLibs + ${GTSAM_PYTHON_VERSION} + EXACT + REQUIRED) + set(_ignore gtsam::Point2 gtsam::Point3) From 75e24ef86755aed0bec7aaffd68747db6847ee9d Mon Sep 17 00:00:00 2001 From: Varun Agrawal Date: Thu, 17 Sep 2020 21:42:53 -0400 Subject: [PATCH 04/26] print version of TBB found --- CMakeLists.txt | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index aa6082cb3..069b8e37c 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -529,9 +529,9 @@ print_build_options_for_target(gtsam) print_config("Use System Eigen" "${GTSAM_USE_SYSTEM_EIGEN} (Using version: ${GTSAM_EIGEN_VERSION})") if(GTSAM_USE_TBB) - print_config("Use Intel TBB" "Yes") + print_config("Use Intel TBB" "Yes (Version: ${TBB_VERSION})") elseif(TBB_FOUND) - print_config("Use Intel TBB" "TBB found but GTSAM_WITH_TBB is disabled") + print_config("Use Intel TBB" "TBB (Version: ${TBB_VERSION}) found but GTSAM_WITH_TBB is disabled") else() print_config("Use Intel TBB" "TBB not found") endif() From f64ced8791e442fb91b999d642b1095c33de0599 Mon Sep 17 00:00:00 2001 From: Varun Agrawal Date: Thu, 17 Sep 2020 21:43:31 -0400 Subject: [PATCH 05/26] Don't print private compile options --- cmake/GtsamPrinting.cmake | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/cmake/GtsamPrinting.cmake b/cmake/GtsamPrinting.cmake index 5757f5ee1..c68679667 100644 --- a/cmake/GtsamPrinting.cmake +++ b/cmake/GtsamPrinting.cmake @@ -46,16 +46,16 @@ endfunction() # Prints all the relevant CMake build options for a given target: function(print_build_options_for_target target_name_) print_padded(GTSAM_COMPILE_FEATURES_PUBLIC) - print_padded(GTSAM_COMPILE_OPTIONS_PRIVATE) + # print_padded(GTSAM_COMPILE_OPTIONS_PRIVATE) print_padded(GTSAM_COMPILE_OPTIONS_PUBLIC) - print_padded(GTSAM_COMPILE_DEFINITIONS_PRIVATE) + # print_padded(GTSAM_COMPILE_DEFINITIONS_PRIVATE) print_padded(GTSAM_COMPILE_DEFINITIONS_PUBLIC) foreach(build_type ${GTSAM_CMAKE_CONFIGURATION_TYPES}) string(TOUPPER "${build_type}" build_type_toupper) - print_padded(GTSAM_COMPILE_OPTIONS_PRIVATE_${build_type_toupper}) + # print_padded(GTSAM_COMPILE_OPTIONS_PRIVATE_${build_type_toupper}) print_padded(GTSAM_COMPILE_OPTIONS_PUBLIC_${build_type_toupper}) - print_padded(GTSAM_COMPILE_DEFINITIONS_PRIVATE_${build_type_toupper}) + # print_padded(GTSAM_COMPILE_DEFINITIONS_PRIVATE_${build_type_toupper}) print_padded(GTSAM_COMPILE_DEFINITIONS_PUBLIC_${build_type_toupper}) endforeach() endfunction() From 25eb4a13fd9dac20efe8decca3855f2c90e77be0 Mon Sep 17 00:00:00 2001 From: Varun Agrawal Date: Fri, 18 Sep 2020 11:49:03 -0400 Subject: [PATCH 06/26] minor formatting --- CMakeLists.txt | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index 82adbc99f..a45c17677 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -127,7 +127,11 @@ if(GTSAM_BUILD_PYTHON) endif() if(${GTSAM_PYTHON_VERSION} STREQUAL "Default") - set(GTSAM_PYTHON_VERSION "${Python3_VERSION_MAJOR}.${Python3_VERSION_MINOR}" CACHE STRING "The version of Python to build the wrappers against." FORCE) + set(GTSAM_PYTHON_VERSION "${Python3_VERSION_MAJOR}.${Python3_VERSION_MINOR}" + CACHE + STRING + "The version of Python to build the wrappers against." + FORCE) endif() if(GTSAM_UNSTABLE_BUILD_PYTHON) From 9670b6661db44f3c731b491d8cc2c4c6952239b3 Mon Sep 17 00:00:00 2001 From: Ayush Baid Date: Sat, 19 Sep 2020 16:47:13 -0400 Subject: [PATCH 07/26] Adding logmap API which applied a logarithmic map taking the object to the argument --- gtsam/gtsam.i | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/gtsam/gtsam.i b/gtsam/gtsam.i index 52f5901ee..f53c00043 100644 --- a/gtsam/gtsam.i +++ b/gtsam/gtsam.i @@ -361,6 +361,7 @@ class LieScalar { // Lie group static gtsam::LieScalar Expmap(Vector v); static Vector Logmap(const gtsam::LieScalar& p); + Vector logmap(const gtsam::LieScalar& p); }; #include @@ -390,6 +391,7 @@ class LieVector { // Lie group static gtsam::LieVector Expmap(Vector v); static Vector Logmap(const gtsam::LieVector& p); + Vector logmap(const gtsam::LieVector& p); // enabling serialization functionality void serialize() const; @@ -422,6 +424,7 @@ class LieMatrix { // Lie group static gtsam::LieMatrix Expmap(Vector v); static Vector Logmap(const gtsam::LieMatrix& p); + Vector logmap(const gtsam::LieMatrix& p); // enabling serialization functionality void serialize() const; @@ -506,6 +509,7 @@ class StereoPoint2 { // Lie Group static gtsam::StereoPoint2 Expmap(Vector v); static Vector Logmap(const gtsam::StereoPoint2& p); + Vector logmap(const gtsam::StereoPoint2& p); // Standard Interface Vector vector() const; @@ -567,6 +571,7 @@ class Rot2 { // Lie Group static gtsam::Rot2 Expmap(Vector v); static Vector Logmap(const gtsam::Rot2& p); + Vector logmap(const gtsam::Rot2& p); // Group Action on Point2 gtsam::Point2 rotate(const gtsam::Point2& point) const; @@ -727,6 +732,7 @@ class Rot3 { // Standard Interface static gtsam::Rot3 Expmap(Vector v); static Vector Logmap(const gtsam::Rot3& p); + Vector logmap(const gtsam::Rot3& p); Matrix matrix() const; Matrix transpose() const; gtsam::Point3 column(size_t index) const; @@ -772,6 +778,7 @@ class Pose2 { // Lie Group static gtsam::Pose2 Expmap(Vector v); static Vector Logmap(const gtsam::Pose2& p); + Vector logmap(const gtsam::Pose2& p); static Matrix ExpmapDerivative(Vector v); static Matrix LogmapDerivative(const gtsam::Pose2& v); Matrix AdjointMap() const; @@ -825,6 +832,7 @@ class Pose3 { // Lie Group static gtsam::Pose3 Expmap(Vector v); static Vector Logmap(const gtsam::Pose3& pose); + Vector logmap(const gtsam::Pose3& pose); Matrix AdjointMap() const; Vector Adjoint(Vector xi) const; static Matrix adjointMap_(Vector xi); @@ -3143,6 +3151,7 @@ class ConstantBias { // Lie Group static gtsam::imuBias::ConstantBias Expmap(Vector v); static Vector Logmap(const gtsam::imuBias::ConstantBias& b); + Vector logmap(const gtsam::imuBias::ConstantBias& b); // Standard Interface Vector vector() const; From 1d0b9d510e0e84f67082400ee68e44f35dfc0077 Mon Sep 17 00:00:00 2001 From: Ayush Baid Date: Sat, 19 Sep 2020 17:57:39 -0400 Subject: [PATCH 08/26] Removing logmap in case of failure --- gtsam/gtsam.i | 3 --- 1 file changed, 3 deletions(-) diff --git a/gtsam/gtsam.i b/gtsam/gtsam.i index f53c00043..d8297c43b 100644 --- a/gtsam/gtsam.i +++ b/gtsam/gtsam.i @@ -361,7 +361,6 @@ class LieScalar { // Lie group static gtsam::LieScalar Expmap(Vector v); static Vector Logmap(const gtsam::LieScalar& p); - Vector logmap(const gtsam::LieScalar& p); }; #include @@ -509,7 +508,6 @@ class StereoPoint2 { // Lie Group static gtsam::StereoPoint2 Expmap(Vector v); static Vector Logmap(const gtsam::StereoPoint2& p); - Vector logmap(const gtsam::StereoPoint2& p); // Standard Interface Vector vector() const; @@ -3151,7 +3149,6 @@ class ConstantBias { // Lie Group static gtsam::imuBias::ConstantBias Expmap(Vector v); static Vector Logmap(const gtsam::imuBias::ConstantBias& b); - Vector logmap(const gtsam::imuBias::ConstantBias& b); // Standard Interface Vector vector() const; From 457e52d05db4841cf713fcd0b736a7824591b279 Mon Sep 17 00:00:00 2001 From: John Lambert Date: Mon, 21 Sep 2020 09:04:17 -0400 Subject: [PATCH 09/26] Update README.md --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index cc8af7248..60eff197a 100644 --- a/README.md +++ b/README.md @@ -62,7 +62,7 @@ GTSAM 4.1 added a new pybind wrapper, and **removed** the deprecated functionali ## Wrappers -We provide support for [MATLAB](matlab/README.md) and [Python](cython/README.md) wrappers for GTSAM. Please refer to the linked documents for more details. +We provide support for [MATLAB](matlab/README.md) and [Python](python/README.md) wrappers for GTSAM. Please refer to the linked documents for more details. ## The Preintegrated IMU Factor From bcaed27c54006010e76488f838ac13c541c82dcf Mon Sep 17 00:00:00 2001 From: John Lambert Date: Mon, 21 Sep 2020 09:38:35 -0400 Subject: [PATCH 10/26] improve python wrapper cmake flag instructions --- python/README.md | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/python/README.md b/python/README.md index b1a3a865f..bb883222c 100644 --- a/python/README.md +++ b/python/README.md @@ -7,7 +7,10 @@ This is the Python wrapper around the GTSAM C++ library. We use our custom [wrap ## Requirements - If you want to build the GTSAM python library for a specific python version (eg 3.6), - use the `-DGTSAM_PYTHON_VERSION=3.6` option when running `cmake` otherwise the default interpreter will be used. + use the `-DGTSAM_PYTHON_VERSION=3.6` option when running `cmake` otherwise the default interpreter will be used. For example: + ```bash + cmake .. -DGTSAM_BUILD_PYTHON=1 -DGTSAM_PYTHON_VERSION=3.6.10 + ``` - If the interpreter is inside an environment (such as an anaconda environment or virtualenv environment), then the environment should be active while building GTSAM. - This wrapper needs `pyparsing(>=2.4.2)`, and `numpy(>=1.11.0)`. These can be installed as follows: From bb22773a810c2d1f341ca2b2f41b7219f3a29114 Mon Sep 17 00:00:00 2001 From: John Lambert Date: Mon, 21 Sep 2020 09:39:46 -0400 Subject: [PATCH 11/26] Update README.md --- python/README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/python/README.md b/python/README.md index bb883222c..be69034f9 100644 --- a/python/README.md +++ b/python/README.md @@ -7,7 +7,7 @@ This is the Python wrapper around the GTSAM C++ library. We use our custom [wrap ## Requirements - If you want to build the GTSAM python library for a specific python version (eg 3.6), - use the `-DGTSAM_PYTHON_VERSION=3.6` option when running `cmake` otherwise the default interpreter will be used. For example: + use the `-DGTSAM_PYTHON_VERSION=3.6` option when running `cmake` otherwise the default interpreter will be used. For example, if you local Python version is 3.6.10, then run: ```bash cmake .. -DGTSAM_BUILD_PYTHON=1 -DGTSAM_PYTHON_VERSION=3.6.10 ``` From 133b81867c1bf4fcd9d619ffa7bddba024e8a28e Mon Sep 17 00:00:00 2001 From: John Lambert Date: Mon, 21 Sep 2020 09:40:42 -0400 Subject: [PATCH 12/26] Update README.md --- python/README.md | 11 +++++------ 1 file changed, 5 insertions(+), 6 deletions(-) diff --git a/python/README.md b/python/README.md index be69034f9..be5d316b1 100644 --- a/python/README.md +++ b/python/README.md @@ -7,10 +7,7 @@ This is the Python wrapper around the GTSAM C++ library. We use our custom [wrap ## Requirements - If you want to build the GTSAM python library for a specific python version (eg 3.6), - use the `-DGTSAM_PYTHON_VERSION=3.6` option when running `cmake` otherwise the default interpreter will be used. For example, if you local Python version is 3.6.10, then run: - ```bash - cmake .. -DGTSAM_BUILD_PYTHON=1 -DGTSAM_PYTHON_VERSION=3.6.10 - ``` + use the `-DGTSAM_PYTHON_VERSION=3.6` option when running `cmake` otherwise the default interpreter will be used. - If the interpreter is inside an environment (such as an anaconda environment or virtualenv environment), then the environment should be active while building GTSAM. - This wrapper needs `pyparsing(>=2.4.2)`, and `numpy(>=1.11.0)`. These can be installed as follows: @@ -21,8 +18,10 @@ This is the Python wrapper around the GTSAM C++ library. We use our custom [wrap ## Install -- Run cmake with the `GTSAM_BUILD_PYTHON` cmake flag enabled to configure building the wrapper. The wrapped module will be built and copied to the directory `/python`. - +- Run cmake with the `GTSAM_BUILD_PYTHON` cmake flag enabled to configure building the wrapper. The wrapped module will be built and copied to the directory `/python`. For example, if you local Python version is 3.6.10, then run: + ```bash + cmake .. -DGTSAM_BUILD_PYTHON=1 -DGTSAM_PYTHON_VERSION=3.6.10 + ``` - Build GTSAM and the wrapper with `make` (or `ninja` if you use `-GNinja`). - To install, simply run `make python-install` (`ninja python-install`). From fe0312fd63b404dd3f304338ee2e70cf1183a28d Mon Sep 17 00:00:00 2001 From: John Lambert Date: Mon, 21 Sep 2020 09:58:03 -0400 Subject: [PATCH 13/26] Update README.md --- python/README.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/python/README.md b/python/README.md index be5d316b1..6e2a30d2e 100644 --- a/python/README.md +++ b/python/README.md @@ -7,7 +7,7 @@ This is the Python wrapper around the GTSAM C++ library. We use our custom [wrap ## Requirements - If you want to build the GTSAM python library for a specific python version (eg 3.6), - use the `-DGTSAM_PYTHON_VERSION=3.6` option when running `cmake` otherwise the default interpreter will be used. + use the `-DGTSAM_PYTHON_VERSION=3.6` option when running `cmake` otherwise the default interpreter will be used. - If the interpreter is inside an environment (such as an anaconda environment or virtualenv environment), then the environment should be active while building GTSAM. - This wrapper needs `pyparsing(>=2.4.2)`, and `numpy(>=1.11.0)`. These can be installed as follows: @@ -18,7 +18,7 @@ This is the Python wrapper around the GTSAM C++ library. We use our custom [wrap ## Install -- Run cmake with the `GTSAM_BUILD_PYTHON` cmake flag enabled to configure building the wrapper. The wrapped module will be built and copied to the directory `/python`. For example, if you local Python version is 3.6.10, then run: +- Run cmake with the `GTSAM_BUILD_PYTHON` cmake flag enabled to configure building the wrapper. The wrapped module will be built and copied to the directory `/python`. For example, if your local Python version is 3.6.10, then you should run: ```bash cmake .. -DGTSAM_BUILD_PYTHON=1 -DGTSAM_PYTHON_VERSION=3.6.10 ``` From 169d42f313bd81a1dedfffd05fa24fe95c34cb90 Mon Sep 17 00:00:00 2001 From: John Lambert Date: Mon, 21 Sep 2020 12:08:18 -0400 Subject: [PATCH 14/26] use cleaner checkmark symbol in readme --- python/gtsam/examples/README.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/python/gtsam/examples/README.md b/python/gtsam/examples/README.md index e998e4dcd..d70867dda 100644 --- a/python/gtsam/examples/README.md +++ b/python/gtsam/examples/README.md @@ -7,7 +7,7 @@ | DiscreteBayesNet_FG | none of the required discrete functionality is exposed through Python | | easyPoint2KalmanFilter | ExtendedKalmanFilter not yet exposed through Python | | elaboratePoint2KalmanFilter | GaussianSequentialSolver not yet exposed through Python | -| ImuFactorExample2 | X | +| ImuFactorExample2 | :heavy_check_mark: | | ImuFactorsExample | | | ISAM2Example_SmartFactor | | | ISAM2_SmartFactorStereo_IMU | | @@ -49,4 +49,4 @@ Extra Examples (with no C++ equivalent) - PlanarManipulatorExample -- SFMData \ No newline at end of file +- SFMData From 7478c19aad268ff8b5e677dad2f405b20b9ea285 Mon Sep 17 00:00:00 2001 From: John Lambert Date: Mon, 21 Sep 2020 12:16:20 -0400 Subject: [PATCH 15/26] update example list --- python/gtsam/examples/README.md | 31 ++++++++++++++++++++----------- 1 file changed, 20 insertions(+), 11 deletions(-) diff --git a/python/gtsam/examples/README.md b/python/gtsam/examples/README.md index d70867dda..3a325aa30 100644 --- a/python/gtsam/examples/README.md +++ b/python/gtsam/examples/README.md @@ -3,49 +3,58 @@ | C++ Example Name | Ported | |-------------------------------------------------------|--------| | CameraResectioning | | +| CombinedImuFactorsExample | | | CreateSFMExampleData | | +| DiscreteBayesNetExample | | | DiscreteBayesNet_FG | none of the required discrete functionality is exposed through Python | | easyPoint2KalmanFilter | ExtendedKalmanFilter not yet exposed through Python | | elaboratePoint2KalmanFilter | GaussianSequentialSolver not yet exposed through Python | -| ImuFactorExample2 | :heavy_check_mark: | +| FisheyeExample | | +| HMMExample | | +| ImuFactorsExample2 | :heavy_check_mark: | | ImuFactorsExample | | +| IMUKittiExampleGPS.cpp | | +| InverseKinematicsExampleExpressions.cpp | | | ISAM2Example_SmartFactor | | | ISAM2_SmartFactorStereo_IMU | | | LocalizationExample | impossible? | | METISOrderingExample | | -| OdometryExample | X | -| PlanarSLAMExample | X | -| Pose2SLAMExample | X | +| OdometryExample | :heavy_check_mark: | +| PlanarSLAMExample | :heavy_check_mark: | +| Pose2SLAMExample | :heavy_check_mark: | | Pose2SLAMExampleExpressions | | -| Pose2SLAMExample_g2o | X | +| Pose2SLAMExample_g2o | :heavy_check_mark: | | Pose2SLAMExample_graph | | | Pose2SLAMExample_graphviz | | | Pose2SLAMExample_lago | lago not yet exposed through Python | | Pose2SLAMStressTest | | | Pose2SLAMwSPCG | | +| Pose3Localization.cpp | | | Pose3SLAMExample_changeKeys | | | Pose3SLAMExampleExpressions_BearingRangeWithTransform | | -| Pose3SLAMExample_g2o | X | +| Pose3SLAMExample_g2o | :heavy_check_mark: | | Pose3SLAMExample_initializePose3Chordal | | | Pose3SLAMExample_initializePose3Gradient | | | RangeISAMExample_plaza2 | | | SelfCalibrationExample | | +| SFMdata | | | SFMExample_bal_COLAMD_METIS | | -| SFMExample_bal | | -| SFMExample | X | +| SFMExample_bal | :heavy_check_mark: | +| SFMExample | :heavy_check_mark: | | SFMExampleExpressions_bal | | | SFMExampleExpressions | | | SFMExample_SmartFactor | | | SFMExample_SmartFactorPCG | | -| SimpleRotation | X | +| ShonanAveragingCLI.cpp | | +| SimpleRotation | :heavy_check_mark: | | SolverComparer | | | StereoVOExample | | | StereoVOExample_large | | | TimeTBB | | | UGM_chain | discrete functionality not yet exposed | | UGM_small | discrete functionality not yet exposed | -| VisualISAM2Example | X | -| VisualISAMExample | X | +| VisualISAM2Example | :heavy_check_mark: | +| VisualISAMExample | :heavy_check_mark: | Extra Examples (with no C++ equivalent) - PlanarManipulatorExample From 0e7719ae9893ef04f13adbf8fa797c11d167a293 Mon Sep 17 00:00:00 2001 From: John Lambert Date: Mon, 21 Sep 2020 12:19:33 -0400 Subject: [PATCH 16/26] update python examples list --- python/gtsam/examples/README.md | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/python/gtsam/examples/README.md b/python/gtsam/examples/README.md index 3a325aa30..84a41990f 100644 --- a/python/gtsam/examples/README.md +++ b/python/gtsam/examples/README.md @@ -13,7 +13,7 @@ | HMMExample | | | ImuFactorsExample2 | :heavy_check_mark: | | ImuFactorsExample | | -| IMUKittiExampleGPS.cpp | | +| IMUKittiExampleGPS | | | InverseKinematicsExampleExpressions.cpp | | | ISAM2Example_SmartFactor | | | ISAM2_SmartFactorStereo_IMU | | @@ -29,11 +29,11 @@ | Pose2SLAMExample_lago | lago not yet exposed through Python | | Pose2SLAMStressTest | | | Pose2SLAMwSPCG | | -| Pose3Localization.cpp | | +| Pose3Localization | | | Pose3SLAMExample_changeKeys | | | Pose3SLAMExampleExpressions_BearingRangeWithTransform | | | Pose3SLAMExample_g2o | :heavy_check_mark: | -| Pose3SLAMExample_initializePose3Chordal | | +| Pose3SLAMExample_initializePose3Chordal | :heavy_check_mark: | | Pose3SLAMExample_initializePose3Gradient | | | RangeISAMExample_plaza2 | | | SelfCalibrationExample | | @@ -45,7 +45,7 @@ | SFMExampleExpressions | | | SFMExample_SmartFactor | | | SFMExample_SmartFactorPCG | | -| ShonanAveragingCLI.cpp | | +| ShonanAveragingCLI | :heavy_check_mark: | | SimpleRotation | :heavy_check_mark: | | SolverComparer | | | StereoVOExample | | @@ -57,5 +57,8 @@ | VisualISAMExample | :heavy_check_mark: | Extra Examples (with no C++ equivalent) +- DogLegOptimizerExample +- GPSFactorExample - PlanarManipulatorExample +- PreintegrationExample - SFMData From 2af56303fc4f05c9ea841398974715ee3c8fe3f8 Mon Sep 17 00:00:00 2001 From: John Lambert Date: Mon, 21 Sep 2020 12:28:16 -0400 Subject: [PATCH 17/26] add p --- python/gtsam/examples/SFMExample_bal.py | 111 ++++++++++++++++++++++++ 1 file changed, 111 insertions(+) create mode 100644 python/gtsam/examples/SFMExample_bal.py diff --git a/python/gtsam/examples/SFMExample_bal.py b/python/gtsam/examples/SFMExample_bal.py new file mode 100644 index 000000000..50e450fea --- /dev/null +++ b/python/gtsam/examples/SFMExample_bal.py @@ -0,0 +1,111 @@ +""" + GTSAM Copyright 2010, Georgia Tech Research Corporation, + Atlanta, Georgia 30332-0415 + All Rights Reserved + Authors: Frank Dellaert, et al. (see THANKS for the full author list) + + + See LICENSE for the license information + + Solve a structure-from-motion problem from a "Bundle Adjustment in the Large" file + Author: John Lambert +""" + +import argparse +import logging + +import gtsam +import matplotlib.pyplot as plt +import numpy as np + +from gtsam import symbol_shorthand +from gtsam import readBal + +L = symbol_shorthand.L +X = symbol_shorthand.X + +import pdb + +#include +#include +#include +#include +#include // for loading BAL datasets ! + + +# using symbol_shorthand::C; +# using symbol_shorthand::P; + +# We will be using a projection factor that ties a SFM_Camera to a 3D point. +# An SFM_Camera is defined in datase.h as a camera with unknown Cal3Bundler calibration +# and has a total of 9 free parameters +#typedef GeneralSFMFactor MyFactor; + + +def run(args): + """ Run LM optimization with BAL input data and report resulting error """ + # Find default file, but if an argument is given, try loading a file + if args.input_file: + input_file = args.input_file + else: + input_file = gtsam.findExampleDataFile("dubrovnik-3-7-pre") + + # # Load the SfM data from file + mydata = readBal(input_file) + logging.info(f"read {mydata.number_tracks()} tracks on {mydata.number_cameras()} cameras\n") + + # # Create a factor graph + graph = gtsam.NonlinearFactorGraph() + + # # We share *one* noiseModel between all projection factors + noise = gtsam.noiseModel.Isotropic.Sigma(2, 1.0) # one pixel in u and v + + # Add measurements to the factor graph + j = 0 + pdb.set_trace() + for t_idx in range(mydata.number_tracks()): + track = mydata.track(t_idx) # SfmTrack + for m_idx in range(track.number_measurements()): + # retrieve the SfmMeasurement + # i represents the camera index, and uv is the 2d measurement + i, uv = track.measurement(0) # + #graph.emplace_shared(uv, noise, C(i), P(j)) # note use of shorthand symbols C and P + #graph.add + j += 1 + pdb.set_trace() + + # # Add a prior on pose x1. This indirectly specifies where the origin is. + # # and a prior on the position of the first landmark to fix the scale + # graph.addPrior(C(0), mydata.cameras[0], gtsam.noiseModel.Isotropic.Sigma(9, 0.1)) + # graph.addPrior(P(0), mydata.tracks[0].p, gtsam.noiseModel.Isotropic.Sigma(3, 0.1)) + + # # Create initial estimate + initial = gtsam.Values() + # i = 0 + # j = 0; + # for(const SfmCamera& camera: mydata.cameras) initial.insert(C(i++), camera) + # for(const SfmTrack& track: mydata.tracks) initial.insert(P(j++), track.p) + + # Optimize the graph and print results + # Values result; + try: + params = gtsam.LevenbergMarquardtParams() + params.setVerbosityLM("ERROR") + lm = gtsam.LevenbergMarquardtOptimizer(graph, initial, params) + result = lm.optimize() + except Exception as e: + logging.exception("LM Optimization failed") + return + + logging.info(f"final error: {graph.error(result)}") + + +if __name__ == "__main__": + parser = argparse.ArgumentParser() + parser.add_argument('-i', '--input_file', type=str, default="", + help='Read SFM data from the specified BAL file') + run(parser.parse_args()) + + + + From 0d19859f8259c81bad5cf97b736b7f4d84e6ced1 Mon Sep 17 00:00:00 2001 From: John Lambert Date: Mon, 21 Sep 2020 12:28:46 -0400 Subject: [PATCH 18/26] add python equivalent of c++ sfm data calls --- python/gtsam/examples/SFMExample_bal.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/python/gtsam/examples/SFMExample_bal.py b/python/gtsam/examples/SFMExample_bal.py index 50e450fea..b70b8057b 100644 --- a/python/gtsam/examples/SFMExample_bal.py +++ b/python/gtsam/examples/SFMExample_bal.py @@ -81,8 +81,8 @@ def run(args): # # Create initial estimate initial = gtsam.Values() - # i = 0 - # j = 0; + i = 0 + j = 0 # for(const SfmCamera& camera: mydata.cameras) initial.insert(C(i++), camera) # for(const SfmTrack& track: mydata.tracks) initial.insert(P(j++), track.p) From afa74c4f575a5cd92442823c6acad41ff31efde7 Mon Sep 17 00:00:00 2001 From: John Lambert Date: Mon, 21 Sep 2020 14:21:33 -0400 Subject: [PATCH 19/26] cannot retrieve p attribute --- python/gtsam/examples/SFMExample_bal.py | 40 ++++++++++++++----------- 1 file changed, 23 insertions(+), 17 deletions(-) diff --git a/python/gtsam/examples/SFMExample_bal.py b/python/gtsam/examples/SFMExample_bal.py index b70b8057b..ab3baa26c 100644 --- a/python/gtsam/examples/SFMExample_bal.py +++ b/python/gtsam/examples/SFMExample_bal.py @@ -4,7 +4,6 @@ All Rights Reserved Authors: Frank Dellaert, et al. (see THANKS for the full author list) - See LICENSE for the license information Solve a structure-from-motion problem from a "Bundle Adjustment in the Large" file @@ -21,20 +20,12 @@ import numpy as np from gtsam import symbol_shorthand from gtsam import readBal -L = symbol_shorthand.L -X = symbol_shorthand.X +C = symbol_shorthand.C +P = symbol_shorthand.P import pdb -#include -#include -#include #include -#include // for loading BAL datasets ! - - -# using symbol_shorthand::C; -# using symbol_shorthand::P; # We will be using a projection factor that ties a SFM_Camera to a 3D point. # An SFM_Camera is defined in datase.h as a camera with unknown Cal3Bundler calibration @@ -65,8 +56,8 @@ def run(args): pdb.set_trace() for t_idx in range(mydata.number_tracks()): track = mydata.track(t_idx) # SfmTrack + # retrieve the SfmMeasurement objects for m_idx in range(track.number_measurements()): - # retrieve the SfmMeasurement # i represents the camera index, and uv is the 2d measurement i, uv = track.measurement(0) # #graph.emplace_shared(uv, noise, C(i), P(j)) # note use of shorthand symbols C and P @@ -76,18 +67,33 @@ def run(args): # # Add a prior on pose x1. This indirectly specifies where the origin is. # # and a prior on the position of the first landmark to fix the scale - # graph.addPrior(C(0), mydata.cameras[0], gtsam.noiseModel.Isotropic.Sigma(9, 0.1)) - # graph.addPrior(P(0), mydata.tracks[0].p, gtsam.noiseModel.Isotropic.Sigma(3, 0.1)) + + + graph.push_back(gtsam.PriorFactorVector( + C(0), mydata.camera(0), gtsam.noiseModel.Isotropic.Sigma(9, 0.1))) + # # graph.addPrior(P(0), mydata.track(0).p, + # equivalent of addPrior + graph.push_back(gtsam.PriorFactorVector( + P(0), mydata.track(0)[1], gtsam.noiseModel.Isotropic.Sigma(3, 0.1))) # # Create initial estimate initial = gtsam.Values() + i = 0 + # add each SfmCamera + for cam_idx in range(mydata.number_cameras()): + camera = mydata.camera(0) + initial.insert(C(i), camera) + i += 1 + j = 0 - # for(const SfmCamera& camera: mydata.cameras) initial.insert(C(i++), camera) - # for(const SfmTrack& track: mydata.tracks) initial.insert(P(j++), track.p) + # add each SfmTrack + for t_idx in range(mydata.number_tracks()): + track = mydata.track(0) + initial.insert(P(j), track.p) + j += 1 # Optimize the graph and print results - # Values result; try: params = gtsam.LevenbergMarquardtParams() params.setVerbosityLM("ERROR") From 34f670e9d5ba6225b7ea8705d4c7ad310429169f Mon Sep 17 00:00:00 2001 From: John Lambert Date: Tue, 22 Sep 2020 15:44:31 -0400 Subject: [PATCH 20/26] remove BAL for now, and add get3dPoint() for p access --- gtsam/gtsam.i | 1 + gtsam/slam/dataset.h | 3 + python/gtsam/examples/SFMExample_bal.py | 117 ------------------------ 3 files changed, 4 insertions(+), 117 deletions(-) delete mode 100644 python/gtsam/examples/SFMExample_bal.py diff --git a/gtsam/gtsam.i b/gtsam/gtsam.i index 52f5901ee..78e004390 100644 --- a/gtsam/gtsam.i +++ b/gtsam/gtsam.i @@ -2847,6 +2847,7 @@ virtual class EssentialMatrixFactor : gtsam::NoiseModelFactor { #include class SfmTrack { + Point3 get3dPoint() const; size_t number_measurements() const; pair measurement(size_t idx) const; pair siftIndex(size_t idx) const; diff --git a/gtsam/slam/dataset.h b/gtsam/slam/dataset.h index 91ceaa5fd..5bb8063bd 100644 --- a/gtsam/slam/dataset.h +++ b/gtsam/slam/dataset.h @@ -233,6 +233,9 @@ struct SfmTrack { SiftIndex siftIndex(size_t idx) const { return siftIndices[idx]; } + Point3 get3dPoint() const { + return p; + } }; /// Define the structure for the camera poses diff --git a/python/gtsam/examples/SFMExample_bal.py b/python/gtsam/examples/SFMExample_bal.py deleted file mode 100644 index ab3baa26c..000000000 --- a/python/gtsam/examples/SFMExample_bal.py +++ /dev/null @@ -1,117 +0,0 @@ -""" - GTSAM Copyright 2010, Georgia Tech Research Corporation, - Atlanta, Georgia 30332-0415 - All Rights Reserved - Authors: Frank Dellaert, et al. (see THANKS for the full author list) - - See LICENSE for the license information - - Solve a structure-from-motion problem from a "Bundle Adjustment in the Large" file - Author: John Lambert -""" - -import argparse -import logging - -import gtsam -import matplotlib.pyplot as plt -import numpy as np - -from gtsam import symbol_shorthand -from gtsam import readBal - -C = symbol_shorthand.C -P = symbol_shorthand.P - -import pdb - -#include - -# We will be using a projection factor that ties a SFM_Camera to a 3D point. -# An SFM_Camera is defined in datase.h as a camera with unknown Cal3Bundler calibration -# and has a total of 9 free parameters -#typedef GeneralSFMFactor MyFactor; - - -def run(args): - """ Run LM optimization with BAL input data and report resulting error """ - # Find default file, but if an argument is given, try loading a file - if args.input_file: - input_file = args.input_file - else: - input_file = gtsam.findExampleDataFile("dubrovnik-3-7-pre") - - # # Load the SfM data from file - mydata = readBal(input_file) - logging.info(f"read {mydata.number_tracks()} tracks on {mydata.number_cameras()} cameras\n") - - # # Create a factor graph - graph = gtsam.NonlinearFactorGraph() - - # # We share *one* noiseModel between all projection factors - noise = gtsam.noiseModel.Isotropic.Sigma(2, 1.0) # one pixel in u and v - - # Add measurements to the factor graph - j = 0 - pdb.set_trace() - for t_idx in range(mydata.number_tracks()): - track = mydata.track(t_idx) # SfmTrack - # retrieve the SfmMeasurement objects - for m_idx in range(track.number_measurements()): - # i represents the camera index, and uv is the 2d measurement - i, uv = track.measurement(0) # - #graph.emplace_shared(uv, noise, C(i), P(j)) # note use of shorthand symbols C and P - #graph.add - j += 1 - pdb.set_trace() - - # # Add a prior on pose x1. This indirectly specifies where the origin is. - # # and a prior on the position of the first landmark to fix the scale - - - graph.push_back(gtsam.PriorFactorVector( - C(0), mydata.camera(0), gtsam.noiseModel.Isotropic.Sigma(9, 0.1))) - # # graph.addPrior(P(0), mydata.track(0).p, - # equivalent of addPrior - graph.push_back(gtsam.PriorFactorVector( - P(0), mydata.track(0)[1], gtsam.noiseModel.Isotropic.Sigma(3, 0.1))) - - # # Create initial estimate - initial = gtsam.Values() - - i = 0 - # add each SfmCamera - for cam_idx in range(mydata.number_cameras()): - camera = mydata.camera(0) - initial.insert(C(i), camera) - i += 1 - - j = 0 - # add each SfmTrack - for t_idx in range(mydata.number_tracks()): - track = mydata.track(0) - initial.insert(P(j), track.p) - j += 1 - - # Optimize the graph and print results - try: - params = gtsam.LevenbergMarquardtParams() - params.setVerbosityLM("ERROR") - lm = gtsam.LevenbergMarquardtOptimizer(graph, initial, params) - result = lm.optimize() - except Exception as e: - logging.exception("LM Optimization failed") - return - - logging.info(f"final error: {graph.error(result)}") - - -if __name__ == "__main__": - parser = argparse.ArgumentParser() - parser.add_argument('-i', '--input_file', type=str, default="", - help='Read SFM data from the specified BAL file') - run(parser.parse_args()) - - - - From 8e0b0c1641fa6baf897250ccb33d3959f1d79ae5 Mon Sep 17 00:00:00 2001 From: John Lambert Date: Tue, 22 Sep 2020 15:49:05 -0400 Subject: [PATCH 21/26] mark SFMExample_bal as still in progress --- python/gtsam/examples/README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/python/gtsam/examples/README.md b/python/gtsam/examples/README.md index 84a41990f..e05a0c7e1 100644 --- a/python/gtsam/examples/README.md +++ b/python/gtsam/examples/README.md @@ -39,7 +39,7 @@ | SelfCalibrationExample | | | SFMdata | | | SFMExample_bal_COLAMD_METIS | | -| SFMExample_bal | :heavy_check_mark: | +| SFMExample_bal | | | SFMExample | :heavy_check_mark: | | SFMExampleExpressions_bal | | | SFMExampleExpressions | | From 0afec43612cfe3fabdc0227ea95c9590c46c7ec9 Mon Sep 17 00:00:00 2001 From: Varun Agrawal Date: Tue, 22 Sep 2020 17:05:42 -0400 Subject: [PATCH 22/26] remove logmap function from LieVector and LieMatrix --- gtsam/gtsam.i | 2 -- 1 file changed, 2 deletions(-) diff --git a/gtsam/gtsam.i b/gtsam/gtsam.i index d8297c43b..d75d29ef2 100644 --- a/gtsam/gtsam.i +++ b/gtsam/gtsam.i @@ -390,7 +390,6 @@ class LieVector { // Lie group static gtsam::LieVector Expmap(Vector v); static Vector Logmap(const gtsam::LieVector& p); - Vector logmap(const gtsam::LieVector& p); // enabling serialization functionality void serialize() const; @@ -423,7 +422,6 @@ class LieMatrix { // Lie group static gtsam::LieMatrix Expmap(Vector v); static Vector Logmap(const gtsam::LieMatrix& p); - Vector logmap(const gtsam::LieMatrix& p); // enabling serialization functionality void serialize() const; From e6057fc4faf5f44f223ca3f0d62018231c5d099a Mon Sep 17 00:00:00 2001 From: John Lambert Date: Tue, 22 Sep 2020 23:20:36 -0400 Subject: [PATCH 23/26] rename get3dPoint() to point3() --- gtsam/gtsam.i | 2 +- gtsam/slam/dataset.h | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/gtsam/gtsam.i b/gtsam/gtsam.i index 78e004390..1764ecd82 100644 --- a/gtsam/gtsam.i +++ b/gtsam/gtsam.i @@ -2847,7 +2847,7 @@ virtual class EssentialMatrixFactor : gtsam::NoiseModelFactor { #include class SfmTrack { - Point3 get3dPoint() const; + Point3 point3() const; size_t number_measurements() const; pair measurement(size_t idx) const; pair siftIndex(size_t idx) const; diff --git a/gtsam/slam/dataset.h b/gtsam/slam/dataset.h index 5bb8063bd..a8fddcfe4 100644 --- a/gtsam/slam/dataset.h +++ b/gtsam/slam/dataset.h @@ -233,7 +233,7 @@ struct SfmTrack { SiftIndex siftIndex(size_t idx) const { return siftIndices[idx]; } - Point3 get3dPoint() const { + Point3 point3() const { return p; } }; From 74c4a60e83d1d6d6e4d4fd14303477af7c6031a7 Mon Sep 17 00:00:00 2001 From: Varun Agrawal Date: Thu, 24 Sep 2020 17:34:30 -0400 Subject: [PATCH 24/26] small fixes to ensure marginals are computed correctly --- python/gtsam/examples/ImuFactorExample.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/python/gtsam/examples/ImuFactorExample.py b/python/gtsam/examples/ImuFactorExample.py index eec7c5ebd..bb707a8f5 100644 --- a/python/gtsam/examples/ImuFactorExample.py +++ b/python/gtsam/examples/ImuFactorExample.py @@ -104,7 +104,7 @@ class ImuFactorExample(PreintegrationExample): if verbose: print(factor) - print(pim.predict(actual_state_i, self.actualBias)) + print(pim.predict(initial_state_i, self.actualBias)) pim.resetIntegration() @@ -125,7 +125,7 @@ class ImuFactorExample(PreintegrationExample): i += 1 # add priors on end - # self.addPrior(num_poses - 1, graph) + self.addPrior(num_poses - 1, graph) initial.print_("Initial values:") From 52fc9cf4ba86582d96d4aec3b7a17fece655bd9f Mon Sep 17 00:00:00 2001 From: JIanzhu Huai Date: Fri, 25 Sep 2020 19:47:07 +0800 Subject: [PATCH 25/26] test Qr with old codebase fails --- gtsam/geometry/Pose3.cpp | 18 +++++------------- gtsam/geometry/Pose3.h | 11 ++++++++++- gtsam/geometry/tests/testPose3.cpp | 20 +++++++++++--------- 3 files changed, 26 insertions(+), 23 deletions(-) diff --git a/gtsam/geometry/Pose3.cpp b/gtsam/geometry/Pose3.cpp index 2e6a2136e..12f628731 100644 --- a/gtsam/geometry/Pose3.cpp +++ b/gtsam/geometry/Pose3.cpp @@ -190,15 +190,7 @@ Vector6 Pose3::ChartAtOrigin::Local(const Pose3& pose, ChartJacobian Hpose) { } /* ************************************************************************* */ -/** - * Compute the 3x3 bottom-left block Q of the SE3 Expmap derivative matrix - * J(xi) = [J_(w) Z_3x3; - * Q J_(w)] - * where J_(w) is the SO3 Expmap derivative. - * (see Chirikjian11book2, pg 44, eq 10.95. - * The closed-form formula is similar to formula 102 in Barfoot14tro) - */ -Matrix3 Pose3::computeQforExpmapDerivative(const Vector6& xi, double nearZeroThreshold) { +Matrix3 Pose3::ComputeQforExpmapDerivative(const Vector6& xi, double nearZeroThreshold) { const auto w = xi.head<3>(); const auto v = xi.tail<3>(); const Matrix3 V = skewSymmetric(v); @@ -230,8 +222,8 @@ Matrix3 Pose3::computeQforExpmapDerivative(const Vector6& xi, double nearZeroThr } else { Q = -0.5*V + 1./6.*(W*V + V*W - W*V*W) - - 1./24.*(W*W*V + V*W*W - 3*W*V*W) - + 1./120.*(W*V*W*W + 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); } #endif @@ -242,7 +234,7 @@ Matrix3 Pose3::computeQforExpmapDerivative(const Vector6& xi, double nearZeroThr Matrix6 Pose3::ExpmapDerivative(const Vector6& xi) { const Vector3 w = xi.head<3>(); const Matrix3 Jw = Rot3::ExpmapDerivative(w); - const Matrix3 Q = computeQforExpmapDerivative(xi); + const Matrix3 Q = ComputeQforExpmapDerivative(xi); Matrix6 J; J << Jw, Z_3x3, Q, Jw; return J; @@ -253,7 +245,7 @@ Matrix6 Pose3::LogmapDerivative(const Pose3& pose) { const Vector6 xi = Logmap(pose); const Vector3 w = xi.head<3>(); const Matrix3 Jw = Rot3::LogmapDerivative(w); - const Matrix3 Q = computeQforExpmapDerivative(xi); + const Matrix3 Q = ComputeQforExpmapDerivative(xi); const Matrix3 Q2 = -Jw*Q*Jw; Matrix6 J; J << Jw, Z_3x3, Q2, Jw; diff --git a/gtsam/geometry/Pose3.h b/gtsam/geometry/Pose3.h index 575fbaa5c..2f0802cab 100644 --- a/gtsam/geometry/Pose3.h +++ b/gtsam/geometry/Pose3.h @@ -181,7 +181,16 @@ public: static Vector6 Local(const Pose3& pose, ChartJacobian Hpose = boost::none); }; - static Matrix3 computeQforExpmapDerivative( + /** + * Compute the 3x3 bottom-left block Q of SE3 Expmap right derivative matrix + * J_r(xi) = [J_(w) Z_3x3; + * Q_r J_(w)] + * where J_(w) is the SO3 Expmap right derivative. + * (see Chirikjian11book2, pg 44, eq 10.95. + * The closed-form formula is identical to formula 102 in Barfoot14tro where + * Q_l of the SE3 Expmap left derivative matrix is given. + */ + static Matrix3 ComputeQforExpmapDerivative( const Vector6& xi, double nearZeroThreshold = 1e-5); using LieGroup::inverse; // version with derivative diff --git a/gtsam/geometry/tests/testPose3.cpp b/gtsam/geometry/tests/testPose3.cpp index 55720abca..9639ffbcf 100644 --- a/gtsam/geometry/tests/testPose3.cpp +++ b/gtsam/geometry/tests/testPose3.cpp @@ -807,6 +807,17 @@ TEST(Pose3, ExpmapDerivative2) { } } +TEST( Pose3, ExpmapDerivativeQr) { + Vector6 w = Vector6::Random(); + w.head<3>().normalize(); + w.head<3>() = w.head<3>() * 0.9e-2; + Matrix3 actualQr = Pose3::ComputeQforExpmapDerivative(w, 0.01); + Matrix expectedH = numericalDerivative21 >(&Pose3::Expmap, w, boost::none); + Matrix3 expectedQr = expectedH.bottomLeftCorner<3, 3>(); + EXPECT(assert_equal(expectedQr, actualQr, 1e-6)); +} + /* ************************************************************************* */ TEST( Pose3, LogmapDerivative) { Matrix6 actualH; @@ -818,15 +829,6 @@ TEST( Pose3, LogmapDerivative) { EXPECT(assert_equal(expectedH, actualH)); } -TEST( Pose3, computeQforExpmapDerivative) { - Vector6 w = Vector6::Random(); - w.head<3>().normalize(); - w.head<3>() = w.head<3>() * 0.09; - Matrix3 Qexact = Pose3::computeQforExpmapDerivative(w); - Matrix3 Qapprox = Pose3::computeQforExpmapDerivative(w, 0.1); - EXPECT(assert_equal(Qapprox, Qexact, 1e-5)); -} - /* ************************************************************************* */ Vector6 testDerivAdjoint(const Vector6& xi, const Vector6& v) { return Pose3::adjointMap(xi) * v; From 2550bf76a493ac5148297c09f5adc45f5cf872b9 Mon Sep 17 00:00:00 2001 From: JIanzhu Huai Date: Fri, 25 Sep 2020 20:26:24 +0800 Subject: [PATCH 26/26] correct Qr coefficients in approximation --- gtsam/geometry/Pose3.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/gtsam/geometry/Pose3.cpp b/gtsam/geometry/Pose3.cpp index 12f628731..ea822b796 100644 --- a/gtsam/geometry/Pose3.cpp +++ b/gtsam/geometry/Pose3.cpp @@ -222,8 +222,8 @@ Matrix3 Pose3::ComputeQforExpmapDerivative(const Vector6& xi, double nearZeroThr } else { Q = -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); + - 1./24.*(W*W*V + V*W*W - 3*W*V*W) + + 1./120.*(W*V*W*W + W*W*V*W); } #endif