Migrating to Isaac Lab 3.0#

Isaac Lab 3.0 introduces a multi-backend architecture that separates simulation backend-specific code from the core Isaac Lab API. This allows for future support of different physics backends while maintaining a consistent user-facing API.

This guide covers the main breaking changes and deprecations you need to address when migrating from Isaac Lab 2.x to Isaac Lab 3.0.

Visualizer CLI and Headless Behavior#

In Isaac Lab 3.0, the --headless argument is deprecated. Instead, use --visualizer / --viz to determine whether viewer apps are launched with an Isaac Lab command.

Visualizers are lightweight viewer apps for monitoring, debugging, and recording workflows (see Visualization).

The details below describe how CLI visualizer arguments resolve together with SimulationCfg.visualizer_cfgs.

  • --viz accepts comma-separated values (for example --viz kit,newton).

  • If omitted, visualizers are resolved from SimulationCfg.visualizer_cfgs.

  • --viz none explicitly disables all visualizers, including config-defined ones.

  • --headless is deprecated (still supported) and overrides --viz by forcing headless mode.

For the full behavior of visualizer resolution, with the visualizer CLI arg, visualizer configs, and --headless, see Common modes.

Multi-Backend Architecture#

Isaac Lab 3.0 introduces a factory-based multi-backend architecture that allows asset classes to be backed by different physics engines — currently PhysX and Newton.

When you instantiate an asset class from the isaaclab package (e.g., Articulation, RigidObject), a factory automatically resolves and loads the correct backend implementation:

from isaaclab.assets import Articulation, ArticulationCfg

# The factory pattern creates the appropriate backend implementation.
# No import changes are needed — the same isaaclab imports work regardless of backend.
robot = Articulation(cfg=ArticulationCfg(...))

The factory works by convention: for a class defined in isaaclab.assets.articulation, it imports the matching class from isaaclab_{backend}.assets.articulation. This means the isaaclab_physx and isaaclab_newton packages mirror the isaaclab module structure.

Note

The backend is currently set to "physx" by default. Newton backend support is being actively developed. When backend selection is fully configurable, you will be able to switch backends without changing any asset import paths.

For a comprehensive overview of the factory pattern, backend selection, and how to add a new backend, see Multi-Backend Architecture.

New isaaclab_physx and isaaclab_newton Extensions#

Two new backend extensions have been introduced:

  • ``isaaclab_physx`` — PhysX-specific implementations of all asset and sensor classes.

  • ``isaaclab_newton`` — Newton-specific implementations of asset classes (Articulation and RigidObject).

The following classes have been moved to isaaclab_physx:

Isaac Lab 2.x

Isaac Lab 3.0

from isaaclab.assets import DeformableObject

from isaaclab_physx.assets import DeformableObject

from isaaclab.assets import DeformableObjectCfg

from isaaclab_physx.assets import DeformableObjectCfg

from isaaclab.assets import DeformableObjectData

from isaaclab_physx.assets import DeformableObjectData

from isaaclab.assets import SurfaceGripper

from isaaclab_physx.assets import SurfaceGripper

from isaaclab.assets import SurfaceGripperCfg

from isaaclab_physx.assets import SurfaceGripperCfg

Note

The isaaclab_physx extension is installed automatically with Isaac Lab. No additional installation steps are required.

Unchanged Imports#

The following asset classes remain in the isaaclab package and can still be imported as before:

These classes now inherit from new abstract base classes but maintain full backward compatibility.

The following sensor classes also remain in the isaaclab package with unchanged imports:

These sensor classes now use factory patterns that automatically instantiate the appropriate backend implementation (PhysX by default), maintaining full backward compatibility.

If you need to import the PhysX sensor implementations directly (e.g., for type hints or subclassing), you can import from isaaclab_physx.sensors:

# Direct PhysX implementation imports
from isaaclab_physx.sensors import ContactSensor, ContactSensorData
from isaaclab_physx.sensors import Imu, ImuData
from isaaclab_physx.sensors import FrameTransformer, FrameTransformerData

New isaaclab_newton Extension#

A new extension isaaclab_newton provides Newton physics backend implementations for:

  • Articulation and ArticulationData

  • RigidObject and RigidObjectData

These classes implement the same base interfaces as their PhysX counterparts (BaseArticulation, BaseRigidObject), ensuring a consistent API across backends. They use the same warp-based data conventions (wp.array with structured types, _index / _mask write methods).

Note

The isaaclab_newton extension requires the newton package and its dependencies (mujoco, mujoco-warp). These are installed automatically when installing the isaaclab_newton package.

If you need to import Newton implementations directly (e.g., for type hints or subclassing):

from isaaclab_newton.assets import Articulation as NewtonArticulation
from isaaclab_newton.assets import RigidObject as NewtonRigidObject

Sensor Pose Properties Deprecation#

The pose_w, pos_w, and quat_w properties on ContactSensorData and ImuData are deprecated and will be removed in a future release.

If you need to track sensor poses in world frame, please use a dedicated sensor such as FrameTransformer instead.

Before (deprecated):

# Using pose properties directly on sensor data
sensor_pos = contact_sensor.data.pos_w
sensor_quat = contact_sensor.data.quat_w

