testOrientedPlane3Factor.cpp
Go to the documentation of this file.
1 /* ----------------------------------------------------------------------------
2 
3  * GTSAM Copyright 2010, Georgia Tech Research Corporation,
4  * Atlanta, Georgia 30332-0415
5  * All Rights Reserved
6  * Authors: Frank Dellaert, et al. (see THANKS for the full author list)
7 
8  * See LICENSE for the license information
9 
10  * -------------------------------------------------------------------------- */
11 
12 /*
13  * @file testOrientedPlane3Factor.cpp
14  * @date Dec 19, 2012
15  * @author Alex Trevor
16  * @brief Tests the OrientedPlane3Factor class
17  */
18 
20 
22 #include <gtsam/inference/Symbol.h>
24 #include <gtsam/nonlinear/ISAM2.h>
25 
27 
28 using namespace std::placeholders;
29 using namespace gtsam;
30 using namespace std;
31 
34 
35 using symbol_shorthand::P; //< Planes
36 using symbol_shorthand::X; //< Pose3
37 
38 // *************************************************************************
39 TEST(OrientedPlane3Factor, lm_translation_error) {
40  // Tests one pose, two measurements of the landmark that differ in range only.
41  // Normal along -x, 3m away
42  OrientedPlane3 test_lm0(-1.0, 0.0, 0.0, 3.0);
43 
45 
46  // Init pose and prior. Pose Prior is needed since a single plane measurement
47  // does not fully constrain the pose
48  Pose3 init_pose(Rot3::Ypr(0.0, 0.0, 0.0), Point3(0.0, 0.0, 0.0));
49  Vector6 sigmas;
50  sigmas << 0.001, 0.001, 0.001, 0.001, 0.001, 0.001;
51  graph.addPrior(X(0), init_pose, noiseModel::Diagonal::Sigmas(sigmas));
52 
53  // Add two landmark measurements, differing in range
54  Vector3 sigmas3 {0.1, 0.1, 0.1};
55  Vector4 measurement0 {-1.0, 0.0, 0.0, 3.0};
56  OrientedPlane3Factor factor0(
57  measurement0, noiseModel::Diagonal::Sigmas(sigmas3), X(0), P(0));
58  graph.add(factor0);
59  Vector4 measurement1 {-1.0, 0.0, 0.0, 1.0};
61  measurement1, noiseModel::Diagonal::Sigmas(sigmas3), X(0), P(0));
62  graph.add(factor1);
63 
64  // Initial Estimate
65  Values values;
66  values.insert(X(0), init_pose);
67  values.insert(P(0), test_lm0);
68 
69  // Optimize
70  ISAM2 isam2;
72  Values result_values = isam2.calculateEstimate();
73  OrientedPlane3 optimized_plane_landmark =
74  result_values.at<OrientedPlane3>(P(0));
75 
76  // Given two noisy measurements of equal weight, expect result between the two
77  OrientedPlane3 expected_plane_landmark(-1.0, 0.0, 0.0, 2.0);
78  EXPECT(assert_equal(optimized_plane_landmark, expected_plane_landmark));
79 }
80 
81 // *************************************************************************
82 // TODO As described in PR #564 after correcting the derivatives in
83 // OrientedPlane3Factor this test fails. It should be debugged and re-enabled.
84 /*
85 TEST (OrientedPlane3Factor, lm_rotation_error) {
86  // Tests one pose, two measurements of the landmark that differ in angle only.
87  // Normal along -x, 3m away
88  OrientedPlane3 test_lm0(-1.0/sqrt(1.01), 0.1/sqrt(1.01), 0.0, 3.0);
89 
90  NonlinearFactorGraph graph;
91 
92  // Init pose and prior. Pose Prior is needed since a single plane measurement does not fully constrain the pose
93  Pose3 init_pose(Rot3::Ypr(0.0, 0.0, 0.0), Point3(0.0, 0.0, 0.0));
94  graph.addPrior(X(0), init_pose,
95  noiseModel::Diagonal::Sigmas(
96  (Vector(6) << 0.001, 0.001, 0.001, 0.001, 0.001, 0.001).finished()));
97 
98  // Add two landmark measurements, differing in angle
99  Vector4 measurement0 {-1.0, 0.0, 0.0, 3.0};
100  OrientedPlane3Factor factor0(measurement0,
101  noiseModel::Diagonal::Sigmas(Vector3(0.1, 0.1, 0.1)), X(0), P(0));
102  graph.add(factor0);
103  Vector4 measurement1 {0.0, -1.0, 0.0, 3.0};
104  OrientedPlane3Factor factor1(measurement1,
105  noiseModel::Diagonal::Sigmas(Vector3(0.1, 0.1, 0.1)), X(0), P(0));
106  graph.add(factor1);
107 
108  // Initial Estimate
109  Values values;
110  values.insert(X(0), init_pose);
111  values.insert(P(0), test_lm0);
112 
113  // Optimize
114  ISAM2 isam2;
115  ISAM2Result result = isam2.update(graph, values);
116  Values result_values = isam2.calculateEstimate();
117  auto optimized_plane_landmark = result_values.at<OrientedPlane3>(P(0));
118 
119  // Given two noisy measurements of equal weight, expect result between the two
120  OrientedPlane3 expected_plane_landmark(-sqrt(2.0) / 2.0, -sqrt(2.0) / 2.0,
121  0.0, 3.0);
122  EXPECT(assert_equal(optimized_plane_landmark, expected_plane_landmark));
123 }
124 */
125 
126 // *************************************************************************
127 TEST( OrientedPlane3Factor, Derivatives ) {
128  // Measurement
129  OrientedPlane3 p(sqrt(2)/2, -sqrt(2)/2, 0, 5);
130 
131  // Linearisation point
132  OrientedPlane3 pLin(sqrt(3)/3, -sqrt(3)/3, sqrt(3)/3, 7);
133  gtsam::Point3 pointLin = gtsam::Point3(1, 2, -4);
134  gtsam::Rot3 rotationLin = gtsam::Rot3::RzRyRx(0.5*M_PI, -0.3*M_PI, 1.4*M_PI);
135  Pose3 poseLin(rotationLin, pointLin);
136 
137  // Factor
138  Key planeKey(1), poseKey(2);
139  SharedGaussian noise = noiseModel::Diagonal::Sigmas(Vector3(0.1, 0.1, 0.1));
140  OrientedPlane3Factor factor(p.planeCoefficients(), noise, poseKey, planeKey);
141 
142  // Calculate numerical derivatives
143  auto f = [&factor](const Pose3& p, const OrientedPlane3& o) {
144  return factor.evaluateError(p, o);
145  };
146  Matrix numericalH1 = numericalDerivative21<Vector, Pose3, OrientedPlane3>(f, poseLin, pLin);
147  Matrix numericalH2 = numericalDerivative22<Vector, Pose3, OrientedPlane3>(f, poseLin, pLin);
148 
149  // Use the factor to calculate the derivative
150  Matrix actualH1, actualH2;
151  factor.evaluateError(poseLin, pLin, actualH1, actualH2);
152 
153  // Verify we get the expected error
154  EXPECT(assert_equal(numericalH1, actualH1, 1e-8));
155  EXPECT(assert_equal(numericalH2, actualH2, 1e-8));
156 }
157 
158 // *************************************************************************
160 
161  // Example: pitch and roll of aircraft in an ENU Cartesian frame.
162  // If pitch and roll are zero for an aerospace frame,
163  // that means Z is pointing down, i.e., direction of Z = (0,0,-1)
164 
165  Vector4 planeOrientation = (Vector(4) << 0.0, 0.0, -1.0, 10.0).finished(); // all vertical planes directly facing the origin
166 
167  // Factor
168  Key key(1);
169  SharedGaussian model = noiseModel::Diagonal::Sigmas(Vector3(0.1, 0.1, 10.0));
170  OrientedPlane3DirectionPrior factor(key, planeOrientation, model);
171 
172  // Create a linearization point at the zero-error point
173  Vector4 theta1 {0.0, 0.02, -1.2, 10.0};
174  Vector4 theta2 {0.0, 0.1, -0.8, 10.0};
175  Vector4 theta3 {0.0, 0.2, -0.9, 10.0};
176 
177  OrientedPlane3 T1(theta1);
178  OrientedPlane3 T2(theta2);
179  OrientedPlane3 T3(theta3);
180 
181  // Calculate numerical derivatives
182  Matrix expectedH1 = numericalDerivative11<Vector, OrientedPlane3>(
183  [&factor](const OrientedPlane3& o) {return factor.evaluateError(o);}, T1);
184 
185  Matrix expectedH2 = numericalDerivative11<Vector, OrientedPlane3>(
186  [&factor](const OrientedPlane3& o) {return factor.evaluateError(o);}, T2);
187 
188  Matrix expectedH3 = numericalDerivative11<Vector, OrientedPlane3>(
189  [&factor](const OrientedPlane3& o) { return factor.evaluateError(o); }, T3);
190 
191  // Use the factor to calculate the derivative
192  Matrix actualH1, actualH2, actualH3;
193  factor.evaluateError(T1, actualH1);
194  factor.evaluateError(T2, actualH2);
195  factor.evaluateError(T3, actualH3);
196 
197  // Verify we get the expected error
198  EXPECT(assert_equal(expectedH1, actualH1, 1e-8));
199  EXPECT(assert_equal(expectedH2, actualH2, 1e-8));
200  EXPECT(assert_equal(expectedH3, actualH3, 1e-8));
201 }
202 
203 /* ************************************************************************* */
204 // Simplified version of the test by Marco Camurri to debug issue #561
205 TEST(OrientedPlane3Factor, Issue561Simplified) {
206  // Typedefs
207  using Plane = OrientedPlane3;
208 
210 
211  // Setup prior factors
212  // Note: If x0 is too far away from the origin (e.g. x=100) this test can fail.
213  Pose3 x0(Rot3::Identity(), Vector3(10, -1, 1));
214  auto x0_noise = noiseModel::Isotropic::Sigma(6, 0.01);
215  graph.addPrior<Pose3>(X(0), x0, x0_noise);
216 
217  // Two horizontal planes with different heights, in the world frame.
218  const Plane p1(0,0,1,1), p2(0,0,1,2);
219  auto p1_noise = noiseModel::Diagonal::Sigmas(Vector3{1, 1, 5});
220  auto p2_noise = noiseModel::Diagonal::Sigmas(Vector3{1, 1, 5});
221  graph.addPrior<Plane>(P(1), p1, p1_noise);
222  graph.addPrior<Plane>(P(2), p2, p2_noise);
223 
224  // Plane factors
225  auto p1_measured_from_x0 = p1.transform(x0); // transform p1 to pose x0 as a measurement
226  auto p2_measured_from_x0 = p2.transform(x0); // transform p2 to pose x0 as a measurement
227  const auto x0_p1_noise = noiseModel::Isotropic::Sigma(3, 0.05);
228  const auto x0_p2_noise = noiseModel::Isotropic::Sigma(3, 0.05);
230  p1_measured_from_x0.planeCoefficients(), x0_p1_noise, X(0), P(1));
232  p2_measured_from_x0.planeCoefficients(), x0_p2_noise, X(0), P(2));
233 
234  // Initial values
235  // Just offset the initial pose by 1m. This is what we are trying to optimize.
236  Values initialEstimate;
237  Pose3 x0_initial = x0.compose(Pose3(Rot3::Identity(), Vector3(1,0,0)));
238  initialEstimate.insert(P(1), p1);
239  initialEstimate.insert(P(2), p2);
240  initialEstimate.insert(X(0), x0_initial);
241 
242  // Optimize
243  try {
244  GaussNewtonOptimizer optimizer(graph, initialEstimate);
245  Values result = optimizer.optimize();
247  EXPECT(x0.equals(result.at<Pose3>(X(0))));
248  EXPECT(p1.equals(result.at<Plane>(P(1))));
249  EXPECT(p2.equals(result.at<Plane>(P(2))));
250  } catch (const IndeterminantLinearSystemException &e) {
251  std::cerr << "CAPTURED THE EXCEPTION: " << DefaultKeyFormatter(e.nearbyVariable()) << std::endl;
252  EXPECT(false); // fail if this happens
253  }
254 }
255 
256 /* ************************************************************************* */
257 int main() {
258  srand(time(nullptr));
259  TestResult tr;
260  return TestRegistry::runAllTests(tr);
261 }
262 /* ************************************************************************* */
TestRegistry::runAllTests
static int runAllTests(TestResult &result)
Definition: TestRegistry.cpp:27
gtsam::NonlinearOptimizer::optimize
virtual const Values & optimize()
Definition: NonlinearOptimizer.h:98
gtsam::ISAM2
Definition: ISAM2.h:45
gtsam::GaussNewtonOptimizer
Definition: GaussNewtonOptimizer.h:38
e
Array< double, 1, 3 > e(1./3., 0.5, 2.)
GTSAM_CONCEPT_TESTABLE_INST
#define GTSAM_CONCEPT_TESTABLE_INST(T)
Definition: Testable.h:176
EXPECT
#define EXPECT(condition)
Definition: Test.h:150
test_constructor::sigmas
Vector1 sigmas
Definition: testHybridNonlinearFactor.cpp:52
TestHarness.h
measurement1
static Point2 measurement1(323.0, 240.0)
gtsam::Symbol::equals
bool equals(const Symbol &expected, double tol=0.0) const
Check equality.
Definition: Symbol.cpp:54
T3
static const Pose3 T3(Rot3::Rodrigues(-90, 0, 0), Point3(1, 2, 3))
gtsam::Matrix
Eigen::MatrixXd Matrix
Definition: base/Matrix.h:39
different_sigmas::values
HybridValues values
Definition: testHybridBayesNet.cpp:245
X
#define X
Definition: icosphere.cpp:20
gtsam::Vector3
Eigen::Vector3d Vector3
Definition: Vector.h:44
gtsam::OrientedPlane3DirectionPrior
Definition: OrientedPlane3Factor.h:55
result
Values result
Definition: OdometryOptimize.cpp:8
gtsam::OrientedPlane3
Represents an infinite plane in 3D, which is composed of a planar normal and its perpendicular distan...
Definition: OrientedPlane3.h:36
poseKey
const gtsam::Key poseKey
Definition: testPoseRotationPrior.cpp:29
gtsam::DefaultKeyFormatter
KeyFormatter DefaultKeyFormatter
Assign default key formatter.
Definition: Key.cpp:30
pruning_fixture::factor
DecisionTreeFactor factor(D &C &B &A, "0.0 0.0 0.0 0.60658897 0.61241912 0.61241969 0.61247685 0.61247742 0.0 " "0.0 0.0 0.99995287 1.0 1.0 1.0 1.0")
GaussNewtonOptimizer.h
simple::p2
static Point3 p2
Definition: testInitializePose3.cpp:51
gtsam::NonlinearFactorGraph::error
double error(const Values &values) const
Definition: NonlinearFactorGraph.cpp:170
gtsam::Values::at
const ValueType at(Key j) const
Definition: Values-inl.h:261
numericalDerivative.h
Some functions to compute numerical derivatives.
TEST
TEST(OrientedPlane3Factor, lm_translation_error)
Definition: testOrientedPlane3Factor.cpp:39
gtsam::Rot3
Rot3 is a 3D rotation represented as a rotation matrix if the preprocessor symbol GTSAM_USE_QUATERNIO...
Definition: Rot3.h:58
gtsam::Pose3
Definition: Pose3.h:37
gtsam::NonlinearFactorGraph
Definition: NonlinearFactorGraph.h:55
gtsam::NonlinearFactorGraph::addPrior
void addPrior(Key key, const T &prior, const SharedNoiseModel &model=nullptr)
Definition: NonlinearFactorGraph.h:199
x0
static Symbol x0('x', 0)
Symbol.h
gtsam::SharedGaussian
noiseModel::Gaussian::shared_ptr SharedGaussian
Definition: NoiseModel.h:763
gtsam::HybridValues::insert
void insert(Key j, const Vector &value)
Definition: HybridValues.cpp:85
T2
static const Pose3 T2(Rot3::Rodrigues(0.3, 0.2, 0.1), P2)
time
#define time
Definition: timeAdaptAutoDiff.cpp:31
model
noiseModel::Diagonal::shared_ptr model
Definition: doc/Code/Pose2SLAMExample.cpp:7
EXPECT_DOUBLES_EQUAL
#define EXPECT_DOUBLES_EQUAL(expected, actual, threshold)
Definition: Test.h:161
p1
Vector3f p1
Definition: MatrixBase_all.cpp:2
TestResult
Definition: TestResult.h:26
OrientedPlane3Factor.h
key
const gtsam::Symbol key('X', 0)
gtsam::ISAM2::calculateEstimate
Values calculateEstimate() const
Definition: ISAM2.cpp:786
tree::f
Point2(* f)(const Point3 &, OptionalJacobian< 2, 3 >)
Definition: testExpression.cpp:218
ISAM2.h
Incremental update functionality (ISAM2) for BayesTree, with fluid relinearization.
gtsam
traits
Definition: SFMdata.h:40
gtsam::ISAM2::update
virtual ISAM2Result update(const NonlinearFactorGraph &newFactors=NonlinearFactorGraph(), const Values &newTheta=Values(), const FactorIndices &removeFactorIndices=FactorIndices(), const std::optional< FastMap< Key, int > > &constrainedKeys={}, const std::optional< FastList< Key > > &noRelinKeys={}, const std::optional< FastList< Key > > &extraReelimKeys={}, bool force_relinearize=false)
Definition: ISAM2.cpp:401
gtsam::Values
Definition: Values.h:65
std
Definition: BFloat16.h:88
p
float * p
Definition: Tutorial_Map_using.cpp:9
gtsam::assert_equal
bool assert_equal(const Matrix &expected, const Matrix &actual, double tol)
Definition: Matrix.cpp:41
gtsam::OrientedPlane3Factor
Definition: OrientedPlane3Factor.h:18
P
static double P[]
Definition: ellpe.c:68
GTSAM_CONCEPT_MANIFOLD_INST
#define GTSAM_CONCEPT_MANIFOLD_INST(T)
‍**
Definition: Manifold.h:177
gtsam::Point3
Vector3 Point3
Definition: Point3.h:38
gtsam::ISAM2Result
Definition: ISAM2Result.h:39
gtsam::IndeterminantLinearSystemException
Definition: linearExceptions.h:94
M_PI
#define M_PI
Definition: mconf.h:117
T1
static const Similarity3 T1(R, Point3(3.5, -8.2, 4.2), 1)
gtsam::FactorGraph::add
IsDerived< DERIVEDFACTOR > add(std::shared_ptr< DERIVEDFACTOR > factor)
add is a synonym for push_back.
Definition: FactorGraph.h:171
graph
NonlinearFactorGraph graph
Definition: doc/Code/OdometryExample.cpp:2
gtsam::Key
std::uint64_t Key
Integer nonlinear key type.
Definition: types.h:97
gtsam::Rot3::RzRyRx
static Rot3 RzRyRx(double x, double y, double z, OptionalJacobian< 3, 1 > Hx={}, OptionalJacobian< 3, 1 > Hy={}, OptionalJacobian< 3, 1 > Hz={})
Rotations around Z, Y, then X axes as in http://en.wikipedia.org/wiki/Rotation_matrix,...
Definition: Rot3M.cpp:84
main
int main()
Definition: testOrientedPlane3Factor.cpp:257
ceres::Vector
Eigen::Matrix< double, Eigen::Dynamic, 1 > Vector
Definition: gtsam/3rdparty/ceres/eigen.h:38
ceres::sqrt
Jet< T, N > sqrt(const Jet< T, N > &f)
Definition: jet.h:418
simple_graph::factor1
auto factor1
Definition: testJacobianFactor.cpp:196
gtsam::FactorGraph::emplace_shared
IsDerived< DERIVEDFACTOR > emplace_shared(Args &&... args)
Emplace a shared pointer to factor of given type.
Definition: FactorGraph.h:153


gtsam
Author(s):
autogenerated on Sat Jan 4 2025 04:06:34