Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add body_incoming_joint_wrench_b to ArticulationData field #2128

Open
wants to merge 5 commits into
base: main
Choose a base branch
from
Open
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -74,6 +74,7 @@ def __init__(self, root_physx_view: physx.ArticulationView, device: str):
self._joint_pos = TimestampedBuffer()
self._joint_acc = TimestampedBuffer()
self._joint_vel = TimestampedBuffer()
self._body_incoming_joint_wrench_b = TimestampedBuffer()

def update(self, dt: float):
# update the simulation timestamp
@@ -509,6 +510,22 @@ def body_acc_w(self):
self._body_acc_w.timestamp = self._sim_timestamp
return self._body_acc_w.data

@property
def body_incoming_joint_wrench_b(self) -> torch.Tensor:
"""Joint reaction wrench applied from body parent to child body in parent body frame.
Shape is (num_instances, num_bodies, 6). All body reaction wrenches are provided including the root body to the
Copy link
Contributor

@Mayankm96 Mayankm96 Mar 22, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe useful to link to this documentation

https://nvidia-omniverse.github.io/PhysX/physx/5.5.1/docs/Articulations.html#link-incoming-joint-force

Also the wording seems slightly confusing. The forces are in CoM frame or link frame? We have been using "_b" for link frame so if it not the latter, we should use something else I feel.

NIT: body joint reaction wrench seems a strange name to call it? Might be simpler to call it body_reaction_wrench or body_incoming_wrench 🤔

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah documentation link is good. The output is the force/torque at a joint in the bodies link frame orientation (need to double check if it's the parent or child body).

Yeah I don't really like it either. I went back and forth on the naming. There are as many wrenches as bodies but they are the result of wrenches imparted at joints. I'm down for other names.

Copy link
Collaborator Author

@jtigue-bdai jtigue-bdai Mar 24, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

what about body_incoming_joint_wrench_b that way it stays a bit more in convention with the physx documentation?

world of an articulation.
`Physx documentation` https://nvidia-omniverse.github.io/PhysX/physx/5.5.1/docs/Articulations.html#link-incoming-joint-force
`Physics tensors API`: https://docs.omniverse.nvidia.com/kit/docs/omni_physics/latest/extensions/runtime/source/omni.physics.tensors/docs/api/python.html#omni.physics.tensors.impl.api.ArticulationView.get_link_incoming_joint_force
"""

if self._body_incoming_joint_wrench_b.timestamp < self._sim_timestamp:
self._body_incoming_joint_wrench_b.data = self._root_physx_view.get_link_incoming_joint_force()
self._body_incoming_joint_wrench_b.time_stamp = self._sim_timestamp
return self._body_incoming_joint_wrench_b.data

@property
def projected_gravity_b(self):
"""Projection of the gravity direction on base frame. Shape is (num_instances, 3)."""
4 changes: 2 additions & 2 deletions source/isaaclab/isaaclab/envs/mdp/observations.py
Original file line number Diff line number Diff line change
@@ -181,8 +181,8 @@ def body_incoming_wrench(env: ManagerBasedEnv, asset_cfg: SceneEntityCfg) -> tor
# extract the used quantities (to enable type-hinting)
asset: Articulation = env.scene[asset_cfg.name]
# obtain the link incoming forces in world frame
link_incoming_forces = asset.root_physx_view.get_link_incoming_joint_force()[:, asset_cfg.body_ids]
return link_incoming_forces.view(env.num_envs, -1)
body_incoming_joint_wrench_b = asset.data.body_incoming_joint_wrench_b[:, asset_cfg.body_ids]
return body_incoming_joint_wrench_b.view(env.num_envs, -1)