After (recommended):

# Use FrameTransformer to track sensor pose
frame_transformer = FrameTransformer(FrameTransformerCfg(
    prim_path="{ENV_REGEX_NS}/Robot/base",
    target_frames=[
        FrameTransformerCfg.FrameCfg(prim_path="{ENV_REGEX_NS}/Robot/sensor_link"),
    ],
))
sensor_pos = frame_transformer.data.target_pos_w
sensor_quat = frame_transformer.data.target_quat_w

Multi-Backend Support: PresetCfg Pattern#

Isaac Lab 3.0 introduces a PresetCfg pattern for writing environment configurations that work with both the PhysX and Newton backends. Instead of hard-coding a single physics config, environments declare named configuration variants. The active variant is selected at launch via a Hydra CLI override.

What is PresetCfg?#

PresetCfg is a base @configclass whose typed fields represent named variants of a configuration section. The field named default is used when no CLI override is given. Other fields are named presets selectable with presets=<name> on the command line:

from isaaclab_tasks.utils import PresetCfg
from isaaclab.utils import configclass

@configclass
class MyPhysicsCfg(PresetCfg):
    default: PhysxCfg = PhysxCfg(...)   # used when no override is given
    physx:   PhysxCfg = PhysxCfg(...)   # selected by presets=physx
    newton:  NewtonCfg = NewtonCfg(...)  # selected by presets=newton

Selecting a preset at launch#

Pass presets=newton (or presets=physx) on the CLI to swap the entire config section:

# Run with Newton backend
python train.py task=Isaac-Franka-Cabinet-v0 presets=newton

# Run with default (PhysX) backend
python train.py task=Isaac-Franka-Cabinet-v0

Adding Multi-Backend Support to an Environment#

Step 1 — Physics config

Replace a plain PhysxCfg(...) assignment in __post_init__ with a PresetCfg subclass that carries both a PhysX and a Newton variant.

Before:

def __post_init__(self):
    self.sim.dt = 1 / 60
    self.sim.physics = PhysxCfg(bounce_threshold_velocity=0.2)

After:

from isaaclab_newton.physics import MJWarpSolverCfg, NewtonCfg
from isaaclab_physx.physics import PhysxCfg
from isaaclab_tasks.utils import PresetCfg

@configclass
class ReachPhysicsCfg(PresetCfg):
    default: PhysxCfg = PhysxCfg(bounce_threshold_velocity=0.2)
    physx:   PhysxCfg = PhysxCfg(bounce_threshold_velocity=0.2)
    newton:  NewtonCfg = NewtonCfg(
        solver_cfg=MJWarpSolverCfg(
            njmax=20, nconmax=20, ls_iterations=20,
            cone="pyramidal", ls_parallel=True,
            integrator="implicitfast", impratio=1,
        ),
        num_substeps=1,
        debug_mode=False,
    )

# In the env cfg __post_init__:
def __post_init__(self):
    self.sim.dt = 1 / 60
    self.sim.physics = ReachPhysicsCfg()

Key Newton solver parameters:

Parameter

Effect

njmax

Max constraint rows; set ≥ expected contact count per env

nconmax

Max contacts per env

ls_iterations

Linear solver iterations (higher = more stable, slower)

cone

"pyramidal" (fast) or "elliptic" (more accurate)

integrator

"implicitfast" (recommended) or "euler"

impratio

Impedance ratio; >1 improves soft contact stability

num_substeps

Physics substeps per environment step

Step 2 — Differentiating Newton and PhysX Configs

Not all configurations may be the same between Newton and PhysX simulations. We can provide a Newton-specific config such as:

@configclass
class EventCfg:
    """Full event config (PhysX-compatible)."""
    robot_physics_material = EventTerm(
        func=mdp.randomize_rigid_body_material,
        mode="startup",
        params={...},
    )
    reset_all = EventTerm(func=mdp.reset_scene_to_default, mode="reset")
    reset_robot_joints = EventTerm(
        func=mdp.reset_joints_by_offset, mode="reset", params={...}
    )


@configclass
class _EnvNewtonEventCfg:
    """Newton-compatible events."""
    reset_all = EventTerm(func=mdp.reset_scene_to_default, mode="reset")
    reset_robot_joints = EventTerm(
        func=mdp.reset_joints_by_offset, mode="reset", params={...}
    )


@configclass
class EnvEventCfg(PresetCfg):
    default: EventCfg = EventCfg()
    physx:   EventCfg = EventCfg()
    newton:  _EnvNewtonEventCfg = _EnvNewtonEventCfg()

Then change the events field in your env cfg from EventCfg to EnvEventCfg:

@configclass
class MyEnvCfg(ManagerBasedRLEnvCfg):
    events: EnvEventCfg = EnvEventCfg()  # was: EventCfg = EventCfg()

RigidObjectCollection API Renaming#

The RigidObjectCollection and RigidObjectCollectionData classes have undergone an API rename to provide consistency with other asset classes. The object_* naming convention has been deprecated in favor of body_*.

Method Renames#

The following methods have been renamed. The old methods are deprecated and will be removed in a future release:

