Skip to content

Commit

Permalink
Add profiling
Browse files Browse the repository at this point in the history
  • Loading branch information
wensi-ai committed Feb 9, 2024
1 parent fcc5c0e commit bfc6a98
Show file tree
Hide file tree
Showing 15 changed files with 680 additions and 289 deletions.
53 changes: 53 additions & 0 deletions .github/workflows/profiling.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
name: Profiling

on:
push:
branches:
- og-develop

permissions:
# deployments permission to deploy GitHub pages website
deployments: write
# contents permission to update profiling contents in gh-pages branch
contents: write

concurrency:
group: ${{ github.workflow }}-${{ github.event_name == 'pull_request' && github.head_ref || github.sha }}
cancel-in-progress: true

jobs:
profiling:
name: Speed Profiling
runs-on: [self-hosted, linux, gpu, dataset-enabled]

defaults:
run:
shell: micromamba run -n omnigibson /bin/bash -leo pipefail {0}

steps:
- name: Fix home
run: echo "HOME=/root" >> $GITHUB_ENV

- name: Checkout source
uses: actions/checkout@v3

- name: Install dev requirements
run: pip install -r requirements-dev.txt

- name: Install
run: pip install -e .

- name: Run performance benchmark
run: bash scripts/profiling.sh

- name: Store benchmark result
uses: benchmark-action/github-action-benchmark@v1
with:
tool: 'customSmallerIsBetter'
output-file-path: output.json
benchmark-data-dir-path: profiling
fail-on-alert: false
alert-threshold: '200%'
github-token: ${{ secrets.GITHUB_TOKEN }}
comment-on-alert: true
auto-push: true
3 changes: 3 additions & 0 deletions omnigibson/macros.py
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,9 @@
# CANNOT be set at runtime
gm.GUI_VIEWPORT_ONLY = False

# Whether to use the viewer camera or not
gm.RENDER_VIEWER_CAMERA = True

# Do not suppress known omni warnings / errors, and also put omnigibson in a debug state
# This includes extra information for things such as object sampling, and also any debug
# logging messages
Expand Down
1 change: 1 addition & 0 deletions omnigibson/simulator.py
Original file line number Diff line number Diff line change
Expand Up @@ -1236,6 +1236,7 @@ def _init_stage(
position=np.array(m.DEFAULT_VIEWER_CAMERA_POS),
orientation=np.array(m.DEFAULT_VIEWER_CAMERA_QUAT),
)
self.viewer_visibility = gm.RENDER_VIEWER_CAMERA

def close(self):
"""
Expand Down
12 changes: 12 additions & 0 deletions omnigibson/systems/macro_particle_system.py
Original file line number Diff line number Diff line change
Expand Up @@ -1102,6 +1102,9 @@ class MacroPhysicalParticleSystem(PhysicalParticleSystem, MacroParticleSystem):
_particle_radius = None
_particle_offset = None

has_particles = False
has_refreshed = False

@classmethod
def initialize(cls):
# Run super method first
Expand Down Expand Up @@ -1189,6 +1192,12 @@ def get_particles_position_orientation(cls):
pos, ori = np.array([]).reshape(0, 3), np.array([]).reshape(0, 4)
return pos, ori

@classmethod
def update(cls):
if cls.has_particles and not cls.has_refreshed:
cls._refresh_particles_view()
cls.has_refreshed = True

@classmethod
def get_particles_local_pose(cls):
return cls.get_particles_position_orientation()
Expand Down Expand Up @@ -1344,6 +1353,9 @@ def generate_particles(
# Set the vels
cls.set_particles_velocities(lin_vels=velocities, ang_vels=angular_velocities)

if not cls.has_particles:
cls.has_particles = True

@classmethod
def create(cls, name, create_particle_template, particle_density, scale=None, **kwargs):
"""
Expand Down
4 changes: 4 additions & 0 deletions omnigibson/systems/system_base.py
Original file line number Diff line number Diff line change
Expand Up @@ -1233,6 +1233,10 @@ def is_physical_particle_system(system_name):
def get_system(system_name, force_active=True):
# Make sure scene exists
assert og.sim.scene is not None, "Cannot get systems until scene is imported!"
# Make sure prefixes preserve their double underscore
for prefix in SYSTEM_PREFIXES:
if f"{prefix}__" not in system_name:
system_name = system_name.replace(f"{prefix}_", f"{prefix}__")
# If system_name is not in REGISTERED_SYSTEMS, create from metadata
system = REGISTERED_SYSTEMS[system_name] if system_name in REGISTERED_SYSTEMS \
else _create_system_from_metadata(system_name=system_name)
Expand Down
3 changes: 1 addition & 2 deletions omnigibson/transition_rules.py
Original file line number Diff line number Diff line change
Expand Up @@ -799,9 +799,8 @@ def _generate_conditions(cls):
@classmethod
def transition(cls, object_candidates):
objs_to_remove = []

