apex-solver 1.3.0

High-performance nonlinear least squares optimization with Lie group support for SLAM and bundle adjustment
Documentation

๐Ÿฆ€ Apex Solver

A high-performance Rust-based nonlinear least squares optimization library designed for computer vision applications including bundle adjustment, SLAM, and pose graph optimization. Built with focus on zero-cost abstractions, memory safety, and mathematical correctness.

Apex Solver is a comprehensive optimization library that bridges the gap between theoretical robotics and practical implementation. It provides manifold-aware optimization for Lie groups commonly used in computer vision, multiple optimization algorithms with unified interfaces, flexible linear algebra backends supporting both sparse Cholesky and QR decompositions, and industry-standard file format support for seamless integration with existing workflows.

Crates.io Documentation License

Key Features (v1.3.0)

  • ๐Ÿ“ท Bundle Adjustment with Camera Intrinsic Optimization: Simultaneous optimization of camera poses, 3D landmarks, and camera intrinsics (8 camera models via apex-camera-models crate) apex-camera-models
  • ๐Ÿ”ง Explicit & Implicit Schur Complement Solvers: Memory-efficient matrix-free PCG for large-scale problems (10,000+ cameras) alongside traditional explicit formulation
  • ๐Ÿ›ก๏ธ 15 Robust Loss Functions: Comprehensive outlier rejection (Huber, Cauchy, Tukey, Welsch, Barron, and more)
  • ๐Ÿ“ Manifold-Aware: Full Lie group support (SE2, SE3, SO2, SO3, SE_2(3), SGal(3), Sim(3), Rn) with analytic Jacobians apex-manifolds
  • ๐Ÿš€ Three Optimization Algorithms: Levenberg-Marquardt, Gauss-Newton, and Dog Leg with unified interface
  • ๐Ÿ“Œ Prior Factors & Fixed Variables: Anchor poses with known values and constrain specific parameter indices
  • ๐Ÿ“Š Uncertainty Quantification: Covariance estimation for both Cholesky and QR solvers
  • ๐ŸŽจ Real-time Visualization: Integrated Rerun support for live debugging of optimization progress
  • ๐Ÿ“ I/O: Read and write G2O, Toro, BAL format files for seamless integration with SLAM ecosystems apex-io
  • โšก High Performance: Sparse linear algebra with persistent symbolic factorization
  • โœ… Production-Grade: Comprehensive error handling, structured tracing, integration test suite

๐Ÿš€ Quick Start

use std::collections::HashMap;
use apex_solver::core::problem::Problem;
use apex_solver::factors::{BetweenFactor, PriorFactor};
use apex_solver::{G2oLoader, LinearSolverType, ManifoldType};
use apex_solver::optimizer::levenberg_marquardt::{LevenbergMarquardt, LevenbergMarquardtConfig};
use nalgebra::dvector;

fn main() -> Result<(), Box<dyn std::error::Error>> {
    // Load pose graph from G2O file
    let graph = G2oLoader::load("data/odometry/sphere2500.g2o")?;
    
    // Create optimization problem
    let mut problem = Problem::new();
    let mut initial_values = HashMap::new();
    
    // Add SE3 poses as variables
    for (&id, vertex) in &graph.vertices_se3 {
        let var_name = format!("x{}", id);
        let quat = vertex.pose.rotation_quaternion();
        let trans = vertex.pose.translation();
        let se3_data = dvector![trans.x, trans.y, trans.z, quat.w, quat.i, quat.j, quat.k];
        initial_values.insert(var_name, (ManifoldType::SE3, se3_data));
    }
    
    // Add between factors (relative pose constraints)
    for edge in &graph.edges_se3 {
        let factor = BetweenFactor::new(edge.measurement.clone());
        problem.add_residual_block(
            &[&format!("x{}", edge.from), &format!("x{}", edge.to)],
            Box::new(factor),
            None,  // Optional: add HuberLoss for robustness
        );
    }
    
    // Configure and run optimizer
    let config = LevenbergMarquardtConfig::new()
        .with_linear_solver_type(LinearSolverType::SparseCholesky)
        .with_max_iterations(100)
        .with_cost_tolerance(1e-6)
        .with_compute_covariances(true);  // Enable uncertainty estimation
    
    let mut solver = LevenbergMarquardt::with_config(config);
    let result = solver.optimize(&problem, &initial_values)?;
    
    println!("Status: {:?}", result.status);
    println!("Initial cost: {:.3e}", result.initial_cost);
    println!("Final cost: {:.3e}", result.final_cost);
    println!("Iterations: {}", result.iterations);
    
    Ok(())
}

Result:

