109 int max_image = tracks.MaxImage();
110 int max_track = tracks.MaxTrack();
112 LG <<
"Max image: " << max_image;
113 LG <<
"Max track: " << max_track;
116 Vec3 zero_rotation = Vec3::Zero();
118 ceres::AngleAxisToQuaternion(&zero_rotation(0), &quaternion(0));
120 for (
int image = 0; image <= max_image; ++
image) {
123 ModalSolverLogProgress(update_callback, (
float)image / max_image);
126 if (all_markers.size() == 0) {
127 LG <<
"Skipping image: " <<
image;
133 ceres::QuaternionToRotation(&quaternion(0), ¤t_R(0, 0));
139 for (
int i = 0; i < all_markers.size(); ++i) {
140 Marker& marker = all_markers[i];
144 ProjectMarkerOnSphere(marker,
X);
146 int last_column = x1.cols();
147 x1.conservativeResize(3, last_column + 1);
148 x2.conservativeResize(3, last_column + 1);
150 x1.col(last_column) = current_R * point->X;
151 x2.col(last_column) =
X;
155 if (x1.cols() >= 2) {
165 Vec3 delta_angle_axis;
166 ceres::RotationMatrixToAngleAxis(&delta_R(0, 0), &delta_angle_axis(0));
168 Vec3 current_angle_axis;
169 ceres::QuaternionToAngleAxis(&quaternion(0), ¤t_angle_axis(0));
171 Vec3 angle_axis = current_angle_axis + delta_angle_axis;
173 ceres::AngleAxisToQuaternion(&angle_axis(0), &quaternion(0));
175 LG <<
"Analytically computed quaternion " << quaternion.transpose();
179 ceres::Problem problem;
183 ceres::Manifold* quaternion_manifold =
NULL;
185 int num_residuals = 0;
186 for (
int i = 0; i < all_markers.size(); ++i) {
187 Marker& marker = all_markers[i];
190 if (point && marker.
weight != 0.0) {
191 problem.AddResidualBlock(
192 new ceres::AutoDiffCostFunction<ModalReprojectionError,
194 4>(
new ModalReprojectionError(
195 marker.
x, marker.
y, marker.
weight, point->X)),
200 if (quaternion_manifold ==
NULL) {
201 quaternion_manifold =
new ceres::QuaternionManifold();
204 problem.SetManifold(&quaternion(0), quaternion_manifold);
208 LG <<
"Number of residuals: " << num_residuals;
212 ceres::Solver::Options solver_options;
213 solver_options.linear_solver_type = ceres::DENSE_QR;
214 solver_options.max_num_iterations = 50;
215 solver_options.update_state_every_iteration =
true;
216 solver_options.gradient_tolerance = 1
e-36;
217 solver_options.parameter_tolerance = 1
e-36;
218 solver_options.function_tolerance = 1
e-36;
221 ceres::Solver::Summary summary;
222 ceres::Solve(solver_options, &problem, &summary);
224 LG <<
"Summary:\n" << summary.FullReport();
225 LG <<
"Refined quaternion " << quaternion.transpose();
230 ceres::QuaternionToRotation(&quaternion(0), &
R(0, 0));
237 for (
int track = 0; track <= max_track; ++track) {
239 Marker marker = tracks.MarkerInImageForTrack(image, track);
241 if (marker.
image == image) {
245 LG <<
"Projecting track " << track <<
" at image " <<
image;
248 ProjectMarkerOnSphere(marker,
X);
input_tx image(0, GPU_RGBA16F, Qualifier::WRITE, ImageType::FLOAT_2D, "preview_img") .compute_source("compositor_compute_preview.glsl") .do_static_compilation(true)