Deprecated (2.x)

New (3.0)

write_object_state_to_sim()

write_body_state_to_sim()

write_object_link_state_to_sim()

write_body_link_state_to_sim()

write_object_pose_to_sim()

write_body_pose_to_sim()

write_object_link_pose_to_sim()

write_body_link_pose_to_sim()

write_object_com_pose_to_sim()

write_body_com_pose_to_sim()

write_object_velocity_to_sim()

write_body_com_velocity_to_sim()

write_object_com_velocity_to_sim()

write_body_com_velocity_to_sim()

write_object_link_velocity_to_sim()

write_body_link_velocity_to_sim()

find_objects()

find_bodies()

Property Renames (Data Class)#

The following properties on RigidObjectCollectionData have been renamed. The old properties are deprecated and will be removed in a future release:

Deprecated (2.x)

New (3.0)

default_object_state

default_body_state

object_names

body_names

object_link_pose_w

body_link_pose_w

object_link_vel_w

body_link_vel_w

object_com_pose_w

body_com_pose_w

object_com_vel_w

body_com_vel_w

object_state_w

body_state_w

object_link_state_w

body_link_state_w

object_com_state_w

body_com_state_w

object_com_acc_w

body_com_acc_w

object_com_pose_b

body_com_pose_b

object_link_pos_w

body_link_pos_w

object_link_quat_w

body_link_quat_w

object_link_lin_vel_w

body_link_lin_vel_w

object_link_ang_vel_w

body_link_ang_vel_w

object_com_pos_w

body_com_pos_w

object_com_quat_w

body_com_quat_w

object_com_lin_vel_w

body_com_lin_vel_w

object_com_ang_vel_w

body_com_ang_vel_w

object_com_lin_acc_w

body_com_lin_acc_w

object_com_ang_acc_w

body_com_ang_acc_w

object_com_pos_b

body_com_pos_b

object_com_quat_b

body_com_quat_b

object_link_lin_vel_b

body_link_lin_vel_b

object_link_ang_vel_b

body_link_ang_vel_b

object_com_lin_vel_b

body_com_lin_vel_b

object_com_ang_vel_b

body_com_ang_vel_b

object_pose_w

body_pose_w

object_pos_w

body_pos_w

object_quat_w

body_quat_w

object_vel_w

body_vel_w

object_lin_vel_w

body_lin_vel_w

object_ang_vel_w

body_ang_vel_w

object_lin_vel_b

body_lin_vel_b

object_ang_vel_b

body_ang_vel_b

object_acc_w

body_acc_w

object_lin_acc_w

body_lin_acc_w

object_ang_acc_w

body_ang_acc_w

Note

All deprecated methods and properties will issue a deprecation warning when used. Your existing code will continue to work, but you should migrate to the new API to avoid issues in future releases.

Migration Example#

Here’s a complete example showing how to update your code:

Before (Isaac Lab 2.x):

from isaaclab.assets import DeformableObject, DeformableObjectCfg
from isaaclab.assets import SurfaceGripper, SurfaceGripperCfg
from isaaclab.assets import RigidObjectCollection

# Using deprecated root_physx_view
robot = scene["robot"]
masses = robot.root_physx_view.get_masses()

# Using deprecated object_* API
collection = scene["object_collection"]
poses = collection.data.object_pose_w
collection.write_object_state_to_sim(state, env_ids=env_ids, object_ids=object_ids)

After (Isaac Lab 3.0):

from isaaclab_physx.assets import DeformableObject, DeformableObjectCfg
from isaaclab_physx.assets import SurfaceGripper, SurfaceGripperCfg
from isaaclab.assets import RigidObjectCollection  # unchanged

# Using new root_view property
robot = scene["robot"]
masses = robot.root_view.get_masses()

# Using new body_* API
collection = scene["object_collection"]
poses = collection.data.body_pose_w
collection.write_body_state_to_sim(state, env_ids=env_ids, body_ids=object_ids)

Quaternion Format#

The quaternion format changed from WXYZ to XYZW.

Component

Old Format (WXYZ)

New Format (XYZW)

Order

(w, x, y, z)

(x, y, z, w)

Identity

(1.0, 0.0, 0.0, 0.0)

(0.0, 0.0, 0.0, 1.0)

Why This Change?#

The new XYZW format aligns with:

  • Warp: NVIDIA’s spatial computing framework

  • PhysX: PhysX physics engine

  • Newton: Newton multi-solver framework

This alignment removes the need for internal quaternion conversions, making the code simpler, faster, and less error-prone.

What You Need to Update#

Any hard-coded quaternion values in your code need to be converted from WXYZ to XYZW. This includes:

  1. Configuration files - rot parameters in asset configs

  2. Task definitions - Goal poses, initial states

  3. Controller parameters - Target orientations

  4. Documentation - Code examples with quaternions

Also, if you were relying on the convert_quat() function to convert quaternions, this should no longer be needed. (This would happen if you were pulling values from the views directly.)

Example: Updating Asset Configuration#

Before (WXYZ):

from isaaclab.assets import AssetBaseCfg