Status: CostToleranceReached
Initial cost: 1.280e+05
Final cost: 2.130e+01
Iterations: 5

๐Ÿ—๏ธ Architecture

The workspace root is the apex-solver crate. Sub-crates for manifolds, I/O, and camera models live in crates/:

apex-solver/                # workspace root = apex-solver crate
โ”œโ”€โ”€ src/
โ”‚   โ”œโ”€โ”€ core/               # Problem formulation, factors, residuals
โ”‚   โ”œโ”€โ”€ factors/            # Factor implementations (projection, between, prior)
โ”‚   โ”œโ”€โ”€ optimizer/          # LM, GN, Dog Leg algorithms
โ”‚   โ”œโ”€โ”€ linalg/             # Cholesky, QR, Explicit/Implicit Schur
โ”‚   โ””โ”€โ”€ observers/          # Optimization observers and callbacks
โ”œโ”€โ”€ bin/                    # Executable binaries
โ”œโ”€โ”€ benches/                # Benchmarks
โ”œโ”€โ”€ examples/               # Example programs
โ”œโ”€โ”€ tests/                  # Integration tests
โ”œโ”€โ”€ doc/                    # Extended documentation
โ””โ”€โ”€ crates/
    โ”œโ”€โ”€ apex-manifolds/     # Lie groups: SE2, SE3, SO2, SO3, SE_2(3), SGal(3), Sim(3), Rn
    โ”œโ”€โ”€ apex-io/            # File I/O: G2O, TORO, BAL formats
    โ””โ”€โ”€ apex-camera-models/ # 8 camera projection models

Core Modules (in src/):

  • core/: Optimization problem definitions, residual blocks, robust loss functions, and variable management
  • optimizer/: Three optimization algorithms (Levenberg-Marquardt with adaptive damping, Gauss-Newton, Dog Leg trust region) with real-time visualization support
  • linalg/: Linear algebra backends including sparse Cholesky decomposition, QR factorization, explicit Schur complement, and implicit Schur complement (matrix-free PCG)
  • observers/: Optimization observers and callbacks (Rerun visualization, custom hooks)

Workspace Sub-crates (in crates/):

  • apex-manifolds: Lie group implementations (SE2, SE3, SO2, SO3, SE_2(3), SGal(3), Sim(3), Rn) with analytic Jacobians
  • apex-io: File format parsers for G2O, TORO, and BAL formats
  • apex-camera-models: Camera projection models with analytic Jacobians (10 models)

Low-level Dependencies:

  • faer / nalgebra: High-performance linear algebra backends

๐Ÿ“‚ Datasets

Datasets are downloaded on demand using the built-in download_datasets tool in the apex-io crate. No Git LFS required.

# List all available datasets and selection numbers
cargo run --release -p apex-io --bin download_datasets -- --list

# Download benchmark datasets (all odometry g2o + largest from each BA dataset)
cargo run --release -p apex-io --bin download_datasets -- --select 10

# Download all odometry g2o datasets (2D + 3D)
cargo run --release -p apex-io --bin download_datasets -- --select 3

# Interactive mode (prompts for selection)
cargo run --release -p apex-io --bin download_datasets

Datasets are saved to data/odometry/ (g2o files) and data/bundle_adjustment/ (BAL format).

Available datasets:

  • Pose Graph SE2 (2D): M3500, mit, city10000, ring
  • Pose Graph SE3 (3D): sphere2500, parking-garage, torus3D, cubicle
  • Bundle Adjustment (UW BAL): ladybug, trafalgar, dubrovnik, venice, final

๐Ÿ“ฆ Workspace Crates

Apex Solver is organized as a Cargo workspace with specialized sub-crates that can be used independently:

Crate Description Docs
apex-manifolds Lie group manifolds (SE2, SE3, SO2, SO3, SE_2(3), SGal(3), Sim(3), Rn) with analytic Jacobians README
apex-camera-models 10 camera projection models for bundle adjustment and SLAM README
apex-io File I/O utilities for G2O, TORO, and BAL formats README

Using sub-crates independently:

[dependencies]
apex-manifolds = "0.2.0"

[dependencies]
apex-camera-models = "0.2.0"

[dependencies]
apex-io = "0.2.0"

๐Ÿ“Š Performance Benchmarks

Detailed benchmark tables comparing apex-solver against Ceres, GTSAM, g2o, factrs, and tiny-solver on 8 pose-graph datasets (SE2/SE3) and 4 BAL bundle-adjustment datasets.

โ†’ Full performance benchmarks


๐Ÿ“Š Examples

Usage examples covering pose graph optimization, custom factor implementation, and self-calibration bundle adjustment.

โ†’ Full examples


๐Ÿงฎ Technical Implementation

