Perturb camera center instead of translation vector.
Change-Id: I636c84566be2596dd7b4eeab0bb511de597be111
diff --git a/examples/bal_problem.cc b/examples/bal_problem.cc
index 2bca792..a705bd2 100644
--- a/examples/bal_problem.cc
+++ b/examples/bal_problem.cc
@@ -36,6 +36,7 @@
#include <glog/logging.h>
#include "ceres/random.h"
#include "ceres/rotation.h"
+#include "Eigen/Core"
namespace ceres {
namespace examples {
@@ -128,6 +129,35 @@
for (int i = 0; i < num_cameras_; ++i) {
double* camera = mutable_cameras() + camera_block_size() * i;
+ // Perturb the location of the camera rather than the translation
+ // vector. This is makes the perturbation physically more sensible.
+ if (translation_sigma > 0.0) {
+ double center[3];
+ Eigen::VectorXd angle_axis(3);
+
+ if (use_quaternions_) {
+ angle_axis = Eigen::Map<Eigen::VectorXd>(camera, 3);
+ } else {
+ QuaternionToAngleAxis(camera, angle_axis.data());
+ }
+ angle_axis *= -1.0;
+
+ // Camera center is c = -R't, the negative sign does not matter.
+ AngleAxisRotatePoint(angle_axis.data(),
+ camera + camera_block_size() - 6,
+ center);
+
+ // Perturb center.
+ for (int j = 0; j < 3; ++j) {
+ center[j] += translation_sigma * RandNormal();
+ }
+
+ // t = -R * (- R' t + perturbation)
+ AngleAxisRotatePoint(angle_axis.data(),
+ center,
+ camera + camera_block_size() - 6);
+ }
+
// First three coordinates of the camera rotation are shared
// between the angle-axis and the quaternion representations.
if (rotation_sigma > 0.0) {
@@ -137,25 +167,10 @@
if (use_quaternions_) {
camera[3] += rotation_sigma * RandNormal();
-
- // Normalize the quaternion.
- double norm = 0.0;
- for (int j = 0; j < 4; ++j) {
- norm += camera[j] * camera[j];
- }
- norm = sqrt(norm);
- for (int j = 0; j < 4; ++j) {
- camera[j] /= norm;
- }
+ Eigen::Map<Eigen::VectorXd>(camera, 4).normalize();
}
}
- if (translation_sigma > 0.0) {
- // Translation.
- camera[camera_block_size() - 6] += translation_sigma * RandNormal();
- camera[camera_block_size() - 5] += translation_sigma * RandNormal();
- camera[camera_block_size() - 4] += translation_sigma * RandNormal();
- }
}
}