Inertial Measurement Unit (IMU)#
Inertial Measurement Units (IMUs) are a type of sensor for measuring the acceleration of an object. These sensors are traditionally designed report linear accelerations and angular velocities, and function on similar principles to that of a digital scale: They report accelerations derived from net force acting on the sensor.
A naive implementation of an IMU would report a negative acceleration due to gravity while the sensor is at rest in some local gravitational field. This is not generally needed for most practical applications, and so most real IMU sensors often include a gravity bias and assume that the device is operating on the surface of the Earth. The IMU we provide in Isaac Lab includes a similar bias term, which defaults to +g. This means that if you add an IMU to your simulation, and do not change this bias term, you will detect an acceleration of \(+ 9.81 m/s^{2}\) anti-parallel to gravity acceleration.
Consider a simple environment with an Anymal Quadruped equipped with an IMU on each of its two front feet.
##
from isaaclab_assets.robots.anymal import ANYMAL_C_CFG # isort: skip
@configclass
class ImuSensorSceneCfg(InteractiveSceneCfg):
"""Design the scene with IMU sensors on the robot."""
# ground plane
ground = AssetBaseCfg(prim_path="/World/defaultGroundPlane", spawn=sim_utils.GroundPlaneCfg())
# lights
dome_light = AssetBaseCfg(
prim_path="/World/Light", spawn=sim_utils.DomeLightCfg(intensity=3000.0, color=(0.75, 0.75, 0.75))
)
# robot
robot = ANYMAL_C_CFG.replace(prim_path="{ENV_REGEX_NS}/Robot")
imu_RF = ImuCfg(prim_path="{ENV_REGEX_NS}/Robot/LF_FOOT")
imu_LF = ImuCfg(prim_path="{ENV_REGEX_NS}/Robot/RF_FOOT")
def run_simulator(sim: sim_utils.SimulationContext, scene: InteractiveScene):
Here we have explicitly removed the bias from one of the sensors, and we can see how this affects the reported values by visualizing the sensor when we run the sample script
Notice that the right front foot explicitly has a bias of (0,0,0). In the visualization, you should see that the arrow indicating the acceleration from the right IMU rapidly changes over time, while the arrow visualizing the left IMU points constantly along the vertical axis.
Retrieving values form the sensor is done in the usual way
def run_simulator(sim: sim_utils.SimulationContext, scene: InteractiveScene):
.
.
.
# Simulate physics
while simulation_app.is_running():
.
.
.
# print information from the sensors
print("-------------------------------")
print(scene["imu_LF"])
print("Received linear velocity: ", scene["imu_LF"].data.lin_vel_b)
print("Received angular velocity: ", scene["imu_LF"].data.ang_vel_b)
print("Received linear acceleration: ", scene["imu_LF"].data.lin_acc_b)
print("Received angular acceleration: ", scene["imu_LF"].data.ang_acc_b)
print("-------------------------------")
print(scene["imu_RF"])
print("Received linear velocity: ", scene["imu_RF"].data.lin_vel_b)
print("Received angular velocity: ", scene["imu_RF"].data.ang_vel_b)
print("Received linear acceleration: ", scene["imu_RF"].data.lin_acc_b)
print("Received angular acceleration: ", scene["imu_RF"].data.ang_acc_b)
The oscillations in the values reported by the sensor are a direct result of of how the sensor calculates the acceleration, which is through a finite difference approximation between adjacent ground truth velocity values as reported by the sim. We can see this in the reported result (pay attention to the linear acceleration) because the acceleration from the right foot is small, but explicitly zero.
Imu sensor @ '/World/envs/env_.*/Robot/LF_FOOT':
view type : <class 'omni.physics.tensors.impl.api.RigidBodyView'>
update period (s) : 0.0
number of sensors : 1
Received linear velocity: tensor([[ 0.0203, -0.0054, 0.0380]], device='cuda:0')
Received angular velocity: tensor([[-0.0104, -0.1189, 0.0080]], device='cuda:0')
Received linear acceleration: tensor([[ 4.8344, -0.0205, 8.5305]], device='cuda:0')
Received angular acceleration: tensor([[-0.0389, -0.0262, -0.0045]], device='cuda:0')
-------------------------------
Imu sensor @ '/World/envs/env_.*/Robot/RF_FOOT':
view type : <class 'omni.physics.tensors.impl.api.RigidBodyView'>
update period (s) : 0.0
number of sensors : 1
Received linear velocity: tensor([[0.0244, 0.0077, 0.0431]], device='cuda:0')
Received angular velocity: tensor([[ 0.0122, -0.1360, -0.0042]], device='cuda:0')
Received linear acceleration: tensor([[-0.0018, 0.0010, -0.0032]], device='cuda:0')
Received angular acceleration: tensor([[-0.0373, -0.0050, -0.0053]], device='cuda:0')
-------------------------------
Code for imu_sensor.py
1# Copyright (c) 2022-2026, The Isaac Lab Project Developers (https://github.com/isaac-sim/IsaacLab/blob/main/CONTRIBUTORS.md).
2# All rights reserved.
3#
4# SPDX-License-Identifier: BSD-3-Clause
5
6"""Launch Isaac Sim Simulator first."""
7
8import argparse
9
10from isaaclab.app import AppLauncher
11
12# add argparse arguments
13parser = argparse.ArgumentParser(description="Example on using the IMU sensor.")
14parser.add_argument("--num_envs", type=int, default=1, help="Number of environments to spawn.")
15# append AppLauncher cli args
16AppLauncher.add_app_launcher_args(parser)
17# demos should open Kit visualizer by default
18parser.set_defaults(visualizer=["kit"])
19# parse the arguments
20args_cli = parser.parse_args()
21
22# launch omniverse app
23app_launcher = AppLauncher(args_cli)
24simulation_app = app_launcher.app
25
26"""Rest everything follows."""
27
28import torch
29import warp as wp
30
31import isaaclab.sim as sim_utils
32from isaaclab.assets import AssetBaseCfg
33from isaaclab.scene import InteractiveScene, InteractiveSceneCfg
34from isaaclab.sensors import ImuCfg
35from isaaclab.utils import configclass
36
37##
38# Pre-defined configs
39##
40from isaaclab_assets.robots.anymal import ANYMAL_C_CFG # isort: skip
41
42
43@configclass
44class ImuSensorSceneCfg(InteractiveSceneCfg):
45 """Design the scene with IMU sensors on the robot."""
46
47 # ground plane
48 ground = AssetBaseCfg(prim_path="/World/defaultGroundPlane", spawn=sim_utils.GroundPlaneCfg())
49
50 # lights
51 dome_light = AssetBaseCfg(
52 prim_path="/World/Light", spawn=sim_utils.DomeLightCfg(intensity=3000.0, color=(0.75, 0.75, 0.75))
53 )
54
55 # robot
56 robot = ANYMAL_C_CFG.replace(prim_path="{ENV_REGEX_NS}/Robot")
57
58 imu_RF = ImuCfg(prim_path="{ENV_REGEX_NS}/Robot/LF_FOOT")
59
60 imu_LF = ImuCfg(prim_path="{ENV_REGEX_NS}/Robot/RF_FOOT")
61
62
63def run_simulator(sim: sim_utils.SimulationContext, scene: InteractiveScene):
64 """Run the simulator."""
65 # Define simulation stepping
66 sim_dt = sim.get_physics_dt()
67 sim_time = 0.0
68 count = 0
69
70 # Simulate physics
71 while simulation_app.is_running():
72 if count % 500 == 0:
73 # reset counter
74 count = 0
75 # reset the scene entities
76 root_pose = wp.to_torch(scene["robot"].data.default_root_pose).clone()
77 root_pose[:, :3] += scene.env_origins
78 scene["robot"].write_root_link_pose_to_sim_index(root_pose=root_pose)
79 root_vel = wp.to_torch(scene["robot"].data.default_root_vel).clone()
80 scene["robot"].write_root_com_velocity_to_sim_index(root_velocity=root_vel)
81 # set joint positions with some noise
82 joint_pos, joint_vel = (
83 wp.to_torch(scene["robot"].data.default_joint_pos).clone(),
84 wp.to_torch(scene["robot"].data.default_joint_vel).clone(),
85 )
86 joint_pos += torch.rand_like(joint_pos) * 0.1
87 scene["robot"].write_joint_position_to_sim_index(position=joint_pos)
88 scene["robot"].write_joint_velocity_to_sim_index(velocity=joint_vel)
89 # clear internal buffers
90 scene.reset()
91 print("[INFO]: Resetting robot state...")
92 # Apply default actions to the robot
93 targets = wp.to_torch(scene["robot"].data.default_joint_pos)
94 scene["robot"].set_joint_position_target_index(target=targets)
95 scene.write_data_to_sim()
96 # perform step
97 sim.step()
98 # update sim-time
99 sim_time += sim_dt
100 count += 1
101 # update buffers
102 scene.update(sim_dt)
103
104 # print information from the sensors
105 print("-------------------------------")
106 print(scene["imu_LF"])
107 print("Received angular velocity: ", scene["imu_LF"].data.ang_vel_b)
108 print("Received linear acceleration: ", scene["imu_LF"].data.lin_acc_b)
109 print("-------------------------------")
110 print(scene["imu_RF"])
111 print("Received angular velocity: ", scene["imu_RF"].data.ang_vel_b)
112 print("Received linear acceleration: ", scene["imu_RF"].data.lin_acc_b)
113
114
115def main():
116 """Main function."""
117
118 # Initialize the simulation context
119 sim_cfg = sim_utils.SimulationCfg(dt=0.005, device=args_cli.device)
120 sim = sim_utils.SimulationContext(sim_cfg)
121 # Set main camera
122 sim.set_camera_view(eye=[3.5, 3.5, 3.5], target=[0.0, 0.0, 0.0])
123 # design scene
124 scene_cfg = ImuSensorSceneCfg(num_envs=args_cli.num_envs, env_spacing=2.0)
125 scene = InteractiveScene(scene_cfg)
126 # Play the simulator
127 sim.reset()
128 # Now we are ready!
129 print("[INFO]: Setup complete...")
130 # Run the simulator
131 run_simulator(sim, scene)
132
133
134if __name__ == "__main__":
135 # run the main function
136 main()
137 # close sim app
138 simulation_app.close()