def imu_orientation(env: ManagerBasedEnv, asset_cfg: SceneEntityCfg = SceneEntityCfg("imu")) -> torch.Tensor:
99 changes: 97 additions & 2 deletions source/isaaclab/test/assets/test_articulation.py
Original file line number Diff line number Diff line change
@@ -96,10 +96,15 @@ def generate_articulation_cfg(
velocity_limit_sim=velocity_limit_sim,
effort_limit=effort_limit,
velocity_limit=velocity_limit,
stiffness=0.0,
damping=10.0,
stiffness=2000.0,
damping=100.0,
),
},
init_state=ArticulationCfg.InitialStateCfg(
pos=(0.0, 0.0, 0.0),
joint_pos=({"RevoluteJoint": 1.5708}),
rot=(0.7071055, 0.7071081, 0, 0),
),
)
elif articulation_type == "single_joint_explicit":
# we set 80.0 default for max force because default in USD is 10e10 which makes testing annoying.
@@ -1509,6 +1514,96 @@ def test_write_root_state(self):
elif state_location == "link":
torch.testing.assert_close(rand_state, articulation.data.root_link_state_w)

def test_body_incoming_joint_wrench_b_single_joint(self):
"""Test the data.body_incoming_joint_wrench_b buffer is populated correctly and statically correct for single joint."""
for num_articulations in (2, 1):
for device in ("cpu", "cuda:0"):
print(num_articulations, device)
with self.subTest(num_articulations=num_articulations, device=device):
with build_simulation_context(
gravity_enabled=True, device=device, add_ground_plane=False, auto_add_lighting=True
) as sim:
sim._app_control_on_stop_handle = None
articulation_cfg = generate_articulation_cfg(articulation_type="single_joint_implicit")
articulation, _ = generate_articulation(
articulation_cfg=articulation_cfg, num_articulations=num_articulations, device=device
)

# Play the simulator
sim.reset()
# apply external force
external_force_vector_b = torch.zeros(
(num_articulations, articulation.num_bodies, 3), device=device
)
external_force_vector_b[:, 1, 1] = 10.0 # 10 N in Y direction
external_torque_vector_b = torch.zeros(
(num_articulations, articulation.num_bodies, 3), device=device
)
external_torque_vector_b[:, 1, 2] = 10.0 # 10 Nm in z direction

# apply action to the articulation
joint_pos = torch.ones_like(articulation.data.joint_pos) * 1.5708 / 2.0
articulation.write_joint_state_to_sim(
torch.ones_like(articulation.data.joint_pos), torch.zeros_like(articulation.data.joint_vel)
)
articulation.set_joint_position_target(joint_pos)
articulation.write_data_to_sim()
for _ in range(50):
articulation.set_external_force_and_torque(
forces=external_force_vector_b, torques=external_torque_vector_b
)
articulation.write_data_to_sim()
# perform step
sim.step()
# update buffers
articulation.update(sim.cfg.dt)

# check shape
self.assertEqual(
articulation.data.body_incoming_joint_wrench_b.shape,
(num_articulations, articulation.num_bodies, 6),
)

# calculate expected static
mass = articulation.data.default_mass
pos_w = articulation.data.body_pos_w
quat_w = articulation.data.body_quat_w

mass_link2 = mass[:, 1].view(num_articulations, -1)
gravity = (
torch.tensor(sim.cfg.gravity, device="cpu")
.repeat(num_articulations, 1)
.view((num_articulations, 3))
)

# NOTE: the com and link pose for single joint are colocated
weight_vector_w = mass_link2 * gravity
# expected wrench from link mass and external wrench
expected_wrench = torch.zeros((num_articulations, 6), device=device)
expected_wrench[:, :3] = math_utils.quat_apply(
math_utils.quat_conjugate(quat_w[:, 0, :]),
weight_vector_w.to(device)
+ math_utils.quat_apply(quat_w[:, 1, :], external_force_vector_b[:, 1, :]),
)
expected_wrench[:, 3:] = math_utils.quat_apply(
math_utils.quat_conjugate(quat_w[:, 0, :]),
torch.cross(
pos_w[:, 1, :].to(device) - pos_w[:, 0, :].to(device),
weight_vector_w.to(device)
+ math_utils.quat_apply(quat_w[:, 1, :], external_force_vector_b[:, 1, :]),
dim=-1,
)
+ math_utils.quat_apply(quat_w[:, 1, :], external_torque_vector_b[:, 1, :]),
)

# check value of last joint wrench
torch.testing.assert_close(
expected_wrench,
articulation.data.body_incoming_joint_wrench_b[:, 1, :].squeeze(1),
atol=1e-2,
rtol=1e-3,
)


if __name__ == "__main__":
run_tests()