Overview¶
ImuFactor (5-way Factor)¶
The ImuFactor is the standard GTSAM factor for incorporating preintegrated IMU measurements into a factor graph. It’s a 5-way factor connecting:
- Pose at time ( - Pose3)
- Velocity at time ( - Vector3)
- Pose at time ( - Pose3)
- Velocity at time ( - Vector3)
- IMU Bias at time ( - imuBias::ConstantBias)
It takes a PreintegratedImuMeasurements object, which summarizes the IMU readings between times  and . The factor’s error function measures the discrepancy between the relative motion predicted by the preintegrated measurements (corrected for the current estimate of the bias at time ) and the relative motion implied by the state variables () connected to the factor.
ImuFactor2: NavState Variant¶
The ImuFactor2 is ternary variant of the ImuFactor that operates directly on NavState objects instead of separate Pose3 and Vector3 variables for pose and velocity. This simplifies the factor graph by reducing the number of connected variables and can make the graph more efficient to optimize.
Instead of connecting five variables (Pose_i, Vel_i, Pose_j, Vel_j, Bias_i), the ImuFactor2 connects three:
- NavStateat time (- NavStatecombines pose and velocity)
- NavStateat time $j`
- IMU Bias at time ( - imuBias::ConstantBias)
Modeling Bias¶
Both factors assume that the bias is constant between time and for the purpose of evaluating its error. That is typically a very good assumption, as bias evolves slowly over time.
The factors do not model the evolution of bias over time; if bias is expected to change, separate BetweenFactors on bias variables are typically needed, or the CombinedImuFactor should be used instead.
Mathematical Formulation¶
Let be the state (Attitude, Position, Velocity) at time , and be the bias estimate at time . Let be the state at time .
The PreintegratedImuMeasurements object (pim) provides:
- , , : Preintegrated measurements calculated using the fixed bias estimate ( - pim.biasHat()).
- Jacobians of the terms with respect to the bias. 
The factor first uses the pim object’s predict method (or equivalent calculations) to find the predicted state  based on  and the current estimate of the bias  from the factor graph values:
This prediction internally applies first-order corrections to the stored values based on the difference between and .
The factor’s 9-dimensional error vector is then calculated as the difference between the predicted state and the actual state in the tangent space of :
Or, more explicitly using the NavState::Logmap definition:
This error vector has components corresponding to errors in rotation (3), position (3), and velocity (3).
Key Functionality / API¶
- Constructor: - ImuFactor(keyPose_i, keyVel_i, keyPose_j, keyVel_j, keyBias_i, pim): Creates the factor, linking the five state/bias keys and providing the preintegrated measurements.
- preintegratedMeasurements(): Returns a const reference to the stored- PreintegratedImuMeasurementsobject.
- evaluateError(pose_i, vel_i, pose_j, vel_j, bias_i): Calculates the 9-dimensional error vector given the current values of the connected variables. Also computes Jacobians if requested.
- noiseModel(): Returns the noise model associated with the preintegrated measurement uncertainty (derived from- pim.preintMeasCov()).
- print/- equals: Standard factor methods.
Usage Example¶
Create parameters, a PIM object, define keys, and construct the factor.
from gtsam import PreintegrationParams, PreintegratedImuMeasurements, ImuFactor
from gtsam import NonlinearFactorGraph, Values, NavState, Pose3
from gtsam.symbol_shorthand import X,V,B
from gtsam.imuBias import ConstantBias
import numpy as np
# 1. Create Parameters and PIM (as in PreintegratedImuMeasurements example)
params = PreintegrationParams.MakeSharedU(9.81)
accel_noise_sigma = 0.1
gyro_noise_sigma = 0.01
params.setAccelerometerCovariance(np.eye(3) * accel_noise_sigma**2)
params.setGyroscopeCovariance(np.eye(3) * gyro_noise_sigma**2)
params.setIntegrationCovariance(np.eye(3) * 1e-8)
bias_hat = ConstantBias() # Assume zero bias used for preintegration
pim = PreintegratedImuMeasurements(params, bias_hat)
# Integrate some dummy measurements
dt = 0.01
acc_meas = np.array([0.0, 0.0, -9.81]) # Stationary
gyro_meas = np.array([0.0, 0.0, 0.0]) # Stationary
for _ in range(10):
    pim.integrateMeasurement(acc_meas, gyro_meas, dt)
# 2. Symbolic Keys will be X(0), V(0), X(1), V(1), B(0)
# 3. Create the ImuFactor
# The noise model is automatically derived from pim.preintMeasCov()
imu_factor = ImuFactor(X(0), V(0), X(1), V(1), B(0), pim)
print("Created ImuFactor:")
imu_factor.print()
# 4. Example: Evaluate error with perfect states (should be near zero)
graph = NonlinearFactorGraph()
graph.add(imu_factor)
values = Values()
pose_i = Pose3() # Identity
vel_i = np.zeros(3)
bias_i = ConstantBias() # Zero bias
# Predict state j using the PIM and the *same* bias used for integration
nav_state_i = NavState(pose_i, vel_i)
nav_state_j = pim.predict(nav_state_i, bias_i) # Use bias_i=bias_hat
pose_j = nav_state_j.pose()
vel_j = nav_state_j.velocity()
values.insert(X(0), pose_i)
values.insert(V(0), vel_i)
values.insert(X(1), pose_j)
values.insert(V(1), vel_j)
values.insert(B(0), bias_i)
error_vector = imu_factor.evaluateError(pose_i, vel_i, pose_j, vel_j, bias_i)
print("\nError vector (should be near zero):", error_vector)
print("Factor error (0.5 * ||error||^2_Sigma):", graph.error(values))Created ImuFactor:
ImuFactor(x0,v0,x1,v1,b0)
preintegrated measurements:
    deltaTij = 0.1
    deltaRij.ypr = ( 0 -0  0)
    deltaPij =        0        0 -0.04905
    deltaVij =      0      0 -0.981
    gyrobias = 0 0 0
    acc_bias = 0 0 0
    preintMeasCov 
[       1e-05            0            0            0  1.39793e-07            0            0   4.4145e-06            0
           0        1e-05            0 -1.39793e-07            0            0  -4.4145e-06            0            0
           0            0        1e-05            0            0            0            0            0            0
           0 -1.39793e-07            0  3.32969e-06            0            0  5.00974e-05            0            0
 1.39793e-07            0            0            0  3.32969e-06            0            0  5.00974e-05            0
           0            0            0            0            0    3.326e-06            0            0        5e-05
           0  -4.4145e-06            0  5.00974e-05            0            0   0.00100274            0            0
  4.4145e-06            0            0            0  5.00974e-05            0            0   0.00100274            0
           0            0            0            0            0        5e-05            0            0        0.001]
  noise model sigmas: 0.00316228 0.00316228 0.00316228 0.00182474 0.00182474 0.00182373  0.0316661  0.0316661  0.0316228
Error vector (should be near zero): [0. 0. 0. 0. 0. 0. 0. 0. 0.]
Factor error (0.5 * ||error||^2_Sigma): 0.0
We can also use ImuFactor2, with NavState, giving exactly the same result:
from gtsam import ImuFactor2
# 1. Create the ImuFactor2
# The noise model is automatically derived from pim.preintMeasCov()
imu_factor2 = ImuFactor2(X(0), X(1), B(0), pim)
print("Created ImuFactor2:")
imu_factor2.print()
# 2. Example: Evaluate error with perfect states (should be near zero)
graph = NonlinearFactorGraph()
graph.add(imu_factor2)
values = Values()
nav_state_i = NavState(pose_i, vel_i)
nav_state_j = pim.predict(nav_state_i, bias_i) # Use bias_i=bias_hat
values.insert(X(0), nav_state_i)
values.insert(X(1), nav_state_j)
values.insert(B(0), bias_i)
error_vector = imu_factor2.evaluateError(nav_state_i, nav_state_j, bias_i)
print("\nError vector (should be near zero):", error_vector)
print("Factor error (0.5 * ||error||^2_Sigma):", graph.error(values))Created ImuFactor2:
ImuFactor2(x0,x1,b0)
preintegrated measurements:
    deltaTij = 0.1
    deltaRij.ypr = ( 0 -0  0)
    deltaPij =        0        0 -0.04905
    deltaVij =      0      0 -0.981
    gyrobias = 0 0 0
    acc_bias = 0 0 0
    preintMeasCov 
[       1e-05            0            0            0  1.39793e-07            0            0   4.4145e-06            0
           0        1e-05            0 -1.39793e-07            0            0  -4.4145e-06            0            0
           0            0        1e-05            0            0            0            0            0            0
           0 -1.39793e-07            0  3.32969e-06            0            0  5.00974e-05            0            0
 1.39793e-07            0            0            0  3.32969e-06            0            0  5.00974e-05            0
           0            0            0            0            0    3.326e-06            0            0        5e-05
           0  -4.4145e-06            0  5.00974e-05            0            0   0.00100274            0            0
  4.4145e-06            0            0            0  5.00974e-05            0            0   0.00100274            0
           0            0            0            0            0        5e-05            0            0        0.001]
  noise model sigmas: 0.00316228 0.00316228 0.00316228 0.00182474 0.00182474 0.00182373  0.0316661  0.0316661  0.0316228
Error vector (should be near zero): [0. 0. 0. 0. 0. 0. 0. 0. 0.]
Factor error (0.5 * ||error||^2_Sigma): 0.0
Custom Preintegration Type¶
- Which implementation is used for the - DefaultPreintegrationTypeused by- ImuFactors and- Preintegrated*Measurementsdepends on the compile flag- GTSAM_TANGENT_PREINTEGRATION, which is true by default.- If false, - ManifoldPreintegrationis used. Please use this setting to get the exact implementation from Forster et al. (2017).
- If true, - TangentPreintegrationis used. This does the integration on the tangent space of the NavState manifold.
 
- In C++, if you wish to use any preintegration type other than the default, you must template your PIMs and factors on the desired preintegration type using the template-supporting classes - PreintegratedImuMeasurementsT,- ImuFactorT,- ImuFactor2T,- PreintegratedCombinedMeasurementsT, or- CombinedImuFactorT.
- Forster, C., Carlone, L., Dellaert, F., & Scaramuzza, D. (2017). On-Manifold Preintegration for Real-Time Visual–Inertial Odometry. IEEE Transactions on Robotics, 33(1), 1–21. 10.1109/tro.2016.2597321