cfg = AssetBaseCfg(
    init_state=AssetBaseCfg.InitialStateCfg(
        pos=(0.0, 0.0, 0.5),
        rot=(1.0, 0.0, 0.0, 0.0),  # OLD: w, x, y, z
    ),
)

After (XYZW):

from isaaclab.assets import AssetBaseCfg

cfg = AssetBaseCfg(
    init_state=AssetBaseCfg.InitialStateCfg(
        pos=(0.0, 0.0, 0.5),
        rot=(0.0, 0.0, 0.0, 1.0),  # NEW: x, y, z, w
    ),
)

Using the Quaternion Finder Tool#

We provide a tool to help you find and fix quaternions in your codebase automatically. This is not a bulletproof tool, but it should help you find most of the quaternions that need to be updated. You should review the results manually.

Warning

Do not run the tool on the whole codebase! If you run the tool on our own packages (isaaclab, or isaaclab_tasks for instance) it will find all the quaternions that we already converted. This tool is only meant to be used on your own codebase with no overlap with our own packages.

Finding Quaternions#

Run the tool to scan your code for potential quaternions:

# Scan the 'source' directory (default)
python scripts/tools/find_quaternions.py

# Scan a specific path
python scripts/tools/find_quaternions.py --path my_project/

# Compare against a different branch
python scripts/tools/find_quaternions.py --base develop

Tip

We recommend always running the tool with a custom base branch and a specific path.

The tool will show you:

  • Quaternions that haven’t been updated (marked as UNCHANGED)

  • Whether each looks like a WXYZ identity quaternion (WXYZ_IDENTITY)

  • Whether the format is likely WXYZ (LIKELY_WXYZ)

Understanding the Output#

my_project/robot_cfg.py:42:8 ⚠ UNCHANGED [WXYZ_IDENTITY]
  Values: [1.0, 0.0, 0.0, 0.0]
  Source: rot=(1.0, 0.0, 0.0, 0.0),

This tells you:

  • File and line: my_project/robot_cfg.py:42

  • Status: UNCHANGED means this line hasn’t been modified yet

  • Flag: WXYZ_IDENTITY means it’s the identity quaternion in old WXYZ format

  • Values: The actual quaternion values found

  • Source: The line of code for context

Filtering Results#

Focus on specific types of quaternions:

# Only show identity quaternions [1, 0, 0, 0]
python scripts/tools/find_quaternions.py --check-identity

# Only show quaternions likely in WXYZ format
python scripts/tools/find_quaternions.py --likely-wxyz

# Show ALL potential quaternions (ignore format heuristics)
python scripts/tools/find_quaternions.py --all-quats

Fixing Quaternions Automatically#

The tool can automatically convert quaternions from WXYZ to XYZW:

# Interactive mode: prompts before each fix
python scripts/tools/find_quaternions.py --fix

# Only fix identity quaternions (safest option)
python scripts/tools/find_quaternions.py --fix-identity-only

# Preview changes without applying them
python scripts/tools/find_quaternions.py --fix --dry-run

# Apply all fixes without prompting
python scripts/tools/find_quaternions.py --fix --force

Interactive Fix Example#

When running with --fix, you’ll see something like:

────────────────────────────────────────────────────────────────────────────────
📍 my_project/robot_cfg.py:42 [WXYZ_IDENTITY]
────────────────────────────────────────────────────────────────────────────────
     40 |     init_state=AssetBaseCfg.InitialStateCfg(
     41 |         pos=(0.0, 0.0, 0.5),
>>>  42 |         rot=(1.0, 0.0, 0.0, 0.0),
     43 |     ),
     44 | )
────────────────────────────────────────────────────────────────────────────────
  Change: [1.0, 0.0, 0.0, 0.0] → [0.0, 0.0, 0.0, 1.0]
  Result: rot=(0.0, 0.0, 0.0, 1.0),
Apply this fix? [Y/n/a/q]:

Options:

  • Y (yes): Apply this fix

  • n (no): Skip this one

  • a (all): Apply all remaining fixes without asking

  • q (quit): Stop fixing

How the Tool Works#

The tool uses several techniques to find quaternions:

  1. Python files: Parses the code using AST (Abstract Syntax Tree) to find 4-element tuples and lists with numeric values.

  2. JSON files: Uses regex to find 4-element arrays.

  3. RST documentation: Searches for quaternion-like patterns in docs.

To identify if something is a quaternion, the tool checks:

  • Is it exactly 4 numeric values?

  • Does the sum of squares ≈ 1? (unit quaternion property)

  • Does it match known patterns like identity quaternions?

To determine if it’s in WXYZ format:

  • Is the first value 1.0 and rest are 0? (WXYZ identity)

  • Is the first value a common cos(θ/2) value like 0.707, 0.866, etc.?

  • Is the pattern consistent with first-element being the scalar part?

Best Practices for Migration#

  1. Start with a clean git state - Commit your work before running fixes.

  2. Run the tool first without ``–fix`` - Review what will be changed.

  3. Fix identity quaternions first - They’re the most common and safest:

    python scripts/tools/find_quaternions.py --fix-identity-only
    
  4. Review non-identity quaternions manually - Some 4-element lists might not be quaternions (e.g., RGBA colors, bounding boxes).

  5. Test your code - Run your simulations to verify everything works correctly.

  6. Check documentation - Update any docs or comments that mention quaternion format.

