Gazebo Common

API Reference

7.0.0
Profiler

Next Tutorial: Hardware-accelerated Video Encoding

Overview

This tutorial describes how to get started using the Gazebo Common profiler to measure and visualize run-time performance of your software.

The gz::common::Profiler provides a common interface that can allow for multiple underlying profiler implementations. Currently, the only available implementation is Remotery.

The goal of the profiler is to provide introspection and analysis when enabled at compile time, but to introduce no overhead when it is disabled at compile-time.

To control if the profiler is enabled, set the GZ_PROFILER_ENABLE flag using cmake on the targets or sources that you are interested in (described below).

Enabling the Profiler

On custom example

In order to use the profiler, inspection points must be added to the source code, and the application or library must be linked to the gz-common::profiler component.

To start, create a new CMake project directory and navigate to it:

mkdir -p ~/profiler_example && cd ~/profiler_example

Next, download the profiler.cc example into the project:

wget https://raw.githubusercontent.com/gazebosim/gz-common/gz-common7/examples/profiler.cc

The relevant corresponding C++ would be as follows:

// Add the profiler header
...
void thread(const char *_thread_name)
{
// Sets the name of the thread to appear in the UI
GZ_PROFILE_THREAD_NAME(_thread_name);
while (running)
{
// Add a profiling point to this scope.
GZ_PROFILE("Loop");
// Execute some arbitrary tasks
for (size_t ii = 0; ii < 10; ++ii)
{
task1();
}
task2();
task3();
}
}

Create a new CMakeLists.txt at the root of the project directory (~/profiler_example). Note that the profiler must be enabled at compile time in order to function.

cmake_minimum_required(VERSION 3.22.1 FATAL_ERROR)
# Find the gz-common library
find_package(gz-common QUIET REQUIRED COMPONENTS profiler)
add_executable(profiler_example profiler.cc)
target_link_libraries(profiler_example gz-common::profiler)
# Enable the profiler for the example
target_compile_definitions(profiler_example PUBLIC "GZ_PROFILER_ENABLE=1")

Run cmake and build the example

mkdir -p build && cd build
cmake ..
make profiler_example

Then use two terminals to execute the example and the profiler visualization:

From terminal 1, inside ~/profiler_example/build:

./profiler_example

From terminal 2, use one of the following commands, depending on your configuration:

  • If you installed gz-common as a package/binary:
    • Open the HTML file directly via a convenience script:

      {.sh} $(find /usr -type f -name 'gz_remotery_vis')

    • Or, if you're running inside a Docker container, start a server:

      {.sh} python3 -m http.server -d $(find / -type d -name 'profiler_vis')

  • If you installed gz-common from source, we assume it is located at $SOURCE_DIR/gz-common, where $SOURCE_DIR is a variable representing a file path
    • Open the HTML file directly:

      {.sh} xdg-open $SOURCE_DIR/gz-common/profiler/src/Remotery/vis/index.html

    • Or, if you're running inside a Docker container, start a server:

      {.sh} python3 -m http.server -d $SOURCE_DIR/gz-common/profiler/src/Remotery/vis

If you are running the profiler visualization as a server, the command will display an output Serving HTTP on [IP_ADDRESS] port 8000 (http://[IP_ADDRESS]:8000/) ... - navigate to that URL in your browser and you should see the profiler displayed.

On Gazebo library

If you want to use profiler on any other Gazebo library, enable the profiler at compile time with ENABLE_PROFILER cmake argument.

When compiling with CMake:

cmake .. -DENABLE_PROFILER=1

When compiling with colcon:

colcon build --cmake-args -DENABLE_PROFILER=1

Run your Gazebo library then go to your Gazebo installation path and open the profiler browser using:

<workspace>/install/libexec/gz/gz-common<N>/gz_remotery_vis

If the profiler is run successfully, you should see output in a browser. Similar to this

Troubleshoot the web viewer

If you see connection error, there are a couple of things to double check

  1. Was the profiler enabled when the project you're trying to run was compiled? Note that this isn't the case if you installed Gazebo libraries from binaries, for example. You need to compile the project from source with the ENABLE_PROFILER variable set.
  2. Are you using the correct port number in the upper left corner Connection Addresss: ws://127.0.0.1:1500/rmt? Running gz sim -v 4 will show the port number in use near the top of the outputted text. The port number will be printed out if the profiler is enabled.
    [Dbg] [RemoteryProfilerImpl.cc:187] Starting gz-common profiler impl: Remotery (port: 1500)
  3. Are you running the program in a separate terminal? The profiler only establishes connection if there is a program running and being actively profiled.
  4. If you want to use a different port, configure the environment variable RMT_PORT by running the following in terminal, and update the web viewer port in your browser accordingly (see 2 above)
    export RMT_PORT=1500

Using the Profiler

The profiler is used through a series of macros.

The two primary ways of profiling a section of code are to either use a matched pair of GZ_PROFILE_BEGIN and GZ_PROFILE_END macros, or to use a single RAII-style macro GZ_PROFILE. The RAII style will stop measuring once the scope that the macro was invoked in is left.

Using begin/end:

// An example of using start/stop profiling.

Using RAII-style:

{
// An example of using scope-based profiling.
GZ_PROFILE("a");
}

Additionally, each thread can be given a name for easy reference in the UI:

Configuring the Profiler

Specific profiler implementations may have further configuration options available.

Configuring Remotery

Remotery can additionally be configured via environment variables. Most users should not need to change these for their applications.

  • RMT_PORT: Port to listen for incoming connections on.
  • RMT_QUEUE_SIZE: Size of the internal message queues
  • RMT_MSGS_PER_UPDATE: Upper limit on messages consumed per loop
  • RMT_SLEEP_BETWEEN_UPDATES: Controls profile server update rate.

These directly set the corresponding parameters in the rmtSettings structure. For more information, consult the Remotery source