Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 1 | // Ceres Solver - A fast non-linear least squares minimizer |
Sameer Agarwal | 5a30cae | 2023-09-19 15:29:34 -0700 | [diff] [blame] | 2 | // Copyright 2023 Google Inc. All rights reserved. |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 3 | // http://ceres-solver.org/ |
| 4 | // |
| 5 | // Redistribution and use in source and binary forms, with or without |
| 6 | // modification, are permitted provided that the following conditions are met: |
| 7 | // |
| 8 | // * Redistributions of source code must retain the above copyright notice, |
| 9 | // this list of conditions and the following disclaimer. |
| 10 | // * Redistributions in binary form must reproduce the above copyright notice, |
| 11 | // this list of conditions and the following disclaimer in the documentation |
| 12 | // and/or other materials provided with the distribution. |
| 13 | // * Neither the name of Google Inc. nor the names of its contributors may be |
| 14 | // used to endorse or promote products derived from this software without |
| 15 | // specific prior written permission. |
| 16 | // |
| 17 | // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
| 18 | // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| 19 | // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE |
| 20 | // ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE |
| 21 | // LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| 22 | // CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| 23 | // SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
| 24 | // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN |
| 25 | // CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
| 26 | // ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE |
| 27 | // POSSIBILITY OF SUCH DAMAGE. |
| 28 | // |
| 29 | // Author: sameeragarwal@google.com (Sameer Agarwal) |
| 30 | |
| 31 | #include "ceres/manifold.h" |
| 32 | |
| 33 | #include <cmath> |
| 34 | #include <limits> |
| 35 | #include <memory> |
Sameer Agarwal | 5723950 | 2022-03-03 09:37:52 -0800 | [diff] [blame] | 36 | #include <utility> |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 37 | |
| 38 | #include "Eigen/Geometry" |
Sergiu Deitsch | 4519b8d | 2023-10-04 22:45:42 +0200 | [diff] [blame] | 39 | #include "ceres/constants.h" |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 40 | #include "ceres/dynamic_numeric_diff_cost_function.h" |
| 41 | #include "ceres/internal/eigen.h" |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 42 | #include "ceres/internal/port.h" |
Sameer Agarwal | eadfead | 2022-03-02 06:48:20 -0800 | [diff] [blame] | 43 | #include "ceres/line_manifold.h" |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 44 | #include "ceres/manifold_test_utils.h" |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 45 | #include "ceres/numeric_diff_options.h" |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 46 | #include "ceres/product_manifold.h" |
Sameer Agarwal | 4a01dcb | 2022-01-06 08:08:46 -0800 | [diff] [blame] | 47 | #include "ceres/rotation.h" |
Sameer Agarwal | eadfead | 2022-03-02 06:48:20 -0800 | [diff] [blame] | 48 | #include "ceres/sphere_manifold.h" |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 49 | #include "ceres/types.h" |
| 50 | #include "gmock/gmock.h" |
| 51 | #include "gtest/gtest.h" |
| 52 | |
Sameer Agarwal | caf614a | 2022-04-21 17:41:10 -0700 | [diff] [blame] | 53 | namespace ceres::internal { |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 54 | |
Sameer Agarwal | 4a01dcb | 2022-01-06 08:08:46 -0800 | [diff] [blame] | 55 | constexpr int kNumTrials = 1000; |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 56 | constexpr double kTolerance = 1e-9; |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 57 | |
Sameer Agarwal | 6a37fbf | 2022-02-28 11:23:43 -0800 | [diff] [blame] | 58 | TEST(EuclideanManifold, StaticNormalFunctionTest) { |
| 59 | EuclideanManifold<3> manifold; |
| 60 | EXPECT_EQ(manifold.AmbientSize(), 3); |
| 61 | EXPECT_EQ(manifold.TangentSize(), 3); |
| 62 | |
| 63 | Vector zero_tangent = Vector::Zero(manifold.TangentSize()); |
| 64 | for (int trial = 0; trial < kNumTrials; ++trial) { |
| 65 | const Vector x = Vector::Random(manifold.AmbientSize()); |
| 66 | const Vector y = Vector::Random(manifold.AmbientSize()); |
| 67 | Vector delta = Vector::Random(manifold.TangentSize()); |
| 68 | Vector x_plus_delta = Vector::Zero(manifold.AmbientSize()); |
| 69 | |
| 70 | manifold.Plus(x.data(), delta.data(), x_plus_delta.data()); |
| 71 | EXPECT_NEAR((x_plus_delta - x - delta).norm() / (x + delta).norm(), |
| 72 | 0.0, |
| 73 | kTolerance); |
| 74 | |
| 75 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
| 76 | } |
| 77 | } |
| 78 | |
| 79 | TEST(EuclideanManifold, DynamicNormalFunctionTest) { |
| 80 | EuclideanManifold<DYNAMIC> manifold(3); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 81 | EXPECT_EQ(manifold.AmbientSize(), 3); |
| 82 | EXPECT_EQ(manifold.TangentSize(), 3); |
| 83 | |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 84 | Vector zero_tangent = Vector::Zero(manifold.TangentSize()); |
| 85 | for (int trial = 0; trial < kNumTrials; ++trial) { |
| 86 | const Vector x = Vector::Random(manifold.AmbientSize()); |
Sameer Agarwal | 4a01dcb | 2022-01-06 08:08:46 -0800 | [diff] [blame] | 87 | const Vector y = Vector::Random(manifold.AmbientSize()); |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 88 | Vector delta = Vector::Random(manifold.TangentSize()); |
| 89 | Vector x_plus_delta = Vector::Zero(manifold.AmbientSize()); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 90 | |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 91 | manifold.Plus(x.data(), delta.data(), x_plus_delta.data()); |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 92 | EXPECT_NEAR((x_plus_delta - x - delta).norm() / (x + delta).norm(), |
| 93 | 0.0, |
| 94 | kTolerance); |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 95 | |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 96 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 97 | } |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 98 | } |
| 99 | |
| 100 | TEST(SubsetManifold, EmptyConstantParameters) { |
| 101 | SubsetManifold manifold(3, {}); |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 102 | for (int trial = 0; trial < kNumTrials; ++trial) { |
Sameer Agarwal | 4a01dcb | 2022-01-06 08:08:46 -0800 | [diff] [blame] | 103 | const Vector x = Vector::Random(3); |
| 104 | const Vector y = Vector::Random(3); |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 105 | Vector delta = Vector::Random(3); |
| 106 | Vector x_plus_delta = Vector::Zero(3); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 107 | |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 108 | manifold.Plus(x.data(), delta.data(), x_plus_delta.data()); |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 109 | EXPECT_NEAR((x_plus_delta - x - delta).norm() / (x + delta).norm(), |
| 110 | 0.0, |
| 111 | kTolerance); |
| 112 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 113 | } |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 114 | } |
| 115 | |
| 116 | TEST(SubsetManifold, NegativeParameterIndexDeathTest) { |
| 117 | EXPECT_DEATH_IF_SUPPORTED(SubsetManifold manifold(2, {-1}), |
| 118 | "greater than equal to zero"); |
| 119 | } |
| 120 | |
| 121 | TEST(SubsetManifold, GreaterThanSizeParameterIndexDeathTest) { |
| 122 | EXPECT_DEATH_IF_SUPPORTED(SubsetManifold manifold(2, {2}), |
| 123 | "less than the size"); |
| 124 | } |
| 125 | |
| 126 | TEST(SubsetManifold, DuplicateParametersDeathTest) { |
| 127 | EXPECT_DEATH_IF_SUPPORTED(SubsetManifold manifold(2, {1, 1}), "duplicates"); |
| 128 | } |
| 129 | |
| 130 | TEST(SubsetManifold, NormalFunctionTest) { |
| 131 | const int kAmbientSize = 4; |
| 132 | const int kTangentSize = 3; |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 133 | |
| 134 | for (int i = 0; i < kAmbientSize; ++i) { |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 135 | SubsetManifold manifold_with_ith_parameter_constant(kAmbientSize, {i}); |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 136 | for (int trial = 0; trial < kNumTrials; ++trial) { |
| 137 | const Vector x = Vector::Random(kAmbientSize); |
Sameer Agarwal | 4a01dcb | 2022-01-06 08:08:46 -0800 | [diff] [blame] | 138 | Vector y = Vector::Random(kAmbientSize); |
| 139 | // x and y must have the same i^th coordinate to be on the manifold. |
| 140 | y[i] = x[i]; |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 141 | Vector delta = Vector::Random(kTangentSize); |
| 142 | Vector x_plus_delta = Vector::Zero(kAmbientSize); |
| 143 | |
| 144 | x_plus_delta.setZero(); |
| 145 | manifold_with_ith_parameter_constant.Plus( |
| 146 | x.data(), delta.data(), x_plus_delta.data()); |
| 147 | int k = 0; |
| 148 | for (int j = 0; j < kAmbientSize; ++j) { |
| 149 | if (j == i) { |
| 150 | EXPECT_EQ(x_plus_delta[j], x[j]); |
| 151 | } else { |
| 152 | EXPECT_EQ(x_plus_delta[j], x[j] + delta[k++]); |
| 153 | } |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 154 | } |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 155 | |
| 156 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD( |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 157 | manifold_with_ith_parameter_constant, x, delta, y, kTolerance); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 158 | } |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 159 | } |
| 160 | } |
| 161 | |
| 162 | TEST(ProductManifold, Size2) { |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 163 | SubsetManifold manifold1(5, {2}); |
| 164 | SubsetManifold manifold2(3, {0, 1}); |
| 165 | ProductManifold<SubsetManifold, SubsetManifold> manifold(manifold1, |
| 166 | manifold2); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 167 | |
| 168 | EXPECT_EQ(manifold.AmbientSize(), |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 169 | manifold1.AmbientSize() + manifold2.AmbientSize()); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 170 | EXPECT_EQ(manifold.TangentSize(), |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 171 | manifold1.TangentSize() + manifold2.TangentSize()); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 172 | } |
| 173 | |
| 174 | TEST(ProductManifold, Size3) { |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 175 | SubsetManifold manifold1(5, {2}); |
| 176 | SubsetManifold manifold2(3, {0, 1}); |
| 177 | SubsetManifold manifold3(4, {1}); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 178 | |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 179 | ProductManifold<SubsetManifold, SubsetManifold, SubsetManifold> manifold( |
| 180 | manifold1, manifold2, manifold3); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 181 | |
| 182 | EXPECT_EQ(manifold.AmbientSize(), |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 183 | manifold1.AmbientSize() + manifold2.AmbientSize() + |
| 184 | manifold3.AmbientSize()); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 185 | EXPECT_EQ(manifold.TangentSize(), |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 186 | manifold1.TangentSize() + manifold2.TangentSize() + |
| 187 | manifold3.TangentSize()); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 188 | } |
| 189 | |
| 190 | TEST(ProductManifold, Size4) { |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 191 | SubsetManifold manifold1(5, {2}); |
| 192 | SubsetManifold manifold2(3, {0, 1}); |
| 193 | SubsetManifold manifold3(4, {1}); |
| 194 | SubsetManifold manifold4(2, {0}); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 195 | |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 196 | ProductManifold<SubsetManifold, |
| 197 | SubsetManifold, |
| 198 | SubsetManifold, |
| 199 | SubsetManifold> |
| 200 | manifold(manifold1, manifold2, manifold3, manifold4); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 201 | |
| 202 | EXPECT_EQ(manifold.AmbientSize(), |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 203 | manifold1.AmbientSize() + manifold2.AmbientSize() + |
| 204 | manifold3.AmbientSize() + manifold4.AmbientSize()); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 205 | EXPECT_EQ(manifold.TangentSize(), |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 206 | manifold1.TangentSize() + manifold2.TangentSize() + |
| 207 | manifold3.TangentSize() + manifold4.TangentSize()); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 208 | } |
| 209 | |
| 210 | TEST(ProductManifold, NormalFunctionTest) { |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 211 | SubsetManifold manifold1(5, {2}); |
| 212 | SubsetManifold manifold2(3, {0, 1}); |
| 213 | SubsetManifold manifold3(4, {1}); |
| 214 | SubsetManifold manifold4(2, {0}); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 215 | |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 216 | ProductManifold<SubsetManifold, |
| 217 | SubsetManifold, |
| 218 | SubsetManifold, |
| 219 | SubsetManifold> |
| 220 | manifold(manifold1, manifold2, manifold3, manifold4); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 221 | |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 222 | for (int trial = 0; trial < kNumTrials; ++trial) { |
Sameer Agarwal | 4a01dcb | 2022-01-06 08:08:46 -0800 | [diff] [blame] | 223 | const Vector x = Vector::Random(manifold.AmbientSize()); |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 224 | Vector delta = Vector::Random(manifold.TangentSize()); |
| 225 | Vector x_plus_delta = Vector::Zero(manifold.AmbientSize()); |
| 226 | Vector x_plus_delta_expected = Vector::Zero(manifold.AmbientSize()); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 227 | |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 228 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), x_plus_delta.data())); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 229 | |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 230 | int ambient_cursor = 0; |
| 231 | int tangent_cursor = 0; |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 232 | |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 233 | EXPECT_TRUE(manifold1.Plus(&x[ambient_cursor], |
| 234 | &delta[tangent_cursor], |
| 235 | &x_plus_delta_expected[ambient_cursor])); |
| 236 | ambient_cursor += manifold1.AmbientSize(); |
| 237 | tangent_cursor += manifold1.TangentSize(); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 238 | |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 239 | EXPECT_TRUE(manifold2.Plus(&x[ambient_cursor], |
| 240 | &delta[tangent_cursor], |
| 241 | &x_plus_delta_expected[ambient_cursor])); |
| 242 | ambient_cursor += manifold2.AmbientSize(); |
| 243 | tangent_cursor += manifold2.TangentSize(); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 244 | |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 245 | EXPECT_TRUE(manifold3.Plus(&x[ambient_cursor], |
| 246 | &delta[tangent_cursor], |
| 247 | &x_plus_delta_expected[ambient_cursor])); |
| 248 | ambient_cursor += manifold3.AmbientSize(); |
| 249 | tangent_cursor += manifold3.TangentSize(); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 250 | |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 251 | EXPECT_TRUE(manifold4.Plus(&x[ambient_cursor], |
| 252 | &delta[tangent_cursor], |
| 253 | &x_plus_delta_expected[ambient_cursor])); |
| 254 | ambient_cursor += manifold4.AmbientSize(); |
| 255 | tangent_cursor += manifold4.TangentSize(); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 256 | |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 257 | for (int i = 0; i < x.size(); ++i) { |
| 258 | EXPECT_EQ(x_plus_delta[i], x_plus_delta_expected[i]); |
| 259 | } |
| 260 | |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 261 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD( |
| 262 | manifold, x, delta, x_plus_delta, kTolerance); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 263 | } |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 264 | } |
| 265 | |
| 266 | TEST(ProductManifold, ZeroTangentSizeAndEuclidean) { |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 267 | SubsetManifold subset_manifold(1, {0}); |
| 268 | EuclideanManifold<2> euclidean_manifold; |
| 269 | ProductManifold<SubsetManifold, EuclideanManifold<2>> manifold( |
| 270 | subset_manifold, euclidean_manifold); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 271 | EXPECT_EQ(manifold.AmbientSize(), 3); |
| 272 | EXPECT_EQ(manifold.TangentSize(), 2); |
| 273 | |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 274 | for (int trial = 0; trial < kNumTrials; ++trial) { |
Sameer Agarwal | 4a01dcb | 2022-01-06 08:08:46 -0800 | [diff] [blame] | 275 | const Vector x = Vector::Random(3); |
| 276 | Vector y = Vector::Random(3); |
| 277 | y[0] = x[0]; |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 278 | Vector delta = Vector::Random(2); |
| 279 | Vector x_plus_delta = Vector::Zero(3); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 280 | |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 281 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), x_plus_delta.data())); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 282 | |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 283 | EXPECT_EQ(x_plus_delta[0], x[0]); |
| 284 | EXPECT_EQ(x_plus_delta[1], x[1] + delta[0]); |
| 285 | EXPECT_EQ(x_plus_delta[2], x[2] + delta[1]); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 286 | |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 287 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 288 | } |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 289 | } |
| 290 | |
| 291 | TEST(ProductManifold, EuclideanAndZeroTangentSize) { |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 292 | SubsetManifold subset_manifold(1, {0}); |
| 293 | EuclideanManifold<2> euclidean_manifold; |
| 294 | ProductManifold<EuclideanManifold<2>, SubsetManifold> manifold( |
| 295 | euclidean_manifold, subset_manifold); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 296 | EXPECT_EQ(manifold.AmbientSize(), 3); |
| 297 | EXPECT_EQ(manifold.TangentSize(), 2); |
| 298 | |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 299 | for (int trial = 0; trial < kNumTrials; ++trial) { |
Sameer Agarwal | 4a01dcb | 2022-01-06 08:08:46 -0800 | [diff] [blame] | 300 | const Vector x = Vector::Random(3); |
| 301 | Vector y = Vector::Random(3); |
| 302 | y[2] = x[2]; |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 303 | Vector delta = Vector::Random(2); |
| 304 | Vector x_plus_delta = Vector::Zero(3); |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 305 | |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 306 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), x_plus_delta.data())); |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 307 | EXPECT_EQ(x_plus_delta[0], x[0] + delta[0]); |
| 308 | EXPECT_EQ(x_plus_delta[1], x[1] + delta[1]); |
| 309 | EXPECT_EQ(x_plus_delta[2], x[2]); |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 310 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
Sameer Agarwal | bdd80fc | 2022-01-04 13:41:00 -0800 | [diff] [blame] | 311 | } |
Sameer Agarwal | 23b204d | 2021-12-28 07:05:03 -0800 | [diff] [blame] | 312 | } |
| 313 | |
Sergiu Deitsch | 7743d2e | 2022-02-21 22:43:32 +0100 | [diff] [blame] | 314 | struct CopyableManifold : ceres::Manifold { |
| 315 | CopyableManifold() = default; |
| 316 | CopyableManifold(const CopyableManifold&) = default; |
| 317 | // Do not care about copy-assignment |
| 318 | CopyableManifold& operator=(const CopyableManifold&) = delete; |
| 319 | // Not moveable |
| 320 | CopyableManifold(CopyableManifold&&) = delete; |
| 321 | CopyableManifold& operator=(CopyableManifold&&) = delete; |
| 322 | |
| 323 | int AmbientSize() const override { return 3; } |
| 324 | int TangentSize() const override { return 2; } |
| 325 | |
| 326 | bool Plus(const double* x, |
| 327 | const double* delta, |
| 328 | double* x_plus_delta) const override { |
| 329 | return true; |
| 330 | } |
| 331 | |
| 332 | bool PlusJacobian(const double* x, double* jacobian) const override { |
| 333 | return true; |
| 334 | } |
| 335 | |
| 336 | bool RightMultiplyByPlusJacobian(const double* x, |
| 337 | const int num_rows, |
| 338 | const double* ambient_matrix, |
| 339 | double* tangent_matrix) const override { |
| 340 | return true; |
| 341 | } |
| 342 | |
| 343 | bool Minus(const double* y, |
| 344 | const double* x, |
| 345 | double* y_minus_x) const override { |
| 346 | return true; |
| 347 | } |
| 348 | |
| 349 | bool MinusJacobian(const double* x, double* jacobian) const override { |
| 350 | return true; |
| 351 | } |
| 352 | }; |
| 353 | |
| 354 | struct MoveableManifold : ceres::Manifold { |
| 355 | MoveableManifold() = default; |
| 356 | MoveableManifold(MoveableManifold&&) = default; |
| 357 | // Do not care about move-assignment |
| 358 | MoveableManifold& operator=(MoveableManifold&&) = delete; |
| 359 | // Not copyable |
| 360 | MoveableManifold(const MoveableManifold&) = delete; |
| 361 | MoveableManifold& operator=(const MoveableManifold&) = delete; |
| 362 | |
| 363 | int AmbientSize() const override { return 3; } |
| 364 | int TangentSize() const override { return 2; } |
| 365 | |
| 366 | bool Plus(const double* x, |
| 367 | const double* delta, |
| 368 | double* x_plus_delta) const override { |
| 369 | return true; |
| 370 | } |
| 371 | |
| 372 | bool PlusJacobian(const double* x, double* jacobian) const override { |
| 373 | return true; |
| 374 | } |
| 375 | |
| 376 | bool RightMultiplyByPlusJacobian(const double* x, |
| 377 | const int num_rows, |
| 378 | const double* ambient_matrix, |
| 379 | double* tangent_matrix) const override { |
| 380 | return true; |
| 381 | } |
| 382 | |
| 383 | bool Minus(const double* y, |
| 384 | const double* x, |
| 385 | double* y_minus_x) const override { |
| 386 | return true; |
| 387 | } |
| 388 | |
| 389 | bool MinusJacobian(const double* x, double* jacobian) const override { |
| 390 | return true; |
| 391 | } |
| 392 | }; |
| 393 | |
| 394 | TEST(ProductManifold, CopyableOnly) { |
| 395 | ProductManifold<CopyableManifold, EuclideanManifold<3>> manifold1{ |
| 396 | CopyableManifold{}, EuclideanManifold<3>{}}; |
| 397 | |
| 398 | CopyableManifold inner2; |
| 399 | ProductManifold<CopyableManifold, EuclideanManifold<3>> manifold2{ |
| 400 | inner2, EuclideanManifold<3>{}}; |
| 401 | |
| 402 | EXPECT_EQ(manifold1.AmbientSize(), manifold2.AmbientSize()); |
| 403 | EXPECT_EQ(manifold1.TangentSize(), manifold2.TangentSize()); |
| 404 | } |
| 405 | |
| 406 | TEST(ProductManifold, MoveableOnly) { |
| 407 | ProductManifold<MoveableManifold, EuclideanManifold<3>> manifold1{ |
| 408 | MoveableManifold{}, EuclideanManifold<3>{}}; |
| 409 | |
| 410 | MoveableManifold inner2; |
| 411 | ProductManifold<MoveableManifold, EuclideanManifold<3>> manifold2{ |
| 412 | std::move(inner2), EuclideanManifold<3>{}}; |
| 413 | |
| 414 | EXPECT_EQ(manifold1.AmbientSize(), manifold2.AmbientSize()); |
| 415 | EXPECT_EQ(manifold1.TangentSize(), manifold2.TangentSize()); |
| 416 | } |
| 417 | |
| 418 | TEST(ProductManifold, CopyableOrMoveable) { |
| 419 | const CopyableManifold inner12{}; |
| 420 | ProductManifold<MoveableManifold, CopyableManifold> manifold1{ |
| 421 | MoveableManifold{}, inner12}; |
| 422 | |
| 423 | MoveableManifold inner21; |
| 424 | CopyableManifold inner22; |
| 425 | ProductManifold<MoveableManifold, CopyableManifold> manifold2{ |
| 426 | std::move(inner21), inner22}; |
| 427 | |
| 428 | EXPECT_EQ(manifold1.AmbientSize(), manifold2.AmbientSize()); |
| 429 | EXPECT_EQ(manifold1.TangentSize(), manifold2.TangentSize()); |
| 430 | } |
| 431 | |
Sergiu Deitsch | 284be88 | 2022-03-03 10:52:16 +0100 | [diff] [blame] | 432 | struct NonDefaultConstructibleManifold : ceres::Manifold { |
| 433 | NonDefaultConstructibleManifold(int, int) {} |
| 434 | int AmbientSize() const override { return 4; } |
| 435 | int TangentSize() const override { return 3; } |
| 436 | |
| 437 | bool Plus(const double* x, |
| 438 | const double* delta, |
| 439 | double* x_plus_delta) const override { |
| 440 | return true; |
| 441 | } |
| 442 | |
| 443 | bool PlusJacobian(const double* x, double* jacobian) const override { |
| 444 | return true; |
| 445 | } |
| 446 | |
| 447 | bool RightMultiplyByPlusJacobian(const double* x, |
| 448 | const int num_rows, |
| 449 | const double* ambient_matrix, |
| 450 | double* tangent_matrix) const override { |
| 451 | return true; |
| 452 | } |
| 453 | |
| 454 | bool Minus(const double* y, |
| 455 | const double* x, |
| 456 | double* y_minus_x) const override { |
| 457 | return true; |
| 458 | } |
| 459 | |
| 460 | bool MinusJacobian(const double* x, double* jacobian) const override { |
| 461 | return true; |
| 462 | } |
| 463 | }; |
| 464 | |
| 465 | TEST(ProductManifold, NonDefaultConstructible) { |
| 466 | ProductManifold<NonDefaultConstructibleManifold, QuaternionManifold> |
| 467 | manifold1{NonDefaultConstructibleManifold{1, 2}, QuaternionManifold{}}; |
| 468 | ProductManifold<QuaternionManifold, NonDefaultConstructibleManifold> |
| 469 | manifold2{QuaternionManifold{}, NonDefaultConstructibleManifold{1, 2}}; |
| 470 | |
| 471 | EXPECT_EQ(manifold1.AmbientSize(), manifold2.AmbientSize()); |
| 472 | EXPECT_EQ(manifold1.TangentSize(), manifold2.TangentSize()); |
| 473 | } |
| 474 | |
| 475 | TEST(ProductManifold, DefaultConstructible) { |
| 476 | ProductManifold<EuclideanManifold<3>, SphereManifold<4>> manifold1; |
| 477 | ProductManifold<SphereManifold<4>, EuclideanManifold<3>> manifold2; |
| 478 | |
| 479 | EXPECT_EQ(manifold1.AmbientSize(), manifold2.AmbientSize()); |
| 480 | EXPECT_EQ(manifold1.TangentSize(), manifold2.TangentSize()); |
| 481 | } |
| 482 | |
Sergiu Deitsch | b0aef21 | 2022-03-03 16:08:34 +0100 | [diff] [blame] | 483 | TEST(ProductManifold, Pointers) { |
| 484 | auto p = std::make_unique<QuaternionManifold>(); |
| 485 | auto q = std::make_shared<EuclideanManifold<3>>(); |
| 486 | |
| 487 | ProductManifold<std::unique_ptr<Manifold>, |
| 488 | EuclideanManifold<3>, |
| 489 | std::shared_ptr<EuclideanManifold<3>>> |
| 490 | manifold1{ |
| 491 | std::make_unique<QuaternionManifold>(), EuclideanManifold<3>{}, q}; |
| 492 | ProductManifold<QuaternionManifold*, |
| 493 | EuclideanManifold<3>, |
| 494 | std::shared_ptr<EuclideanManifold<3>>> |
| 495 | manifold2{p.get(), EuclideanManifold<3>{}, q}; |
| 496 | |
| 497 | EXPECT_EQ(manifold1.AmbientSize(), manifold2.AmbientSize()); |
| 498 | EXPECT_EQ(manifold1.TangentSize(), manifold2.TangentSize()); |
| 499 | } |
| 500 | |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 501 | TEST(QuaternionManifold, PlusPiBy2) { |
| 502 | QuaternionManifold manifold; |
Sameer Agarwal | b81a8bb | 2022-01-04 13:38:18 -0800 | [diff] [blame] | 503 | Vector x = Vector::Zero(4); |
| 504 | x[0] = 1.0; |
| 505 | |
| 506 | for (int i = 0; i < 3; ++i) { |
| 507 | Vector delta = Vector::Zero(3); |
Sergiu Deitsch | 4519b8d | 2023-10-04 22:45:42 +0200 | [diff] [blame] | 508 | delta[i] = constants::pi / 2; |
Sameer Agarwal | b81a8bb | 2022-01-04 13:38:18 -0800 | [diff] [blame] | 509 | Vector x_plus_delta = Vector::Zero(4); |
| 510 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), x_plus_delta.data())); |
| 511 | |
| 512 | // Expect that the element corresponding to pi/2 is +/- 1. All other |
| 513 | // elements should be zero. |
| 514 | for (int j = 0; j < 4; ++j) { |
| 515 | if (i == (j - 1)) { |
| 516 | EXPECT_LT(std::abs(x_plus_delta[j]) - 1, |
| 517 | std::numeric_limits<double>::epsilon()) |
| 518 | << "\ndelta = " << delta.transpose() |
| 519 | << "\nx_plus_delta = " << x_plus_delta.transpose() |
| 520 | << "\n expected the " << j |
| 521 | << "th element of x_plus_delta to be +/- 1."; |
| 522 | } else { |
| 523 | EXPECT_LT(std::abs(x_plus_delta[j]), |
| 524 | std::numeric_limits<double>::epsilon()) |
| 525 | << "\ndelta = " << delta.transpose() |
| 526 | << "\nx_plus_delta = " << x_plus_delta.transpose() |
| 527 | << "\n expected the " << j << "th element of x_plus_delta to be 0."; |
| 528 | } |
| 529 | } |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 530 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD( |
| 531 | manifold, x, delta, x_plus_delta, kTolerance); |
Sameer Agarwal | b81a8bb | 2022-01-04 13:38:18 -0800 | [diff] [blame] | 532 | } |
| 533 | } |
| 534 | |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 535 | // Compute the expected value of QuaternionManifold::Plus via functions in |
| 536 | // rotation.h and compares it to the one computed by QuaternionManifold::Plus. |
| 537 | MATCHER_P2(QuaternionManifoldPlusIsCorrectAt, x, delta, "") { |
Sameer Agarwal | b81a8bb | 2022-01-04 13:38:18 -0800 | [diff] [blame] | 538 | // This multiplication by 2 is needed because AngleAxisToQuaternion uses |
| 539 | // |delta|/2 as the angle of rotation where as in the implementation of |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 540 | // QuaternionManifold for historical reasons we use |delta|. |
Sameer Agarwal | b81a8bb | 2022-01-04 13:38:18 -0800 | [diff] [blame] | 541 | const Vector two_delta = delta * 2; |
| 542 | Vector delta_q(4); |
| 543 | AngleAxisToQuaternion(two_delta.data(), delta_q.data()); |
| 544 | |
| 545 | Vector expected(4); |
| 546 | QuaternionProduct(delta_q.data(), x.data(), expected.data()); |
| 547 | Vector actual(4); |
| 548 | EXPECT_TRUE(arg.Plus(x.data(), delta.data(), actual.data())); |
| 549 | |
| 550 | const double n = (actual - expected).norm(); |
| 551 | const double d = expected.norm(); |
| 552 | const double diffnorm = n / d; |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 553 | if (diffnorm > kTolerance) { |
Sameer Agarwal | b81a8bb | 2022-01-04 13:38:18 -0800 | [diff] [blame] | 554 | *result_listener << "\nx: " << x.transpose() |
| 555 | << "\ndelta: " << delta.transpose() |
| 556 | << "\nexpected: " << expected.transpose() |
| 557 | << "\nactual: " << actual.transpose() |
| 558 | << "\ndiff: " << (expected - actual).transpose() |
| 559 | << "\ndiffnorm : " << diffnorm; |
| 560 | return false; |
| 561 | } |
| 562 | return true; |
| 563 | } |
| 564 | |
Johannes Beck | af5e48c | 2022-01-23 17:57:45 +0100 | [diff] [blame] | 565 | static Vector RandomQuaternion() { |
Sameer Agarwal | b81a8bb | 2022-01-04 13:38:18 -0800 | [diff] [blame] | 566 | Vector x = Vector::Random(4); |
| 567 | x.normalize(); |
| 568 | return x; |
| 569 | } |
| 570 | |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 571 | TEST(QuaternionManifold, GenericDelta) { |
| 572 | QuaternionManifold manifold; |
Sameer Agarwal | b81a8bb | 2022-01-04 13:38:18 -0800 | [diff] [blame] | 573 | for (int trial = 0; trial < kNumTrials; ++trial) { |
| 574 | const Vector x = RandomQuaternion(); |
| 575 | const Vector y = RandomQuaternion(); |
| 576 | Vector delta = Vector::Random(3); |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 577 | EXPECT_THAT(manifold, QuaternionManifoldPlusIsCorrectAt(x, delta)); |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 578 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
Sameer Agarwal | b81a8bb | 2022-01-04 13:38:18 -0800 | [diff] [blame] | 579 | } |
| 580 | } |
| 581 | |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 582 | TEST(QuaternionManifold, SmallDelta) { |
| 583 | QuaternionManifold manifold; |
Sameer Agarwal | b81a8bb | 2022-01-04 13:38:18 -0800 | [diff] [blame] | 584 | for (int trial = 0; trial < kNumTrials; ++trial) { |
| 585 | const Vector x = RandomQuaternion(); |
| 586 | const Vector y = RandomQuaternion(); |
| 587 | Vector delta = Vector::Random(3); |
| 588 | delta.normalize(); |
| 589 | delta *= 1e-6; |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 590 | EXPECT_THAT(manifold, QuaternionManifoldPlusIsCorrectAt(x, delta)); |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 591 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
Sameer Agarwal | b81a8bb | 2022-01-04 13:38:18 -0800 | [diff] [blame] | 592 | } |
| 593 | } |
| 594 | |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 595 | TEST(QuaternionManifold, DeltaJustBelowPi) { |
| 596 | QuaternionManifold manifold; |
Sameer Agarwal | b81a8bb | 2022-01-04 13:38:18 -0800 | [diff] [blame] | 597 | for (int trial = 0; trial < kNumTrials; ++trial) { |
| 598 | const Vector x = RandomQuaternion(); |
| 599 | const Vector y = RandomQuaternion(); |
| 600 | Vector delta = Vector::Random(3); |
| 601 | delta.normalize(); |
Sergiu Deitsch | 4519b8d | 2023-10-04 22:45:42 +0200 | [diff] [blame] | 602 | delta *= (constants::pi - 1e-6); |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 603 | EXPECT_THAT(manifold, QuaternionManifoldPlusIsCorrectAt(x, delta)); |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 604 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
Sameer Agarwal | b81a8bb | 2022-01-04 13:38:18 -0800 | [diff] [blame] | 605 | } |
| 606 | } |
| 607 | |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 608 | // Compute the expected value of EigenQuaternionManifold::Plus using Eigen and |
| 609 | // compares it to the one computed by QuaternionManifold::Plus. |
| 610 | MATCHER_P2(EigenQuaternionManifoldPlusIsCorrectAt, x, delta, "") { |
Sameer Agarwal | 7e2f9d9 | 2022-01-10 17:45:03 -0800 | [diff] [blame] | 611 | // This multiplication by 2 is needed because AngleAxisToQuaternion uses |
| 612 | // |delta|/2 as the angle of rotation where as in the implementation of |
| 613 | // Quaternion for historical reasons we use |delta|. |
| 614 | const Vector two_delta = delta * 2; |
| 615 | Vector delta_q(4); |
| 616 | AngleAxisToQuaternion(two_delta.data(), delta_q.data()); |
| 617 | Eigen::Quaterniond delta_eigen_q( |
| 618 | delta_q[0], delta_q[1], delta_q[2], delta_q[3]); |
| 619 | |
| 620 | Eigen::Map<const Eigen::Quaterniond> x_eigen_q(x.data()); |
| 621 | |
| 622 | Eigen::Quaterniond expected = delta_eigen_q * x_eigen_q; |
| 623 | double actual[4]; |
| 624 | EXPECT_TRUE(arg.Plus(x.data(), delta.data(), actual)); |
| 625 | Eigen::Map<Eigen::Quaterniond> actual_eigen_q(actual); |
| 626 | |
| 627 | const double n = (actual_eigen_q.coeffs() - expected.coeffs()).norm(); |
| 628 | const double d = expected.norm(); |
| 629 | const double diffnorm = n / d; |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 630 | if (diffnorm > kTolerance) { |
Sameer Agarwal | 7e2f9d9 | 2022-01-10 17:45:03 -0800 | [diff] [blame] | 631 | *result_listener |
| 632 | << "\nx: " << x.transpose() << "\ndelta: " << delta.transpose() |
| 633 | << "\nexpected: " << expected.coeffs().transpose() |
| 634 | << "\nactual: " << actual_eigen_q.coeffs().transpose() << "\ndiff: " |
| 635 | << (expected.coeffs() - actual_eigen_q.coeffs()).transpose() |
| 636 | << "\ndiffnorm : " << diffnorm; |
| 637 | return false; |
| 638 | } |
| 639 | return true; |
| 640 | } |
| 641 | |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 642 | TEST(EigenQuaternionManifold, GenericDelta) { |
| 643 | EigenQuaternionManifold manifold; |
Sameer Agarwal | 7e2f9d9 | 2022-01-10 17:45:03 -0800 | [diff] [blame] | 644 | for (int trial = 0; trial < kNumTrials; ++trial) { |
| 645 | const Vector x = RandomQuaternion(); |
| 646 | const Vector y = RandomQuaternion(); |
| 647 | Vector delta = Vector::Random(3); |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 648 | EXPECT_THAT(manifold, EigenQuaternionManifoldPlusIsCorrectAt(x, delta)); |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 649 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
Sameer Agarwal | 7e2f9d9 | 2022-01-10 17:45:03 -0800 | [diff] [blame] | 650 | } |
| 651 | } |
| 652 | |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 653 | TEST(EigenQuaternionManifold, SmallDelta) { |
| 654 | EigenQuaternionManifold manifold; |
Sameer Agarwal | 7e2f9d9 | 2022-01-10 17:45:03 -0800 | [diff] [blame] | 655 | for (int trial = 0; trial < kNumTrials; ++trial) { |
| 656 | const Vector x = RandomQuaternion(); |
| 657 | const Vector y = RandomQuaternion(); |
| 658 | Vector delta = Vector::Random(3); |
| 659 | delta.normalize(); |
| 660 | delta *= 1e-6; |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 661 | EXPECT_THAT(manifold, EigenQuaternionManifoldPlusIsCorrectAt(x, delta)); |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 662 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
Sameer Agarwal | 7e2f9d9 | 2022-01-10 17:45:03 -0800 | [diff] [blame] | 663 | } |
| 664 | } |
| 665 | |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 666 | TEST(EigenQuaternionManifold, DeltaJustBelowPi) { |
| 667 | EigenQuaternionManifold manifold; |
Sameer Agarwal | 7e2f9d9 | 2022-01-10 17:45:03 -0800 | [diff] [blame] | 668 | for (int trial = 0; trial < kNumTrials; ++trial) { |
| 669 | const Vector x = RandomQuaternion(); |
| 670 | const Vector y = RandomQuaternion(); |
| 671 | Vector delta = Vector::Random(3); |
| 672 | delta.normalize(); |
Sergiu Deitsch | 4519b8d | 2023-10-04 22:45:42 +0200 | [diff] [blame] | 673 | delta *= (constants::pi - 1e-6); |
Sameer Agarwal | 19eef54 | 2022-01-22 11:07:52 -0800 | [diff] [blame] | 674 | EXPECT_THAT(manifold, EigenQuaternionManifoldPlusIsCorrectAt(x, delta)); |
Sameer Agarwal | 97d7e07 | 2022-01-19 14:31:00 -0800 | [diff] [blame] | 675 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
Sameer Agarwal | 7e2f9d9 | 2022-01-10 17:45:03 -0800 | [diff] [blame] | 676 | } |
| 677 | } |
| 678 | |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 679 | using Eigen::Vector2d; |
| 680 | using Eigen::Vector3d; |
| 681 | using Vector6d = Eigen::Matrix<double, 6, 1>; |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 682 | using Eigen::Vector4d; |
| 683 | using Vector8d = Eigen::Matrix<double, 8, 1>; |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 684 | |
Johannes Beck | af5e48c | 2022-01-23 17:57:45 +0100 | [diff] [blame] | 685 | TEST(SphereManifold, ZeroTest) { |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 686 | Vector4d x{0.0, 0.0, 0.0, 1.0}; |
| 687 | Vector3d delta = Vector3d::Zero(); |
| 688 | Vector4d y = Vector4d::Zero(); |
Johannes Beck | af5e48c | 2022-01-23 17:57:45 +0100 | [diff] [blame] | 689 | |
| 690 | SphereManifold<4> manifold; |
| 691 | manifold.Plus(x.data(), delta.data(), y.data()); |
| 692 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
| 693 | } |
| 694 | |
| 695 | TEST(SphereManifold, NearZeroTest1) { |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 696 | Vector4d x{1e-5, 1e-5, 1e-5, 1.0}; |
Johannes Beck | af5e48c | 2022-01-23 17:57:45 +0100 | [diff] [blame] | 697 | x.normalize(); |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 698 | Vector3d delta{0.0, 1.0, 0.0}; |
| 699 | Vector4d y = Vector4d::Zero(); |
Johannes Beck | af5e48c | 2022-01-23 17:57:45 +0100 | [diff] [blame] | 700 | |
| 701 | SphereManifold<4> manifold; |
| 702 | manifold.Plus(x.data(), delta.data(), y.data()); |
| 703 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
| 704 | } |
| 705 | |
| 706 | TEST(SphereManifold, NearZeroTest2) { |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 707 | Vector4d x{0.01, 0.0, 0.0, 0.0}; |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 708 | Vector3d delta{0.0, 1.0, 0.0}; |
| 709 | Vector4d y = Vector4d::Zero(); |
Johannes Beck | af5e48c | 2022-01-23 17:57:45 +0100 | [diff] [blame] | 710 | SphereManifold<4> manifold; |
| 711 | manifold.Plus(x.data(), delta.data(), y.data()); |
| 712 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
| 713 | } |
| 714 | |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 715 | TEST(SphereManifold, Plus2DTest) { |
| 716 | Eigen::Vector2d x{0.0, 1.0}; |
| 717 | SphereManifold<2> manifold; |
| 718 | |
| 719 | { |
Sergiu Deitsch | 5ccab18 | 2023-10-05 00:26:46 +0200 | [diff] [blame] | 720 | double delta[1]{constants::pi / 4}; |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 721 | Eigen::Vector2d y = Eigen::Vector2d::Zero(); |
| 722 | EXPECT_TRUE(manifold.Plus(x.data(), delta, y.data())); |
| 723 | const Eigen::Vector2d gtY(std::sqrt(2.0) / 2.0, std::sqrt(2.0) / 2.0); |
| 724 | EXPECT_LT((y - gtY).norm(), kTolerance); |
| 725 | } |
| 726 | |
| 727 | { |
Sergiu Deitsch | 5ccab18 | 2023-10-05 00:26:46 +0200 | [diff] [blame] | 728 | double delta[1]{constants::pi / 2}; |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 729 | Eigen::Vector2d y = Eigen::Vector2d::Zero(); |
| 730 | EXPECT_TRUE(manifold.Plus(x.data(), delta, y.data())); |
| 731 | const Eigen::Vector2d gtY = Eigen::Vector2d::UnitX(); |
| 732 | EXPECT_LT((y - gtY).norm(), kTolerance); |
| 733 | } |
| 734 | |
| 735 | { |
Sergiu Deitsch | 4519b8d | 2023-10-04 22:45:42 +0200 | [diff] [blame] | 736 | double delta[1]{constants::pi}; |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 737 | Eigen::Vector2d y = Eigen::Vector2d::Zero(); |
| 738 | EXPECT_TRUE(manifold.Plus(x.data(), delta, y.data())); |
| 739 | const Eigen::Vector2d gtY = -Eigen::Vector2d::UnitY(); |
| 740 | EXPECT_LT((y - gtY).norm(), kTolerance); |
| 741 | } |
| 742 | |
| 743 | { |
Sergiu Deitsch | 4519b8d | 2023-10-04 22:45:42 +0200 | [diff] [blame] | 744 | double delta[1]{2.0 * constants::pi}; |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 745 | Eigen::Vector2d y = Eigen::Vector2d::Zero(); |
| 746 | EXPECT_TRUE(manifold.Plus(x.data(), delta, y.data())); |
| 747 | const Eigen::Vector2d gtY = Eigen::Vector2d::UnitY(); |
| 748 | EXPECT_LT((y - gtY).norm(), kTolerance); |
| 749 | } |
| 750 | } |
| 751 | |
| 752 | TEST(SphereManifold, Plus3DTest) { |
Johannes Beck | af5e48c | 2022-01-23 17:57:45 +0100 | [diff] [blame] | 753 | Eigen::Vector3d x{0.0, 0.0, 1.0}; |
| 754 | SphereManifold<3> manifold; |
| 755 | |
| 756 | { |
Sergiu Deitsch | 5ccab18 | 2023-10-05 00:26:46 +0200 | [diff] [blame] | 757 | Eigen::Vector2d delta{constants::pi / 2, 0.0}; |
Johannes Beck | af5e48c | 2022-01-23 17:57:45 +0100 | [diff] [blame] | 758 | Eigen::Vector3d y = Eigen::Vector3d::Zero(); |
| 759 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
| 760 | const Eigen::Vector3d gtY = Eigen::Vector3d::UnitX(); |
| 761 | EXPECT_LT((y - gtY).norm(), kTolerance); |
| 762 | } |
| 763 | |
| 764 | { |
Sergiu Deitsch | 4519b8d | 2023-10-04 22:45:42 +0200 | [diff] [blame] | 765 | Eigen::Vector2d delta{constants::pi, 0.0}; |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 766 | Eigen::Vector3d y = Eigen::Vector3d::Zero(); |
| 767 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
| 768 | const Eigen::Vector3d gtY = -Eigen::Vector3d::UnitZ(); |
| 769 | EXPECT_LT((y - gtY).norm(), kTolerance); |
| 770 | } |
| 771 | |
| 772 | { |
Sergiu Deitsch | 4519b8d | 2023-10-04 22:45:42 +0200 | [diff] [blame] | 773 | Eigen::Vector2d delta{2.0 * constants::pi, 0.0}; |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 774 | Eigen::Vector3d y = Eigen::Vector3d::Zero(); |
| 775 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
| 776 | const Eigen::Vector3d gtY = Eigen::Vector3d::UnitZ(); |
| 777 | EXPECT_LT((y - gtY).norm(), kTolerance); |
| 778 | } |
| 779 | |
| 780 | { |
Sergiu Deitsch | 5ccab18 | 2023-10-05 00:26:46 +0200 | [diff] [blame] | 781 | Eigen::Vector2d delta{0.0, constants::pi / 2}; |
Johannes Beck | af5e48c | 2022-01-23 17:57:45 +0100 | [diff] [blame] | 782 | Eigen::Vector3d y = Eigen::Vector3d::Zero(); |
| 783 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
| 784 | const Eigen::Vector3d gtY = Eigen::Vector3d::UnitY(); |
| 785 | EXPECT_LT((y - gtY).norm(), kTolerance); |
| 786 | } |
| 787 | |
| 788 | { |
Sergiu Deitsch | 4519b8d | 2023-10-04 22:45:42 +0200 | [diff] [blame] | 789 | Eigen::Vector2d delta{0.0, constants::pi}; |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 790 | Eigen::Vector3d y = Eigen::Vector3d::Zero(); |
| 791 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
| 792 | const Eigen::Vector3d gtY = -Eigen::Vector3d::UnitZ(); |
| 793 | EXPECT_LT((y - gtY).norm(), kTolerance); |
| 794 | } |
| 795 | |
| 796 | { |
Sergiu Deitsch | 4519b8d | 2023-10-04 22:45:42 +0200 | [diff] [blame] | 797 | Eigen::Vector2d delta{0.0, 2.0 * constants::pi}; |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 798 | Eigen::Vector3d y = Eigen::Vector3d::Zero(); |
| 799 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
| 800 | const Eigen::Vector3d gtY = Eigen::Vector3d::UnitZ(); |
| 801 | EXPECT_LT((y - gtY).norm(), kTolerance); |
| 802 | } |
| 803 | |
| 804 | { |
Sergiu Deitsch | 5ccab18 | 2023-10-05 00:26:46 +0200 | [diff] [blame] | 805 | Eigen::Vector2d delta = |
| 806 | Eigen::Vector2d(1, 1).normalized() * constants::pi / 2; |
Johannes Beck | af5e48c | 2022-01-23 17:57:45 +0100 | [diff] [blame] | 807 | Eigen::Vector3d y = Eigen::Vector3d::Zero(); |
| 808 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
| 809 | const Eigen::Vector3d gtY(std::sqrt(2.0) / 2.0, std::sqrt(2.0) / 2.0, 0.0); |
| 810 | EXPECT_LT((y - gtY).norm(), kTolerance); |
| 811 | } |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 812 | |
| 813 | { |
Sergiu Deitsch | 4519b8d | 2023-10-04 22:45:42 +0200 | [diff] [blame] | 814 | Eigen::Vector2d delta = Eigen::Vector2d(1, 1).normalized() * constants::pi; |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 815 | Eigen::Vector3d y = Eigen::Vector3d::Zero(); |
| 816 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
| 817 | const Eigen::Vector3d gtY = -Eigen::Vector3d::UnitZ(); |
| 818 | EXPECT_LT((y - gtY).norm(), kTolerance); |
| 819 | } |
| 820 | } |
| 821 | |
| 822 | TEST(SphereManifold, Minus2DTest) { |
| 823 | Eigen::Vector2d x{1.0, 0.0}; |
| 824 | SphereManifold<2> manifold; |
| 825 | |
| 826 | { |
| 827 | double delta[1]; |
| 828 | const Eigen::Vector2d y(std::sqrt(2.0) / 2.0, std::sqrt(2.0) / 2.0); |
Sergiu Deitsch | 5ccab18 | 2023-10-05 00:26:46 +0200 | [diff] [blame] | 829 | const double gtDelta{constants::pi / 4}; |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 830 | EXPECT_TRUE(manifold.Minus(y.data(), x.data(), delta)); |
| 831 | EXPECT_LT(std::abs(delta[0] - gtDelta), kTolerance); |
| 832 | } |
| 833 | |
| 834 | { |
| 835 | double delta[1]; |
| 836 | const Eigen::Vector2d y(-1, 0); |
Sergiu Deitsch | 4519b8d | 2023-10-04 22:45:42 +0200 | [diff] [blame] | 837 | const double gtDelta{constants::pi}; |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 838 | EXPECT_TRUE(manifold.Minus(y.data(), x.data(), delta)); |
| 839 | EXPECT_LT(std::abs(delta[0] - gtDelta), kTolerance); |
| 840 | } |
| 841 | } |
| 842 | |
| 843 | TEST(SphereManifold, Minus3DTest) { |
| 844 | Eigen::Vector3d x{1.0, 0.0, 0.0}; |
| 845 | SphereManifold<3> manifold; |
| 846 | |
| 847 | { |
| 848 | Eigen::Vector2d delta; |
| 849 | const Eigen::Vector3d y(std::sqrt(2.0) / 2.0, 0.0, std::sqrt(2.0) / 2.0); |
Sergiu Deitsch | 5ccab18 | 2023-10-05 00:26:46 +0200 | [diff] [blame] | 850 | const Eigen::Vector2d gtDelta(constants::pi / 4, 0.0); |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 851 | EXPECT_TRUE(manifold.Minus(y.data(), x.data(), delta.data())); |
| 852 | EXPECT_LT((delta - gtDelta).norm(), kTolerance); |
| 853 | } |
| 854 | |
| 855 | { |
| 856 | Eigen::Vector2d delta; |
| 857 | const Eigen::Vector3d y(-1, 0, 0); |
Sergiu Deitsch | 4519b8d | 2023-10-04 22:45:42 +0200 | [diff] [blame] | 858 | const Eigen::Vector2d gtDelta(0.0, constants::pi); |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 859 | EXPECT_TRUE(manifold.Minus(y.data(), x.data(), delta.data())); |
| 860 | EXPECT_LT((delta - gtDelta).norm(), kTolerance); |
| 861 | } |
Johannes Beck | af5e48c | 2022-01-23 17:57:45 +0100 | [diff] [blame] | 862 | } |
| 863 | |
| 864 | TEST(SphereManifold, DeathTests) { |
| 865 | EXPECT_DEATH_IF_SUPPORTED(SphereManifold<Eigen::Dynamic> x(1), "size"); |
| 866 | } |
| 867 | |
| 868 | TEST(SphereManifold, NormalFunctionTest) { |
| 869 | SphereManifold<4> manifold; |
| 870 | EXPECT_EQ(manifold.AmbientSize(), 4); |
| 871 | EXPECT_EQ(manifold.TangentSize(), 3); |
| 872 | |
| 873 | Vector zero_tangent = Vector::Zero(manifold.TangentSize()); |
| 874 | for (int trial = 0; trial < kNumTrials; ++trial) { |
| 875 | const Vector x = Vector::Random(manifold.AmbientSize()); |
| 876 | Vector y = Vector::Random(manifold.AmbientSize()); |
| 877 | Vector delta = Vector::Random(manifold.TangentSize()); |
| 878 | |
| 879 | if (x.norm() == 0.0 || y.norm() == 0.0) { |
| 880 | continue; |
| 881 | } |
| 882 | |
| 883 | // X and y need to have the same length. |
| 884 | y *= x.norm() / y.norm(); |
| 885 | |
| 886 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
| 887 | } |
| 888 | } |
| 889 | |
| 890 | TEST(SphereManifold, NormalFunctionTestDynamic) { |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 891 | SphereManifold<ceres::DYNAMIC> manifold(5); |
Johannes Beck | af5e48c | 2022-01-23 17:57:45 +0100 | [diff] [blame] | 892 | EXPECT_EQ(manifold.AmbientSize(), 5); |
| 893 | EXPECT_EQ(manifold.TangentSize(), 4); |
| 894 | |
| 895 | Vector zero_tangent = Vector::Zero(manifold.TangentSize()); |
| 896 | for (int trial = 0; trial < kNumTrials; ++trial) { |
| 897 | const Vector x = Vector::Random(manifold.AmbientSize()); |
| 898 | Vector y = Vector::Random(manifold.AmbientSize()); |
| 899 | Vector delta = Vector::Random(manifold.TangentSize()); |
| 900 | |
| 901 | if (x.norm() == 0.0 || y.norm() == 0.0) { |
| 902 | continue; |
| 903 | } |
| 904 | |
| 905 | // X and y need to have the same length. |
| 906 | y *= x.norm() / y.norm(); |
| 907 | |
| 908 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
| 909 | } |
| 910 | } |
| 911 | |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 912 | TEST(LineManifold, ZeroTest3D) { |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 913 | const Vector6d x = Vector6d::Unit(5); |
| 914 | const Vector4d delta = Vector4d::Zero(); |
| 915 | Vector6d y = Vector6d::Zero(); |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 916 | |
| 917 | LineManifold<3> manifold; |
| 918 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
| 919 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
| 920 | } |
| 921 | |
| 922 | TEST(LineManifold, ZeroTest4D) { |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 923 | const Vector8d x = Vector8d::Unit(7); |
| 924 | const Vector6d delta = Vector6d::Zero(); |
| 925 | Vector8d y = Vector8d::Zero(); |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 926 | |
| 927 | LineManifold<4> manifold; |
| 928 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
| 929 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
| 930 | } |
| 931 | |
| 932 | TEST(LineManifold, ZeroOriginPointTest3D) { |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 933 | const Vector6d x = Vector6d::Unit(5); |
| 934 | Vector4d delta; |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 935 | delta << 0.0, 0.0, 1.0, 2.0; |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 936 | Vector6d y = Vector6d::Zero(); |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 937 | |
| 938 | LineManifold<3> manifold; |
| 939 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
| 940 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
| 941 | } |
| 942 | |
| 943 | TEST(LineManifold, ZeroOriginPointTest4D) { |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 944 | const Vector8d x = Vector8d::Unit(7); |
| 945 | Vector6d delta; |
Julio L. Paneque | f62dccd | 2022-08-04 12:41:10 +0200 | [diff] [blame] | 946 | delta << 0.0, 0.0, 0.0, 0.5, 1.0, 1.5; |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 947 | Vector8d y = Vector8d::Zero(); |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 948 | |
| 949 | LineManifold<4> manifold; |
| 950 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
| 951 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
| 952 | } |
| 953 | |
| 954 | TEST(LineManifold, ZeroDirTest3D) { |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 955 | Vector6d x = Vector6d::Unit(5); |
| 956 | Vector4d delta; |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 957 | delta << 3.0, 2.0, 0.0, 0.0; |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 958 | Vector6d y = Vector6d::Zero(); |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 959 | |
| 960 | LineManifold<3> manifold; |
| 961 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
| 962 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
| 963 | } |
| 964 | |
| 965 | TEST(LineManifold, ZeroDirTest4D) { |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 966 | Vector8d x = Vector8d::Unit(7); |
| 967 | Vector6d delta; |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 968 | delta << 3.0, 2.0, 1.0, 0.0, 0.0, 0.0; |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 969 | Vector8d y = Vector8d::Zero(); |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 970 | |
| 971 | LineManifold<4> manifold; |
| 972 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
| 973 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
| 974 | } |
| 975 | |
| 976 | TEST(LineManifold, Plus) { |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 977 | Vector6d x = Vector6d::Unit(5); |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 978 | LineManifold<3> manifold; |
| 979 | |
| 980 | { |
Sergiu Deitsch | 5ccab18 | 2023-10-05 00:26:46 +0200 | [diff] [blame] | 981 | Vector4d delta{0.0, 2.0, constants::pi / 2, 0.0}; |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 982 | Vector6d y = Vector6d::Random(); |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 983 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 984 | Vector6d gtY; |
| 985 | gtY << 2.0 * Vector3d::UnitY(), Vector3d::UnitX(); |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 986 | EXPECT_LT((y - gtY).norm(), kTolerance); |
| 987 | } |
| 988 | |
| 989 | { |
Sergiu Deitsch | 5ccab18 | 2023-10-05 00:26:46 +0200 | [diff] [blame] | 990 | Vector4d delta{3.0, 0.0, 0.0, constants::pi / 2}; |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 991 | Vector6d y = Vector6d::Zero(); |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 992 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 993 | Vector6d gtY; |
| 994 | gtY << 3.0 * Vector3d::UnitX(), Vector3d::UnitY(); |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 995 | EXPECT_LT((y - gtY).norm(), kTolerance); |
| 996 | } |
| 997 | |
| 998 | { |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 999 | Vector4d delta; |
Sergiu Deitsch | 5ccab18 | 2023-10-05 00:26:46 +0200 | [diff] [blame] | 1000 | delta << Vector2d(1.0, 2.0), |
| 1001 | Vector2d(1, 1).normalized() * constants::pi / 2; |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 1002 | Vector6d y = Vector6d::Zero(); |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 1003 | EXPECT_TRUE(manifold.Plus(x.data(), delta.data(), y.data())); |
Sergiu Deitsch | 6777111 | 2022-02-10 16:21:55 +0100 | [diff] [blame] | 1004 | Vector6d gtY; |
| 1005 | gtY << Vector3d(1.0, 2.0, 0.0), |
| 1006 | Vector3d(std::sqrt(2.0) / 2.0, std::sqrt(2.0) / 2.0, 0.0); |
Johannes Beck | 4742bf3 | 2022-02-09 09:56:15 +0100 | [diff] [blame] | 1007 | EXPECT_LT((y - gtY).norm(), kTolerance); |
| 1008 | } |
| 1009 | } |
| 1010 | |
| 1011 | TEST(LineManifold, NormalFunctionTest) { |
| 1012 | LineManifold<3> manifold; |
| 1013 | EXPECT_EQ(manifold.AmbientSize(), 6); |
| 1014 | EXPECT_EQ(manifold.TangentSize(), 4); |
| 1015 | |
| 1016 | Vector zero_tangent = Vector::Zero(manifold.TangentSize()); |
| 1017 | for (int trial = 0; trial < kNumTrials; ++trial) { |
| 1018 | Vector x = Vector::Random(manifold.AmbientSize()); |
| 1019 | Vector y = Vector::Random(manifold.AmbientSize()); |
| 1020 | Vector delta = Vector::Random(manifold.TangentSize()); |
| 1021 | |
| 1022 | if (x.tail<3>().norm() == 0.0) { |
| 1023 | continue; |
| 1024 | } |
| 1025 | |
| 1026 | x.tail<3>().normalize(); |
| 1027 | manifold.Plus(x.data(), delta.data(), y.data()); |
| 1028 | |
| 1029 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
| 1030 | } |
| 1031 | } |
| 1032 | |
| 1033 | TEST(LineManifold, NormalFunctionTestDynamic) { |
| 1034 | LineManifold<ceres::DYNAMIC> manifold(3); |
| 1035 | EXPECT_EQ(manifold.AmbientSize(), 6); |
| 1036 | EXPECT_EQ(manifold.TangentSize(), 4); |
| 1037 | |
| 1038 | Vector zero_tangent = Vector::Zero(manifold.TangentSize()); |
| 1039 | for (int trial = 0; trial < kNumTrials; ++trial) { |
| 1040 | Vector x = Vector::Random(manifold.AmbientSize()); |
| 1041 | Vector y = Vector::Random(manifold.AmbientSize()); |
| 1042 | Vector delta = Vector::Random(manifold.TangentSize()); |
| 1043 | |
| 1044 | if (x.tail<3>().norm() == 0.0) { |
| 1045 | continue; |
| 1046 | } |
| 1047 | |
| 1048 | x.tail<3>().normalize(); |
| 1049 | manifold.Plus(x.data(), delta.data(), y.data()); |
| 1050 | |
| 1051 | EXPECT_THAT_MANIFOLD_INVARIANTS_HOLD(manifold, x, delta, y, kTolerance); |
| 1052 | } |
| 1053 | } |
| 1054 | |
Sameer Agarwal | caf614a | 2022-04-21 17:41:10 -0700 | [diff] [blame] | 1055 | } // namespace ceres::internal |