Robust Loss Functions

15 robust loss functions for handling outliers in optimization:

  • L2Loss: Standard least squares (no outliers)
  • L1Loss: Linear growth (light outliers)
  • HuberLoss: Quadratic near zero, linear after threshold (moderate outliers)
  • CauchyLoss: Logarithmic growth (heavy outliers)
  • FairLoss, GemanMcClureLoss, WelschLoss, TukeyBiweightLoss, AndrewsWaveLoss: Various robustness profiles
  • RamsayEaLoss: Asymmetric outliers
  • TrimmedMeanLoss: Ignores worst residuals
  • LpNormLoss: Generalized Lp norm
  • BarronGeneralLoss, AdaptiveBarronLoss: Adaptive robustness
  • TDistributionLoss: Statistical outliers

Usage:

use apex_solver::core::loss_functions::HuberLoss;

let loss = HuberLoss::new(1.345);  // 95% efficiency threshold
problem.add_residual_block(Box::new(factor), Some(Box::new(loss)));

Optimization Algorithms

Levenberg-Marquardt (Recommended)

  • Adaptive damping between gradient descent and Gauss-Newton
  • Robust convergence from poor initial estimates
  • Supports covariance estimation for uncertainty quantification
  • 9 comprehensive termination criteria (gradient norm, cost change, trust region radius, etc.)

Gauss-Newton

  • Fast convergence near solution
  • Minimal memory requirements
  • Best for well-initialized problems

Dog Leg Trust Region

  • Combines steepest descent and Gauss-Newton
  • Global convergence guarantees
  • Adaptive trust region management

Linear Algebra Backends

Four sparse linear solvers for different use cases:

  • Sparse Cholesky: Direct factorization of J^T J + ฮปI - fast, moderate memory, best for well-conditioned problems
  • Sparse QR: QR factorization of Jacobian - robust for rank-deficient systems, slightly slower
  • Explicit Schur Complement: Constructs reduced camera matrix S = B - E Cโปยน Eแต€ explicitly in memory - most accurate for bundle adjustment, moderate memory usage
  • Implicit Schur Complement: Matrix-free PCG solver computing only Sยทx products - memory-efficient for large-scale problems (10,000+ cameras), highly scalable

Configure via LinearSolverType in optimizer config:

config.with_linear_solver_type(LinearSolverType::ExplicitSchur)  // For bundle adjustment
config.with_linear_solver_type(LinearSolverType::ImplicitSchur)  // For very large BA

๐ŸŽจ Interactive Visualization

Real-time optimization debugging with integrated Rerun visualization using the observer pattern:

use apex_solver::optimizer::levenberg_marquardt::{LevenbergMarquardt, LevenbergMarquardtConfig};

let config = LevenbergMarquardtConfig::new()
    .with_max_iterations(100);

let mut solver = LevenbergMarquardt::with_config(config);

// Add Rerun visualization observer (requires `visualization` feature)
#[cfg(feature = "visualization")]
{
    use apex_solver::observers::RerunObserver;
    solver.add_observer(RerunObserver::new(true)?);  // true = spawn viewer
}

let result = solver.optimize(&problem, &initial_values)?;

Visualized Metrics:

  • Time series: Cost, gradient norm, damping (ฮป), step quality (ฯ), step norm
  • Matrix visualizations: Hessian heat map, gradient vector
  • 3D poses: SE3 camera frusta, SE2 2D points

Run Examples:

# Enable visualization feature and run
cargo run --release --features visualization --bin pose_graph_g2o -- --dataset sphere2500 --with-visualizer
cargo run --release --features visualization --bin pose_graph_g2o -- --dataset intel --with-visualizer

Note: The data files (e.g., sphere2500.g2o) must be downloaded first. See ๐Ÿ“‚ Datasets โ€” run cargo run --release -p apex-io --bin download_datasets -- --select 10 to get all benchmark datasets.

Zero overhead when disabled (feature-gated).


๐Ÿง  Learning Resources

Computer Vision Background

Lie Group Theory

Optimization Theory


๐Ÿ™ Acknowledgments

Apex Solver draws inspiration and reference implementations from:

  • Ceres Solver - Google's C++ optimization library
  • g2o - General framework for graph optimization
  • GTSAM - Georgia Tech Smoothing and Mapping library
  • tiny-solver - Lightweight nonlinear least squares solver
  • factrs - Rust factor graph optimization library
  • faer - High-performance linear algebra library for Rust
  • manif - C++ Lie theory library (for manifold conventions)
  • nalgebra - Geometry and linear algebra primitives

๐Ÿ“œ License

Licensed under the Apache License, Version 2.0. See LICENSE for details.