This site has been retired. For up to date information, see handbook.gnome.org or gitlab.gnome.org.


[Home] [TitleIndex] [WordIndex

This page tries to collate a set of best practices to help you quickly get to the bottom of any performance problems you are having using Clutter.

/!\ the content is currently a work in progress /!\

1. Introducing the tools

1.1. Sysprof

Sysprof is a stochastic profiler which rapidly samples what line of code the CPU is executing so when you stop profiling it can build up a graph of which lines of code account for the most samples which can give a strong indication of what code should be optimized or at least if you can see what code is to blame you may be able to change something else to stop that codepath being hit at all.

A big advantage of sysprof is that you don't have to modify your application to use it and it packaged for most major Linux distributions.

One disadvantage is that is only considers the CPU, so if your Clutter application is GPU bound then the results from sysprof have little use. Another Problem is that the results from sysprof can be somewhat overwhelming and quite hard to analyse if you aren't familiar with the the internals of clutter or the other libraries involved in running your application.

Sysprof screenshot

1.2. UProf

UProf is a toolkit for adding domain specific instrumentation to a project. Clutter and Cogl use UProf to define timers and counters throughout the Clutter code that we think may be of interest when analyzing the performance of Clutter applications. We have timers around the redrawing code, the picking code around specific stages of the journal flushing code and more. UProf can provide a textual report at the end of an application run or you can use the ncurses ui to provide a live view of where time is being spent in your application.

To use uprof you can fetch the code from here: git clone git://github.com/rib/UProf.git and then, after building and installing it, configure Clutter with --enable-profile.

Unlike sysprof, the timing is not based on frequent sampling of the CPU and instead we programatically define where timing should start and stop. The timers measure the real, wall clock time that has elapsed and that means if you have GPU or IO work happening between timers then that will be reflected in the results.

Here is an example of uprof output:

context: Clutter

counters:
Name                            Total Per Frame 

Actor real-paint counter        3185  13        
Actor pick-paint counter        632   2         
_clutter_backend_redraw counter 245   1         
_clutter_do_pick counter        79    0         
glsl vertex compile counter     20    0         
arbfp compile counter           2     0         

timers:
Name                                      Total   Per Frame  Percent                                                
                                          msecs                                                                     

Mainloop                                  2157.99 8.81       100.000% █████████████████████████████████████████████ 
  Master Clock                            1925.95 7.86        89.247% ████████████████████████████████████████▏     
    Redrawing                             1132.60 4.62        52.484% ███████████████████████▌                      
      Painting actors                     926.40  3.78        42.929% ███████████████████▎                          
        Stage clear                       828.56  3.38        38.395% █████████████████▎                            
      glXSwapBuffers                      204.71  0.84         9.486% ████▎                                         
    Event Processing                      596.40  2.43        27.637% ████████████▍                                 
    Timelines Advancement                 191.67  0.78         8.882% ███▉                                          
  Picking                                 588.22  2.40        27.258% ████████████▎                                 
    Read Pixels                           339.82  1.39        15.747% ███████                                       
    Stage clear (pick)                    219.89  0.90        10.189% ████▌                                         
    Painting actors (pick mode)           15.20   0.06         0.704% ▎                                             
  Journal Flush                           52.64   0.21         2.440% █                                             
    flush: vbo+texcoords+material+entries 37.96   0.15         1.759% ▊                                             
      flush: texcoords+material+entries   36.73   0.15         1.702% ▊                                             
        flush: material+entries           34.77   0.14         1.611% ▋                                             
          flush: modelview+entries        30.57   0.12         1.417% ▋                                             
  Journal Log                             4.29    0.02         0.199%                                               
  Material Flush                          3.94    0.02         0.183%                                               
  Layouting                               1.05    0.00         0.049%                                               
  Mainloop Idle                           0.99    0.00         0.046%                                               
  _cogl_material_equal                    0.23    0.00         0.011%                

Follows a brief explanation of some of the items in the UProf report:

1.3. Clutter and Cogl environment variables

Clutter and Cogl come with quite a lot of options that can be tuned at runtime using the CLUTTER_DEBUG, CLUTTER_PAINT and COGL_DEBUG environment variables. A lot of these options will enable tracing of various sub-systems, but other options may disable a set of functionality so that you can eliminate them from the equation to decide if they are the cause of a given problem.

To use these options you need to build clutter with --enable-debug and --enable-cogl-debug. To get a feel for the Cogl debug options available you can export COGL_DEBUG=help before running a Clutter application and you will see the following list:

     Supported debug values:
                     handle: debug ref counting issues for Cogl objects
                    slicing: debug the creation of texture slices
                      atlas: debug texture atlas management
              blend-strings: debug blend-string parsing
                    journal: view all geometry passing through the journal
                   batching: show how geometry is being batched in the journal
                   matrices: trace all matrix manipulation
                       draw: misc tracing of some drawing operations
                      pango: trace the pango renderer
             texture-pixmap: trace the Cogl texture pixmap backend
                 rectangles: add wire outlines for all rectangular geometry
           disable-batching: disable the journal batching
               disable-vbos: disable use of OpenGL vertex buffer objects
               disable-pbos: disable use of OpenGL pixel buffer objects
  disable-software-transform use the GPU to transform rectangular geometry
           dump-atlas-image: dump atlas changes to an image file
              disable-atlas: disable texture atlasing
          disable-texturing: disable texturing primitives
              disable-arbfp: disable use of ARBfp
               disable-glsl: disable use of GLSL
           disable-blending: disable use of blending
                show-source: show generated ARBfp/GLSL
                     opengl: traces some select OpenGL calls
                  offscreen: debug offscreen support

Of particular interest when debugging performance issues are the disable-XYZ options. If you can disable a broad range of functionality and understand how it affects your program that can give some big hints about your problem.

/!\ TODO: explain in more detail what each option really does, and how you can expect them to affect a Clutter application.

2. Ideas about methodology

It's hard to really define the one true methodology for finding the root cause of all performance problems so instead here we'll try and collate some of the insight we have gained profiling Clutter applications and probably others may have other good ideas. Please add your ideas if they aren't currently covered.

2.1. Understand if you are CPU, GPU or IO bound

A good place to start when getting to the bottom of a performance issue to to understand if it primarily relates to a CPU, GPU or IO bottlneck. A simple way to indicate if you have a CPU bottlneck is just to run top and look at the CPU% column for your application. If top shows that your application isn't busy then you can expect you have a GPU or IO problem.

Currently I don't know of a convenient way to determine if an application is IO bound vs GPU bound, but usually that can be guessed with some understanding of the application being debugged. At some point we should add a debug option to simply NOP all rendering and which would give an easy way to eliminate the GPU from the equation but we don't have that currently.

You should only use sysprof if you are CPU bound!

2.2. Are you performing redundant clears?

For immediate mode renderers, redundant clears can be a big problem. When Clutter paints the stage for each frame the first thing it normally does is clear the stage according to the current stage color. If you have an application that is actually drawing over every pixel of the stage though this clear is redundant and actually just wasting resources. For Clutter 1.4 we've introduced a new API: clutter_stage_set_no_clear_hint. You should use this to tell clutter that you will be covering every pixel of the stage so there's no need to do the clear.

2.3. Is your geometry being batched?

Most Clutter scenes are basically comprised of lots of textured rectangles. For simple rectangles we implement some special purpose batching to try and reduce the number of OpenGL state changes and draw calls we do because OpenGL drivers don't typically cope well if you use separate draw calls for such tiny primitives. The component that handles this batching is called the "Cogl Journal".

The Cogl Journal batching depends on rectangles being drawn sequentially with similar state (It wont re-order your geometry to achieve batching). If you submit 10 red rectangles (or more specifically 10 rectangles that use the same CoglMaterial state) then they will be batched into one vertex buffer and we will make just one draw call.

Some state changes immediately act as a synchronization point and result in the Journal being flushed. Examples are; use of CoglVertexBuffer (Used for long runs of text); changes of depth testing state; fog state; switching to draw to a different framebuffer.

How to find out if your rectangles are being batched? The easiest way it to run your application with COGL_DEBUG=rectangles exported in your environment. The enables a visual debugging mode that will draw outlines around rectangles in different colors. Rectangles that are batched together will have the same color outline. Another option is to use COGL_DEBUG=batching. This debug option will print details about batching to the console. For test actors we see this:

BATCHING: journal len = 6
BATCHING:  vbo offset batch len = 6
BATCHING:   material batch len = 6
BATCHING:    modelview batch len = 6

which shows that the hands are drawn together. If wouldn't be so good it you see lots of "len = 1" lines instead.

2.4. See if you have extra journal flushes happening

In an ideal situation we would accumulate all the geometry for a full frame into the journal and only flush it once before performing a swap-buffers operation to display the contents of your back buffer. Because some state changes aren't currently trackable through the journal (For example changes to the stencil test functions) there are times when we forcibly flush the journal and that may impact the performance of your application. One fairly straightforward way to investigate this is to attach to your program with GDB and set a breakpoint on _cogl_journal_flush. Each time that is hit look at the backtrace and hopefully you can figure out from that what is causing extra flushes.

Eventually we expect that most state changes will be trackable through journal entries so that we can remove this problem but until then it's an important gotcha to look out for!

2.5. Understand that profiling applications running under a compositors is hard!

There are a number of facets that make profiling applications under a compositor hard, but one of the biggest problems to be aware of is that current compositors provide no synchronization between client and compositor rendering. This means there can be a big discrepancy between the frame rate of the compositor and of your application. In one extreme this can result with you running a client faster than 60fps and starving the compositor of resources to be able to render at 60fps, or vice versa a compositor that is rendering too fast or requiring a lot of resources to render may starve your application of resources.

There is ongoing work to synchronize Clutter clients running under Mutter, but as of February 2013 this hasn't been finished yet.

2.6. Too many actors

Clutter will avoid drawing actors unnecessarily by tracking where in the screen each actor draws and skipping (culling) those that aren't visible. In order to be able to do so, Clutter queries each actor's "paint volume", but it's not completely trivial so may be good to check that your actors are reporting correctly their paint volumes and are skipped when completely hidden. This can be done by running with CLUTTER_DEBUG=clipping, which will tell what actors are being culled or not, and why. If this isn't working as expected, the paint volumes can be visualized with CLUTTER_PAINT=paint-volumes.

Even when culling is working as expected, excessive unneeded actors can be still a performance problem because of overhead due to allocation in containers, generic book-keeping and memory consumption.

Sometimes calculating the paint-volume of an actor can be too expensive and it may be more convenient for containers to skip hidden actors in their ::paint implementation and maybe in ::allocate.

Actor creation can be expensive as well. If you are going to have several hundreds of entities that can be expressed as actors, consider packing the data inside a ClutterListModel and using a container that fetches that data lazily.


2024-10-23 10:58