API Changes#

The convert_quat function has been removed#

Previously, IsaacLab had a utility function to convert between quaternion formats:

# OLD - No longer needed
from isaaclab.utils.math import convert_quat
quat_xyzw = convert_quat(quat_wxyz, "xyzw")

Since everything now uses XYZW natively, this function is no longer needed. If you were using it, simply remove the conversion calls.

Math utility functions now expect XYZW#

All quaternion functions in isaaclab.utils.math now expect and return quaternions in XYZW format:

Warp Backend for Asset and Sensor Data#

All .data.* properties on asset and sensor classes now return wp.array instead of torch.Tensor. This change applies to all asset classes (Articulation, RigidObject, RigidObjectCollection, DeformableObject) and all sensor classes (ContactSensor, Imu, FrameTransformer).

To convert back to torch.Tensor for use with PyTorch operations, wrap the property access with wp.to_torch():

import warp as wp

# Before (Isaac Lab 2.x)
root_pos = robot.data.root_pos_w             # torch.Tensor
joint_pos = robot.data.joint_pos              # torch.Tensor
contact_forces = sensor.data.net_forces_w     # torch.Tensor

# After (Isaac Lab 3.x)
root_pos = robot.data.root_pos_w              # wp.array
joint_pos = robot.data.joint_pos              # wp.array
contact_forces = sensor.data.net_forces_w     # wp.array

# To use with torch operations, wrap with wp.to_torch()
root_pos_torch = wp.to_torch(robot.data.root_pos_w)        # torch.Tensor
joint_pos_torch = wp.to_torch(robot.data.joint_pos)        # torch.Tensor
contact_torch = wp.to_torch(sensor.data.net_forces_w)      # torch.Tensor

Common patterns that need updating:

# Cloning data
# Before:
pos = robot.data.root_pos_w.clone()
# After:
pos = wp.to_torch(robot.data.root_pos_w).clone()

# Creating zero tensors with matching shape
# Before:
zeros = torch.zeros_like(robot.data.root_pos_w)
# After:
zeros = torch.zeros_like(wp.to_torch(robot.data.root_pos_w))

# Assertions in tests
# Before:
torch.testing.assert_close(robot.data.root_pos_w, expected)
# After:
torch.testing.assert_close(wp.to_torch(robot.data.root_pos_w), expected)
Affected classes#

Class

Package

Articulation

isaaclab / isaaclab_physx

RigidObject

isaaclab / isaaclab_physx

RigidObjectCollection

isaaclab / isaaclab_physx

DeformableObject

isaaclab_physx

ContactSensor

isaaclab_physx

Imu

isaaclab_physx

FrameTransformer

isaaclab_physx

Note

An automated migration tool is provided at scripts/tools/wrap_warp_to_torch.py. It scans Python files for .data.<property> accesses and wraps them with wp.to_torch(). Usage:

# Dry run (preview changes)
python scripts/tools/wrap_warp_to_torch.py path/to/your/code --dry-run

# Apply changes in-place
python scripts/tools/wrap_warp_to_torch.py path/to/your/code

Always review the changes after running the tool, as some accesses (e.g., those already passed to warp-native functions) should not be wrapped.

Write Method Index/Mask Split#

All asset write methods have been split into two explicit variants:

  • write_*_to_sim_index(data, env_ids) — accepts partial data for a sparse set of environment indices. The data tensor has shape (len(env_ids), ...).

  • write_*_to_sim_mask(data, env_mask) — accepts full data for all environments with a boolean mask selecting which environments to update. The data tensor has shape (num_envs, ...).

The previous write_*_to_sim(data, env_ids) methods have been removed.

# Before (Isaac Lab 2.x)
robot.write_root_pose_to_sim(pose_data, env_ids)

# After (Isaac Lab 3.x) — indexed variant (partial data)
robot.write_root_pose_to_sim_index(root_pose=pose_data, env_ids=env_ids)

# After (Isaac Lab 3.x) — mask variant (full data, boolean mask)
robot.write_root_pose_to_sim_mask(root_pose=pose_data, env_mask=env_mask)
Affected write methods (RigidObject / Articulation)#

Old method

New methods

write_root_pose_to_sim

write_root_pose_to_sim_index / write_root_pose_to_sim_mask

write_root_link_pose_to_sim

write_root_link_pose_to_sim_index / write_root_link_pose_to_sim_mask

write_root_com_pose_to_sim

write_root_com_pose_to_sim_index / write_root_com_pose_to_sim_mask

write_root_velocity_to_sim

write_root_velocity_to_sim_index / write_root_velocity_to_sim_mask

write_root_com_velocity_to_sim

write_root_com_velocity_to_sim_index / write_root_com_velocity_to_sim_mask

write_root_link_velocity_to_sim

write_root_link_velocity_to_sim_index / write_root_link_velocity_to_sim_mask

Additional Articulation-specific write methods#

Old method

New methods

write_joint_position_to_sim

write_joint_position_to_sim_index / write_joint_position_to_sim_mask

