Skip to content

Headless Rendering

Polyscope supports headless rendering on linux machines which do not have displays, such as remote servers and clusters.

Polyscope’s default interactive backend uses GLFW to create an OpenGL context, which essentially requires that a physical display be attached for any rendering to be performed. This would prevent workflows like running compute jobs on remote clusters and logging images or saving videos rendered via Polyscope—headless rendering offers a solution.

What is headless rendering not?

“Headless” just means that Polyscope can run its rendering engine even when no display is present, rendering images to buffers or files. You can save them, store them, or even log them to viewers like Tensorboard.

Headless rendering is not the same as running the UI in a browser tab or a cell of a Jupyter notebook. There is currently no support for either of these uses (although local Jupyter notebooks work; the window will be opened outside of the notebook).

Also, headless rendering is not necessary if you are remotely connecting to a machine which has a physical display attached; for instance, when remotely connecting to macOS or Windows machines with displays.

Unix only

Headless rendering is only supported on Unix/Linux, because EGL is only available there.

Using headless rendering

Headless rendering requires compiling with and initializing a separate EGL backend.

Fortunately, the linux Polyscope wheels published on pip are already built with support for EGL.

Your system must also have EGL available at runtime. It is likely packaged with graphics drivers, or can be installed from a package manager. Another alternative is to fall back on (much slower!) rendering by installing OSMesa from your package manager.

For developers: compiling with EGL support

If you are manually compiling Polyscope’s python bindings, you must ensure EGL support is enabled. The default CMake settings enable it only if you are on Linux and EGL headers seem to be present on your system.

Set the CMake variable POLYSCOPE_BACKEND_OPENGL3_EGL=ON to force-enable building with EGL support.

cmake .. -DPOLYSCOPE_BACKEND_OPENGL3_EGL=ON

At runtime, you must tell Polyscope to initialize with the EGL engine. The simplest way to do this is to specify it explicitly when you call init(). Alternately, the default automatic initialization scheme will select the EGL headless backend if no others are available, but only if you enable the headless backends option first.

Example: headless usage

import polyscope as ps

# explicit initialization
ps.init("openGL3_egl")
# OR
ps.set_allow_headless_backends(True)  # enable headless backends
ps.init() # automatic initialization will use the EGL backend
          # if it is available and there is no display

# ... add structures and quantities to polyscope as usual ...

# save screenshots, render movies, etc
ps.screenshot() 
viz_img = ps.screenshot_to_buffer() 

Once you have initialized the backend and added your data as usual, you can preform rendering operations such as saving rendered screenshots to disk or to buffer.

Unless you really know what you are doing, you probably do not want to call show() – by default this would initiate a render loop which spins endlessly with no effect. However it is still permitted to do so, e.g. if you set a user callback which has side effects.

ImGui not rendered

In headless mode, the ImGui UI is not actually drawn to the screen. You can still issue ImGui calls in your code if you would like, but they will never have any effect.

Options

is_headless()
is_headless()

Returns True if polyscope is currently running in headless mode. May only be called after initialization.

set_allow_headless_backends(b)
set_allow_headless_backends

If True default automatic initialization is allowed to initialize with a headless backend, if that is the only option.

Default: False

This is set to false by default, because sometimes misconfigured or low-capability machines will have no display, and automatically creating headless backends on such machines would confuse new users.

set_egl_device_index(i)
set_egl_device_index()

Pass an index to manually specify which EGL device to use for headless EGL rendering. If set to -1 (default), Polyscope will try all devices, with some heuristics to attempt to use hardware renderers before software renderers.

Default: -1

If you are trying to figure out which device index to use, set Polyscope’s verbosity to a large value (> 5). At high verbosity, Polyscope will print the list of found EGL devices and their indices when attempting initialization.