for diceable_obj in object_candidates["diceable"]:
system = get_system(f"diced_{diceable_obj.category}")
system = get_system(f"diced__{diceable_obj.category}")
system.generate_particles_from_link(diceable_obj, diceable_obj.root_link, check_contact=False, use_visual_meshes=False)

# Delete original object from stage.
Expand Down
83 changes: 83 additions & 0 deletions omnigibson/utils/profiling_utils.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,83 @@
import gym
import omnigibson as og
import os
import psutil
from pynvml.smi import nvidia_smi

from time import time
from omnigibson.envs.env_base import Environment

PROFILING_FIELDS = ["total time", "physics time", "render time", "non physics time", "get observation time", "task time", "action time"]

class ProfilingEnv(Environment):
def step(self, action):
try:
start = time()
# If the action is not a dictionary, convert into a dictionary
if not isinstance(action, dict) and not isinstance(action, gym.spaces.Dict):
action_dict = dict()
idx = 0
for robot in self.robots:
action_dim = robot.action_dim
action_dict[robot.name] = action[idx: idx + action_dim]
idx += action_dim
else:
# Our inputted action is the action dictionary
action_dict = action

# Iterate over all robots and apply actions
for robot in self.robots:
robot.apply_action(action_dict[robot.name])

# Run simulation step
sim_start = time()
if len(og.sim._objects_to_initialize) > 0:
og.sim.render()
super(type(og.sim), og.sim).step(render=True)
omni_time = (time() - sim_start) * 1e3

# Additionally run non physics things
og.sim._non_physics_step()

# Grab observations
obs = self.get_obs()

# Step the scene graph builder if necessary
if self._scene_graph_builder is not None:
self._scene_graph_builder.step(self.scene)

# Grab reward, done, and info, and populate with internal info
reward, done, info = self.task.step(self, action)
self._populate_info(info)

if done and self._automatic_reset:
# Add lost observation to our information dict, and reset
info["last_observation"] = obs
obs = self.reset()

# Increment step
self._current_step += 1

# collect profiling data
total_frame_time = (time() - start) * 1e3
og_time = total_frame_time - omni_time
# memory usage in GB
memory_usage = psutil.Process(os.getpid()).memory_info().rss / 1024 ** 3
# VRAM usage in GB
for gpu in nvidia_smi.getInstance().DeviceQuery()['gpu']:
found = False
for process in gpu['processes']:
if process['pid'] == os.getpid():
vram_usage = process['used_memory'] / 1024
found = True
break
if found:
break

ret = [total_frame_time, omni_time, og_time, memory_usage, vram_usage]
if self._current_step % 100 == 0:
print("total time: {:.3f} ms, Omni time: {:.3f} ms, OG time: {:.3f} ms, memory: {:.3f} GB, vram: {:.3f} GB.".format(*ret))

return obs, reward, done, info, ret
except:
raise ValueError(f"Failed to execute environment step {self._current_step} in episode {self._current_episode}")
3 changes: 2 additions & 1 deletion requirements-dev.txt
Original file line number Diff line number Diff line change
Expand Up @@ -7,4 +7,5 @@ mkdocs-material
mkdocs-material-extensions
mkdocstrings[python]
mkdocs-section-index
mkdocs-literate-nav
mkdocs-literate-nav
pynvml
134 changes: 0 additions & 134 deletions scripts/benchmark.html

This file was deleted.

Loading

0 comments on commit bfc6a98

Please sign in to comment.