write_joint_velocity_to_sim

write_joint_velocity_to_sim_index / write_joint_velocity_to_sim_mask

write_joint_stiffness_to_sim

write_joint_stiffness_to_sim_index / write_joint_stiffness_to_sim_mask

write_joint_damping_to_sim

write_joint_damping_to_sim_index / write_joint_damping_to_sim_mask

write_joint_position_limit_to_sim

write_joint_position_limit_to_sim_index / write_joint_position_limit_to_sim_mask

write_joint_velocity_limit_to_sim

write_joint_velocity_limit_to_sim_index / write_joint_velocity_limit_to_sim_mask

write_joint_effort_limit_to_sim

write_joint_effort_limit_to_sim_index / write_joint_effort_limit_to_sim_mask

write_joint_armature_to_sim

write_joint_armature_to_sim_index / write_joint_armature_to_sim_mask

write_joint_friction_coefficient_to_sim

write_joint_friction_coefficient_to_sim_index / write_joint_friction_coefficient_to_sim_mask

RigidObjectCollection write methods#

Old method

New methods

write_body_pose_to_sim

write_body_pose_to_sim_index / write_body_pose_to_sim_mask

write_body_link_pose_to_sim

write_body_link_pose_to_sim_index / write_body_link_pose_to_sim_mask

write_body_com_pose_to_sim

write_body_com_pose_to_sim_index / write_body_com_pose_to_sim_mask

write_body_velocity_to_sim

write_body_velocity_to_sim_index / write_body_velocity_to_sim_mask

write_body_com_velocity_to_sim

write_body_com_velocity_to_sim_index / write_body_com_velocity_to_sim_mask

write_body_link_velocity_to_sim

write_body_link_velocity_to_sim_index / write_body_link_velocity_to_sim_mask

TimestampedBufferWarp#

If you have custom asset or sensor data classes that subclass the Isaac Lab base data classes, note that internal buffers have changed from TimestampedBuffer to TimestampedBufferWarp. The new class takes (shape, device, wp_dtype) as constructor arguments instead of a torch.Tensor:

import warp as wp
from isaaclab.utils.buffers import TimestampedBufferWarp

# Before (Isaac Lab 2.x)
self._data.root_pos_w = TimestampedBuffer(torch.zeros(num_envs, 3, device=device))

# After (Isaac Lab 3.x)
self._data.root_pos_w = TimestampedBufferWarp(
    shape=(num_envs,), device=device, wp_dtype=wp.vec3f
)

URDF Importer#

The URDF importer in Isaac Sim was rewritten to version 3.0, using the urdf-usd-converter library and the isaacsim.asset.transformer.rules extension to produce structured USD output. The old C++ binding-based API (using Kit commands URDFParseFile/URDFImportRobot and the _urdf interface from acquire_urdf_interface()) has been replaced with a new Python-based pipeline.

The IsaacLab UrdfConverter has been updated to replicate the new URDFImporter.import_urdf() pipeline, inserting IsaacLab-specific post-processing (fix base, joint drives, link density) on the intermediate USD stage before the asset transformer restructures the output.

Important

The previous version-pinning mechanism that locked the URDF importer extension to isaacsim.asset.importer.urdf-2.4.31 has been removed. The converter now uses whichever version of the extension is available in your Isaac Sim installation.

Deprecated Settings#

The following UrdfConverterCfg settings are deprecated because the new URDF importer 3.0 no longer supports them. They are kept for backward compatibility but will log warnings if enabled:

Setting

Notes

convert_mimic_joints_to_normal_joints

No longer supported by the importer.

replace_cylinders_with_capsules

No longer supported by the importer.

root_link_name

No longer supported by the importer.

Note

The merge_fixed_joints setting is still supported. It is now implemented as a URDF XML pre-processing step that runs before the USD conversion. Fixed joints are removed and child link elements (visual, collision, inertial) are merged into the parent link with correct transform composition.

Additionally, the NaturalFrequencyGainsCfg gains mode is deprecated. The compute_natural_stiffness function that it depended on has been removed from the importer. If NaturalFrequencyGainsCfg is used, a DeprecationWarning is emitted and joint drive gains are left at the values produced by the URDF importer. Use PDGainsCfg instead.

The make_instanceable setting from the base class is also no longer supported and will be ignored. Assets will be made instanceable by default.

Updated CLI Tool#

The convert_urdf.py script has been updated. The usd_file_name is now determined automatically by the importer based on the robot name and cannot be overridden.

Before (Isaac Lab 2.x):

./isaaclab.sh -p scripts/tools/convert_urdf.py \
  robot.urdf \
  /output/dir/robot.usd \
  --fix-base \
  --merge-joints

After (Isaac Lab 3.0):

./isaaclab.sh -p scripts/tools/convert_urdf.py \
  robot.urdf \
  /output/dir \
  --fix-base \
  --joint-stiffness 100.0 \
  --joint-damping 1.0

Note

The --merge-joints flag is still accepted and correctly triggers the pre-processing step to merge fixed joints.

Updated Python API#

