Source code for omni.isaac.lab.assets.deformable_object.deformable_object

# Copyright (c) 2022-2025, The Isaac Lab Project Developers.
# All rights reserved.
#
# SPDX-License-Identifier: BSD-3-Clause

from __future__ import annotations

import torch
from collections.abc import Sequence
from typing import TYPE_CHECKING

import omni.log
import omni.physics.tensors.impl.api as physx
from pxr import PhysxSchema, UsdShade

import omni.isaac.lab.sim as sim_utils
import omni.isaac.lab.utils.math as math_utils
from omni.isaac.lab.markers import VisualizationMarkers

from ..asset_base import AssetBase
from .deformable_object_data import DeformableObjectData

if TYPE_CHECKING:
    from .deformable_object_cfg import DeformableObjectCfg


[docs]class DeformableObject(AssetBase): """A deformable object asset class. Deformable objects are assets that can be deformed in the simulation. They are typically used for soft bodies, such as stuffed animals and food items. Unlike rigid object assets, deformable objects have a more complex structure and require additional handling for simulation. The simulation of deformable objects follows a finite element approach, where the object is discretized into a mesh of nodes and elements. The nodes are connected by elements, which define the material properties of the object. The nodes can be moved and deformed, and the elements respond to these changes. The state of a deformable object comprises of its nodal positions and velocities, and not the object's root position and orientation. The nodal positions and velocities are in the simulation frame. Soft bodies can be `partially kinematic`_, where some nodes are driven by kinematic targets, and the rest are simulated. The kinematic targets are the desired positions of the nodes, and the simulation drives the nodes towards these targets. This is useful for partial control of the object, such as moving a stuffed animal's head while the rest of the body is simulated. .. attention:: This class is experimental and subject to change due to changes on the underlying PhysX API on which it depends. We will try to maintain backward compatibility as much as possible but some changes may be necessary. .. _partially kinematic: https://nvidia-omniverse.github.io/PhysX/physx/5.4.1/docs/SoftBodies.html#kinematic-soft-bodies """ cfg: DeformableObjectCfg """Configuration instance for the deformable object."""
[docs] def __init__(self, cfg: DeformableObjectCfg): """Initialize the deformable object. Args: cfg: A configuration instance. """ super().__init__(cfg)
""" Properties """ @property def data(self) -> DeformableObjectData: return self._data @property def num_instances(self) -> int: return self.root_physx_view.count @property def num_bodies(self) -> int: """Number of bodies in the asset. This is always 1 since each object is a single deformable body. """ return 1 @property def root_physx_view(self) -> physx.SoftBodyView: """Deformable body view for the asset (PhysX). Note: Use this view with caution. It requires handling of tensors in a specific way. """ return self._root_physx_view @property def material_physx_view(self) -> physx.SoftBodyMaterialView | None: """Deformable material view for the asset (PhysX). This view is optional and may not be available if the material is not bound to the deformable body. If the material is not available, then the material properties will be set to default values. Note: Use this view with caution. It requires handling of tensors in a specific way. """ return self._material_physx_view @property def max_sim_elements_per_body(self) -> int: """The maximum number of simulation mesh elements per deformable body.""" return self.root_physx_view.max_sim_elements_per_body @property def max_collision_elements_per_body(self) -> int: """The maximum number of collision mesh elements per deformable body.""" return self.root_physx_view.max_elements_per_body @property def max_sim_vertices_per_body(self) -> int: """The maximum number of simulation mesh vertices per deformable body.""" return self.root_physx_view.max_sim_vertices_per_body @property def max_collision_vertices_per_body(self) -> int: """The maximum number of collision mesh vertices per deformable body.""" return self.root_physx_view.max_vertices_per_body """ Operations. """
[docs] def reset(self, env_ids: Sequence[int] | None = None): # Think: Should we reset the kinematic targets when resetting the object? # This is not done in the current implementation. We assume users will reset the kinematic targets. pass
[docs] def write_data_to_sim(self): pass
[docs] def update(self, dt: float): self._data.update(dt)
""" Operations - Write to simulation. """
[docs] def write_nodal_state_to_sim(self, nodal_state: torch.Tensor, env_ids: Sequence[int] | None = None): """Set the nodal state over selected environment indices into the simulation. The nodal state comprises of the nodal positions and velocities. Since these are nodes, the velocity only has a translational component. All the quantities are in the simulation frame. Args: nodal_state: Nodal state in simulation frame. Shape is (len(env_ids), max_sim_vertices_per_body, 6). env_ids: Environment indices. If None, then all indices are used. """ # set into simulation self.write_nodal_pos_to_sim(nodal_state[..., :3], env_ids=env_ids) self.write_nodal_velocity_to_sim(nodal_state[..., 3:], env_ids=env_ids)
[docs] def write_nodal_pos_to_sim(self, nodal_pos: torch.Tensor, env_ids: Sequence[int] | None = None): """Set the nodal positions over selected environment indices into the simulation. The nodal position comprises of individual nodal positions of the simulation mesh for the deformable body. The positions are in the simulation frame. Args: nodal_pos: Nodal positions in simulation frame. Shape is (len(env_ids), max_sim_vertices_per_body, 3). env_ids: Environment indices. If None, then all indices are used. """ # resolve all indices physx_env_ids = env_ids if env_ids is None: env_ids = slice(None) physx_env_ids = self._ALL_INDICES # note: we need to do this here since tensors are not set into simulation until step. # set into internal buffers self._data.nodal_pos_w[env_ids] = nodal_pos.clone() # set into simulation self.root_physx_view.set_sim_nodal_positions(self._data.nodal_pos_w, indices=physx_env_ids)
[docs] def write_nodal_velocity_to_sim(self, nodal_vel: torch.Tensor, env_ids: Sequence[int] | None = None): """Set the nodal velocity over selected environment indices into the simulation. The nodal velocity comprises of individual nodal velocities of the simulation mesh for the deformable body. Since these are nodes, the velocity only has a translational component. The velocities are in the simulation frame. Args: nodal_vel: Nodal velocities in simulation frame. Shape is (len(env_ids), max_sim_vertices_per_body, 3). env_ids: Environment indices. If None, then all indices are used. """ # resolve all indices physx_env_ids = env_ids if env_ids is None: env_ids = slice(None) physx_env_ids = self._ALL_INDICES # note: we need to do this here since tensors are not set into simulation until step. # set into internal buffers self._data.nodal_vel_w[env_ids] = nodal_vel.clone() # set into simulation self.root_physx_view.set_sim_nodal_velocities(self._data.nodal_vel_w, indices=physx_env_ids)
[docs] def write_nodal_kinematic_target_to_sim(self, targets: torch.Tensor, env_ids: Sequence[int] | None = None): """Set the kinematic targets of the simulation mesh for the deformable bodies indicated by the indices. The kinematic targets comprise of individual nodal positions of the simulation mesh for the deformable body and a flag indicating whether the node is kinematically driven or not. The positions are in the simulation frame. Note: The flag is set to 0.0 for kinematically driven nodes and 1.0 for free nodes. Args: targets: The kinematic targets comprising of nodal positions and flags. Shape is (len(env_ids), max_sim_vertices_per_body, 4). env_ids: Environment indices. If None, then all indices are used. """ # resolve all indices physx_env_ids = env_ids if env_ids is None: env_ids = slice(None) physx_env_ids = self._ALL_INDICES # store into internal buffers self._data.nodal_kinematic_target[env_ids] = targets.clone() # set into simulation self.root_physx_view.set_sim_kinematic_targets(self._data.nodal_kinematic_target, indices=physx_env_ids)
""" Operations - Helper. """
[docs] def transform_nodal_pos( self, nodal_pos: torch.tensor, pos: torch.Tensor | None = None, quat: torch.Tensor | None = None ) -> torch.Tensor: """Transform the nodal positions based on the pose transformation. This function computes the transformation of the nodal positions based on the pose transformation. It multiplies the nodal positions with the rotation matrix of the pose and adds the translation. Internally, it calls the :meth:`omni.isaac.lab.utils.math.transform_points` function. Args: nodal_pos: The nodal positions in the simulation frame. Shape is (N, max_sim_vertices_per_body, 3). pos: The position transformation. Shape is (N, 3). Defaults to None, in which case the position is assumed to be zero. quat: The orientation transformation as quaternion (w, x, y, z). Shape is (N, 4). Defaults to None, in which case the orientation is assumed to be identity. Returns: The transformed nodal positions. Shape is (N, max_sim_vertices_per_body, 3). """ # offset the nodal positions to center them around the origin mean_nodal_pos = nodal_pos.mean(dim=1, keepdim=True) nodal_pos = nodal_pos - mean_nodal_pos # transform the nodal positions based on the pose around the origin return math_utils.transform_points(nodal_pos, pos, quat) + mean_nodal_pos
""" Internal helper. """ def _initialize_impl(self): # create simulation view self._physics_sim_view = physx.create_simulation_view(self._backend) self._physics_sim_view.set_subspace_roots("/") # obtain the first prim in the regex expression (all others are assumed to be a copy of this) template_prim = sim_utils.find_first_matching_prim(self.cfg.prim_path) if template_prim is None: raise RuntimeError(f"Failed to find prim for expression: '{self.cfg.prim_path}'.") template_prim_path = template_prim.GetPath().pathString # find deformable root prims root_prims = sim_utils.get_all_matching_child_prims( template_prim_path, predicate=lambda prim: prim.HasAPI(PhysxSchema.PhysxDeformableBodyAPI) ) if len(root_prims) == 0: raise RuntimeError( f"Failed to find a deformable body when resolving '{self.cfg.prim_path}'." " Please ensure that the prim has 'PhysxSchema.PhysxDeformableBodyAPI' applied." ) if len(root_prims) > 1: raise RuntimeError( f"Failed to find a single deformable body when resolving '{self.cfg.prim_path}'." f" Found multiple '{root_prims}' under '{template_prim_path}'." " Please ensure that there is only one deformable body in the prim path tree." ) # we only need the first one from the list root_prim = root_prims[0] # find deformable material prims material_prim = None # obtain material prim from the root prim # note: here we assume that all the root prims have their material prims at similar paths # and we only need to find the first one. This may not be the case for all scenarios. # However, the checks in that case get cumbersome and are not included here. if root_prim.HasAPI(UsdShade.MaterialBindingAPI): # check the materials that are bound with the purpose 'physics' material_paths = UsdShade.MaterialBindingAPI(root_prim).GetDirectBindingRel("physics").GetTargets() # iterate through targets and find the deformable body material if len(material_paths) > 0: for mat_path in material_paths: mat_prim = root_prim.GetStage().GetPrimAtPath(mat_path) if mat_prim.HasAPI(PhysxSchema.PhysxDeformableBodyMaterialAPI): material_prim = mat_prim break if material_prim is None: omni.log.info( f"Failed to find a deformable material binding for '{root_prim.GetPath().pathString}'." " The material properties will be set to default values and are not modifiable at runtime." " If you want to modify the material properties, please ensure that the material is bound" " to the deformable body." ) # resolve root path back into regex expression # -- root prim expression root_prim_path = root_prim.GetPath().pathString root_prim_path_expr = self.cfg.prim_path + root_prim_path[len(template_prim_path) :] # -- object view self._root_physx_view = self._physics_sim_view.create_soft_body_view(root_prim_path_expr.replace(".*", "*")) # Return if the asset is not found if self._root_physx_view._backend is None: raise RuntimeError(f"Failed to create deformable body at: {self.cfg.prim_path}. Please check PhysX logs.") # resolve material path back into regex expression if material_prim is not None: # -- material prim expression material_prim_path = material_prim.GetPath().pathString # check if the material prim is under the template prim # if not then we are assuming that the single material prim is used for all the deformable bodies if template_prim_path in material_prim_path: material_prim_path_expr = self.cfg.prim_path + material_prim_path[len(template_prim_path) :] else: material_prim_path_expr = material_prim_path # -- material view self._material_physx_view = self._physics_sim_view.create_soft_body_material_view( material_prim_path_expr.replace(".*", "*") ) else: self._material_physx_view = None # log information about the deformable body omni.log.info(f"Deformable body initialized at: {root_prim_path_expr}") omni.log.info(f"Number of instances: {self.num_instances}") omni.log.info(f"Number of bodies: {self.num_bodies}") if self._material_physx_view is not None: omni.log.info(f"Deformable material initialized at: {material_prim_path_expr}") omni.log.info(f"Number of instances: {self._material_physx_view.count}") else: omni.log.info("No deformable material found. Material properties will be set to default values.") # container for data access self._data = DeformableObjectData(self.root_physx_view, self.device) # create buffers self._create_buffers() # update the deformable body data self.update(0.0) def _create_buffers(self): """Create buffers for storing data.""" # constants self._ALL_INDICES = torch.arange(self.num_instances, dtype=torch.long, device=self.device) # default state # we use the initial nodal positions at spawn time as the default state # note: these are all in the simulation frame nodal_positions = self.root_physx_view.get_sim_nodal_positions() nodal_velocities = torch.zeros_like(nodal_positions) self._data.default_nodal_state_w = torch.cat((nodal_positions, nodal_velocities), dim=-1) # kinematic targets self._data.nodal_kinematic_target = self.root_physx_view.get_sim_kinematic_targets() # set all nodes as non-kinematic targets by default self._data.nodal_kinematic_target[..., -1] = 1.0 """ Internal simulation callbacks. """ def _set_debug_vis_impl(self, debug_vis: bool): # set visibility of markers # note: parent only deals with callbacks. not their visibility if debug_vis: if not hasattr(self, "target_visualizer"): self.target_visualizer = VisualizationMarkers(self.cfg.visualizer_cfg) # set their visibility to true self.target_visualizer.set_visibility(True) else: if hasattr(self, "target_visualizer"): self.target_visualizer.set_visibility(False) def _debug_vis_callback(self, event): # check where to visualize targets_enabled = self.data.nodal_kinematic_target[:, :, 3] == 0.0 num_enabled = int(torch.sum(targets_enabled).item()) # get positions if any targets are enabled if num_enabled == 0: # create a marker below the ground positions = torch.tensor([[0.0, 0.0, -10.0]], device=self.device) else: positions = self.data.nodal_kinematic_target[targets_enabled][..., :3] # show target visualizer self.target_visualizer.visualize(positions) def _invalidate_initialize_callback(self, event): """Invalidates the scene elements.""" # call parent super()._invalidate_initialize_callback(event) # set all existing views to None to invalidate them self._physics_sim_view = None self._root_physx_view = None