If you use UrdfConverter or UrdfConverterCfg directly in your code, note the following changes:

  1. The usd_file_name is now set automatically by the converter based on the URDF file name. The importer generates output at {usd_dir}/{robot_name}/{robot_name}.usda.

  2. The make_instanceable setting is no longer supported. Assets will be made instanceable by default.

  3. The merge_fixed_joints parameter is now implemented as a pre-processing step.

Before (Isaac Lab 2.x):

from isaaclab.sim.converters import UrdfConverter, UrdfConverterCfg

cfg = UrdfConverterCfg(
    asset_path="robot.urdf",
    usd_dir="/output/dir",
    usd_file_name="robot.usd",
    fix_base=True,
    merge_fixed_joints=True,
    make_instanceable=True,
    joint_drive=UrdfConverterCfg.JointDriveCfg(
        gains=UrdfConverterCfg.JointDriveCfg.PDGainsCfg(
            stiffness=None,  # use URDF values
            damping=None,
        ),
    ),
)

After (Isaac Lab 3.0):

from isaaclab.sim.converters import UrdfConverter, UrdfConverterCfg

cfg = UrdfConverterCfg(
    asset_path="robot.urdf",
    usd_dir="/output/dir",
    # usd_file_name is determined automatically from the robot name
    fix_base=True,
    merge_fixed_joints=True,  # supported via pre-processing
    joint_drive=UrdfConverterCfg.JointDriveCfg(
        gains=UrdfConverterCfg.JointDriveCfg.PDGainsCfg(
            stiffness=None,  # use URDF values
            damping=None,
        ),
    ),
)

MJCF Importer#

The MJCF importer in Isaac Sim was rewritten to use the mujoco-usd-converter library. The old C++ binding-based API (using Kit commands MJCFCreateAsset/MJCFCreateImportConfig and the ImportConfig class) has been replaced with a new pure-Python MJCFImporter class and MJCFImporterConfig dataclass.

Important

The new MJCF importer produces USD assets with nested rigid bodies (i.e., RigidBodyAPI is applied to each link prim individually) instead of a single articulation root with rigid body applied only at the top level. This matches how MuJoCo represents bodies and is physically more accurate, but it may affect code that assumes a flat rigid body hierarchy. If you have downstream logic that traverses the USD structure of MJCF-imported assets, verify that it handles nested rigid body prims correctly.

Removed Settings#

The following MjcfConverterCfg settings have been removed because the new converter handles them automatically based on the MJCF file content:

  • fix_base — base fixedness is now inferred from the MJCF <freejoint> tag.

  • link_density — density is now read directly from the MJCF model.

  • import_inertia_tensor — inertia tensors are always imported.

  • import_sites — sites are always imported.

The make_instanceable setting from the base class is also no longer supported and will be ignored.

New Settings#

The following new settings were added to MjcfConverterCfg:

Setting

Description

merge_mesh

Merge meshes where possible to optimize the model.

collision_from_visuals

Generate collision geometry from visuals.

collision_type

Type of collision geometry (e.g. "default", "Convex Hull", "Convex Decomposition").

Renamed Settings#

Old (2.x)

New (3.0)

self_collision

self_collision (unchanged)

Note

The underlying Isaac Sim API renamed self_collision to allow_self_collision. The IsaacLab MjcfConverterCfg keeps using self_collision for backward compatibility and maps it to the new name internally.

Updated CLI Tool#

The convert_mjcf.py script has been updated to match the new importer settings. Old command-line flags (--fix-base, --make-instanceable, --import-sites) are no longer available.

Before (Isaac Lab 2.x):

./isaaclab.sh -p scripts/tools/convert_mjcf.py \
  ../mujoco_menagerie/unitree_h1/h1.xml \
  source/isaaclab_assets/data/Robots/Unitree/h1.usd \
  --import-sites \
  --make-instanceable

After (Isaac Lab 3.0):

./isaaclab.sh -p scripts/tools/convert_mjcf.py \
  ../mujoco_menagerie/unitree_h1/h1.xml \
  source/isaaclab_assets/data/Robots/Unitree/h1.usd \
  --merge-mesh \
  --self-collision

New flags: --merge-mesh, --collision-from-visuals, --collision-type, --self-collision.

Updated Python API#

If you use MjcfConverter or MjcfConverterCfg directly in your code, update your configuration:

Before (Isaac Lab 2.x):

from isaaclab.sim.converters import MjcfConverter, MjcfConverterCfg

cfg = MjcfConverterCfg(
    asset_path="robot.xml",
    usd_dir="/output/dir",
    fix_base=True,
    import_sites=True,
    make_instanceable=True,
)

After (Isaac Lab 3.0):

from isaaclab.sim.converters import MjcfConverter, MjcfConverterCfg

cfg = MjcfConverterCfg(
    asset_path="robot.xml",
    usd_dir="/output/dir",
    merge_mesh=True,
    collision_from_visuals=False,
    self_collision=False,
)

XR Teleoperation: Isaac Teleop Integration#

The native XR teleoperation stack in isaaclab.devices.openxr has been deprecated and replaced by Isaac Teleop, integrated via the isaaclab_teleop extension. The isaac-teleop-device-plugins repository has also been deprecated; all device plugin support is now in Isaac Teleop.

For full documentation on the new stack, see Isaac Teleop.

Installation Requirement#

Isaac Teleop must now be installed in your Isaac Lab environment:

pip install isaacteleop~=1.0 --extra-index-url https://pypi.nvidia.com

See Install Isaac Teleop for complete installation instructions.

Import Changes#

Deprecated (2.x)

New (3.0)

from isaaclab.devices.openxr import OpenXRDevice

from isaaclab_teleop import IsaacTeleopDevice

from isaaclab.devices.openxr import OpenXRDeviceCfg

from isaaclab_teleop import IsaacTeleopCfg

from isaaclab.devices.openxr import XrCfg

from isaaclab_teleop import XrCfg

from isaaclab.devices.openxr import ManusVive

from isaaclab_teleop import IsaacTeleopDevice (with Manus plugin configured)

from isaaclab.devices import RetargeterBase

Use Isaac Teleop BaseRetargeter and pipeline builder pattern

from isaaclab.devices.openxr.retargeters import Se3AbsRetargeter

from isaacteleop.retargeters import Se3AbsRetargeter

Environment Configuration Changes#

The teleop_devices field with OpenXRDeviceCfg has been replaced by the isaac_teleop field with IsaacTeleopCfg and a pipeline builder callable.

Before (Isaac Lab 2.x):

from isaaclab.devices import DevicesCfg, OpenXRDeviceCfg
from isaaclab.devices.openxr import XrCfg
from isaaclab.devices.openxr.retargeters import Se3AbsRetargeterCfg, GripperRetargeterCfg

@configclass
class MyEnvCfg(ManagerBasedRLEnvCfg):

    xr: XrCfg = XrCfg(anchor_pos=[0.0, 0.0, 0.0])

    teleop_devices: DevicesCfg = field(default_factory=lambda: DevicesCfg(
        handtracking=OpenXRDeviceCfg(
            xr_cfg=None,
            retargeters=[
                Se3AbsRetargeterCfg(bound_hand=0, zero_out_xy_rotation=True),
                GripperRetargeterCfg(bound_hand=0),
            ]
        ),
    ))

After (Isaac Lab 3.0):

from isaaclab_teleop import IsaacTeleopCfg, XrCfg

def _build_pipeline():
    from isaacteleop.retargeting_engine.deviceio_source_nodes import ControllersSource, HandsSource
    from isaacteleop.retargeting_engine.interface import OutputCombiner, ValueInput
    from isaacteleop.retargeters import (
        GripperRetargeter, GripperRetargeterConfig,
        Se3AbsRetargeter, Se3RetargeterConfig, TensorReorderer,
    )
    from isaacteleop.retargeting_engine.tensor_types import TransformMatrix

    controllers = ControllersSource(name="controllers")
    hands = HandsSource(name="hands")
    transform = ValueInput("world_T_anchor", TransformMatrix())
    t_controllers = controllers.transformed(transform.output(ValueInput.VALUE))

    se3 = Se3AbsRetargeter(Se3RetargeterConfig(input_device=ControllersSource.RIGHT), name="ee")
    c_se3 = se3.connect({ControllersSource.RIGHT: t_controllers.output(ControllersSource.RIGHT)})

    grip = GripperRetargeter(GripperRetargeterConfig(hand_side="right"), name="grip")
    c_grip = grip.connect({
        ControllersSource.RIGHT: t_controllers.output(ControllersSource.RIGHT),
        HandsSource.RIGHT: hands.output(HandsSource.RIGHT),
    })

    reorder = TensorReorderer(
        input_config={"ee": ["pos_x","pos_y","pos_z","quat_x","quat_y","quat_z","quat_w"],
                      "grip": ["gripper_value"]},
        output_order=["pos_x","pos_y","pos_z","quat_x","quat_y","quat_z","quat_w","gripper_value"],
        name="reorder", input_types={"ee": "array", "grip": "scalar"},
    )
    c_reorder = reorder.connect({"ee": c_se3.output("ee_pose"), "grip": c_grip.output("gripper_command")})
    return OutputCombiner({"action": c_reorder.output("output")})

@configclass
class MyEnvCfg(ManagerBasedRLEnvCfg):

    xr: XrCfg = XrCfg(anchor_pos=(0.0, 0.0, 0.0))

    def __post_init__(self):
        super().__post_init__()
        self.isaac_teleop = IsaacTeleopCfg(
            pipeline_builder=_build_pipeline,
            sim_device=self.sim.device,
            xr_cfg=self.xr,
        )

Backward Compatibility#

The old classes still exist and will issue DeprecationWarning when used:

  • isaaclab.devices.openxr.OpenXRDevice and OpenXRDeviceCfg

  • isaaclab.devices.openxr.ManusVive and ManusViveCfg

  • All retargeters under isaaclab.devices.openxr.retargeters

Deprecated retargeters have been moved to isaaclab_teleop.deprecated.openxr.retargeters for compatibility. These will be removed in a future release.

Need Help?#

If you encounter issues during migration:

  1. Check the IsaacLab GitHub Issues

  2. Review the CHANGELOG

  3. Join the community on Discord