/usr/share/qt5/doc/qtquick/qtquick-visualcanvas-scenegraph-renderer.html is in qtdeclarative5-doc-html 5.2.1-3ubuntu15.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 | <?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en_US" lang="en_US">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<!-- scenegraph.qdoc -->
<title>Qt Quick Scene Graph Renderer | QtQuick 5.2</title>
<link rel="stylesheet" type="text/css" href="style/offline.css" />
</head>
<body>
<div class="header" id="qtdocheader">
<div class="main">
<div class="main-rounded">
<div class="navigationbar">
<ul>
<li>Qt 5.2</li>
<li><a href="qtquick-index.html">Qt Quick</a></li>
<li>Qt Quick Scene Graph Renderer</li>
<li id="buildversion">
Qt 5.2.1 Reference Documentation</li>
</ul>
</div>
</div>
<div class="content">
<div class="line">
<div class="content mainContent">
<div class="toc">
<h3><a name="toc">Contents</a></h3>
<ul>
<li class="level1"><a href="#batching">Batching</a></li>
<li class="level2"><a href="#opaque-primitives">Opaque Primitives</a></li>
<li class="level2"><a href="#alpha-blended-primitives">Alpha Blended Primitives</a></li>
<li class="level2"><a href="#mixing-with-3d-primitives">Mixing with 3D primitives</a></li>
<li class="level2"><a href="#texture-atlas">Texture Atlas</a></li>
<li class="level1"><a href="#batch-roots">Batch Roots</a></li>
<li class="level2"><a href="#transform-nodes">Transform Nodes</a></li>
<li class="level2"><a href="#clipping">Clipping</a></li>
<li class="level2"><a href="#vertex-buffers">Vertex Buffers</a></li>
<li class="level1"><a href="#antialiasing">Antialiasing</a></li>
<li class="level2"><a href="#vertex-antialiasing">Vertex Antialiasing</a></li>
<li class="level2"><a href="#multisample-antialiasing">Multisample Antialiasing</a></li>
<li class="level1"><a href="#performance">Performance</a></li>
</ul>
</div>
<h1 class="title">Qt Quick Scene Graph Renderer</h1>
<span class="subtitle"></span>
<!-- $$$qtquick-visualcanvas-scenegraph-renderer.html-description -->
<div class="descr"> <a name="details"></a>
<p>This document explains how the scene graph renderer works internally so that one can write code that uses it in an optimal fashion, both performance-wise and feature-wise.</p>
<p>One does not need to understand the internals of the renderer to get good performance. However, it might help when integrating with the scene graph or to figure out why it is not possible to squeeze the maximum efficiency out of the graphics chip.</p>
<p><b>Note: </b>Even in the case where every frame is unique and everything is uploaded from scratch, the default renderer will perform well.</p><p>The Qt Quick items in a QML scene populates a tree of <a href="qsgnode.html">QSGNode</a> instances. Once created, this tree is a complete description of how a certain frame should be rendered. It does not contain any references back to the Qt Quick items at all and will on most platforms be processed and rendered in a separate thread. The renderer is a self contained part of the scene graph which traverses the <a href="qsgnode.html">QSGNode</a> tree and uses geometry defined in <a href="qsggeometrynode.html">QSGGeometryNode</a> and shader state defined in <a href="qsgmaterial.html">QSGMaterial</a> to schedule OpenGL state change and draw calls.</p>
<p>If needed, the renderer can be completely replaced using the internal scene graph back-end API. This is mostly interesting for platform vendors who wish to take advantage of non-standard hardware features. For majority of use cases, the default renderer will be sufficient.</p>
<p>The default renderer focuses on two primary strategies to optimize the rendering. Batching of draw calls and retention of geometry on the GPU.</p>
<a name="batching"></a>
<h2>Batching</h2>
<p>Where a traditional 2D API, such as QPainter, Cairo or <a href="qml-qtquick-context2d.html">Context2D</a>, is written to handle thousands of individual draw calls per frame, OpenGL is a pure hardware API and performs best when the number of draw calls is very low and state changes are kept to a minimum. Consider the following use case:</p>
<p class="centerAlign"><img src="images/visualcanvas_list.png" alt="" /></p><p>The simplest way of drawing this list is on a cell-by-cell basis. First the background is drawn. This is a rectangle of a specific color. In OpenGL terms this means selecting a shader program to do solid color fills, setting up the fill color, setting the transformation matrix containing the x and y offsets and then using for instance <tt>glDrawArrays</tt> to draw two triangles making up the rectangle. The icon is drawn next. In OpenGL terms this means selecting a shader program to draw textures, selecting the active texture to use, setting the transformation matrix, enabling alpha-blending and then using for instance <tt>glDrawArrays</tt> to draw the two triangles making up the bounding rectangle of the icon. The text and separator line between cells follow a similar pattern. And this process is repeated for every cell in the list, so for a longer list, the overhead imposed by OpenGL state changes and draw calls completely outweighs the benefit that using a hardware accelerated API could provide.</p>
<p>When each primitive is large, this overhead is negligible, but in the case of a typical UI, there are many small items which add up to a considerable overhead.</p>
<p>The default scene graph renderer works within these limitations and will try to merge individual primitives together into batches while preserving the exact same visual result. The result is fewer OpenGL state changes and a minimal amount of draw calls, resulting in optimal performance.</p>
<a name="opaque-primitives"></a>
<h3>Opaque Primitives</h3>
<p>The renderer separates between opaque primitives and primitives which require alpha blending. By using OpenGL's Z-buffer and giving each primitive a unique z position, the renderer can freely reorder opaque primitives without any regard for their location on screen and which other elements they overlap with. By looking at each primitive's material state, the renderer will create opaque batches. From Qt Quick core item set, this includes Rectangle items with opaque colors and fully opaque images, such as JPEGs or BMPs.</p>
<p>Another benefit of using opaque primitives, is that opaque primitives does not require <tt>GL_BLEND</tt> to be enabled which can be quite costly, especially on mobile and embedded GPUs.</p>
<p>Opaque primitives are rendered in a front-to-back manner with <tt>glDepthMask</tt> and <tt>GL_DEPTH_TEST</tt> enabled. On GPUs that internally do early-z checks, this means that the fragment shader does not need to run for pixels or blocks of pixels that are obscured. Beware that the renderer still needs to take these nodes into account and the vertex shader is still run for every vertex in these primitives, so if the application knows that something is fully obscured, the best thing to do is to explicitly hide it using <a href="qml-qtquick-item.html#visible-prop">Item::visible</a> or <a href="qml-qtquick-item.html#opacity-prop">Item::opacity</a>.</p>
<p><b>Note: </b>The <a href="qml-qtquick-item.html#z-prop">Item::z</a> is used to control an Item's stacking order relative to its siblings. It has no direct relation to the renderer and OpenGL's Z-buffer.</p><a name="alpha-blended-primitives"></a>
<h3>Alpha Blended Primitives</h3>
<p>Once opaque primitives have been drawn, the renderer will disable <tt>glDepthMask</tt>, enable <tt>GL_BLEND</tt> and render all alpha blended primitives in a back-to-front manner.</p>
<p>Batching of alpha blended primitives requires a bit more effort in the renderer as elements that are overlapping need to be rendered in the correct order for alpha blending to look correct. Relying on the Z-buffer alone is not enough. The renderer does a pass over all alpha blended primitives and will look at their bounding rect in addition to their material state to figure out which elements can be batched and which can not.</p>
<p class="centerAlign"><img src="images/visualcanvas_overlap.png" alt="" /></p><p>In the left-most case, the blue backgrounds can be drawn in one call and the two text elements in another call, as the texts only overlap a background which they are stacked in front of. In the right-most case, the background of "Item 4" overlaps the text of "Item 3" so in this case, each of backgrounds and texts need to be drawn using separate calls.</p>
<p>Z-wise, the alpha primitives are interleaved with the opaque nodes and may trigger early-z when available, but again, setting <a href="qml-qtquick-item.html#visible-prop">Item::visible</a> to false is always faster.</p>
<a name="mixing-with-3d-primitives"></a>
<h3>Mixing with 3D primitives</h3>
<p>The scene graph can support pseudo 3D and proper 3D primitives. For instance, one can implement a "page curl" effect using a <a href="qml-qtquick-shadereffect.html">ShaderEffect</a> or implement a bumpmapped torus using <a href="qsggeometry.html">QSGGeometry</a> and a custom material. While doing so, one needs to take into account that the default renderer already makes use of the depth buffer.</p>
<p>The renderer modifies the vertex shader returned from <a href="qsgmaterialshader.html#vertexShader">QSGMaterialShader::vertexShader</a>() and compresses the z values of the vertex after the model-view and projection matrices has been applied and then adds a small translation on the z to position it the correct z position.</p>
<p>The compression assumes that the z values are in the range of 0 to 1.</p>
<a name="texture-atlas"></a>
<h3>Texture Atlas</h3>
<p>The active texture is a unique OpenGL state, which means that multiple primitives using different OpenGL textures cannot be batched. The Qt Quick scene graph for this reason allows multiple <a href="qsgtexture.html">QSGTexture</a> instances to be allocated as smaller sub-regions of a larger texture; a texture atlas.</p>
<p>The biggest benefit of texture atlases is that multiple <a href="qsgtexture.html">QSGTexture</a> instances now refer to the same OpenGL texture instance. This makes it possible to batch textured draw calls as well, such as Image items, <a href="qml-qtquick-borderimage.html">BorderImage</a> items, <a href="qml-qtquick-shadereffect.html">ShaderEffect</a> items and also C++ types such as <a href="qsgsimpletexturenode.html">QSGSimpleTextureNode</a> and custom QSGGeometryNodes using textures.</p>
<p><b>Note: </b>Large textures do not go into the texture atlas.</p><p>Atlas based textures are created by passing <a href="qquickwindow.html#CreateTextureOption-enum">QQuickWindow::TextureCanUseAtlas</a> to the <a href="qquickwindow.html#createTextureFromImage">QQuickWindow::createTextureFromImage</a>().</p>
<p><b>Note: </b>Atlas based textures do not have texture coordinates ranging from 0 to 1. Use <a href="qsgtexture.html#normalizedTextureSubRect">QSGTexture::normalizedTextureSubRect</a>() to get the atlas texture coordinates.</p><p>The scene graph uses heuristics to figure out how large the atlas should be and what the size threshold for being entered into the atlas is. If different values are needed, it is possible to override them using the environment variables <tt>QSG_ATLAS_WIDTH=[width]</tt>, <tt>QSG_ATLAS_HEIGHT=[height]</tt> and <tt>QSG_ATLAS_SIZE_LIMIT=[size]</tt>. Changing these values will mostly be interesting for platform vendors.</p>
<a name="batch-roots"></a>
<h2>Batch Roots</h2>
<p>In addition to mergin compatible primitives into batches, the default renderer also tries to minimize the amount of data that needs to be sent to the GPU for every frame. The default renderer identifies subtrees which belong together and tries to put these into separate batches. Once batches are identified, they are merged, uploaded and stored in GPU memory, using Vertex Buffer Objects.</p>
<a name="transform-nodes"></a>
<h3>Transform Nodes</h3>
<p>Each Qt Quick Item inserts a <a href="qsgtransformnode.html">QSGTransformNode</a> into the scene graph tree to manage its x, y, scale or rotation. Child items will be populated under this transform node. The default renderer tracks the state of transform nodes between frames, and will look at subtrees to decide if a transform node is a good candidate to become a root for a set of batches. A transform node which changes between frames and which has a fairly complex subtree, can become a batch root.</p>
<p>QSGGeometryNodes in the subtree of a batch root are pre-transformed relative to the root on the CPU. They are then uploaded and retained on the GPU. When the transform changes, the renderer only needs to update the matrix of the root, not each individual item, making list and grid scrolling very fast. For successive frames, as long as nodes are not being added or removed, rendering the list is effectively for free. When new content enters the subtree, the batch that gets it is rebuilt, but this is still relatively fast. There are usually several unchanging frames for every frame with added or removed nodes when panning through a grid or list.</p>
<p>Another benefit of identifying transform nodes as batch roots is that it allows the renderer to retain the parts of the tree that has not changed. For instance, say a UI consists of a list and a button row. When the list is being scrolled and delegates are being added and removed, the rest of the UI, the button row, is unchanged and can be drawn using the geometry already stored on the GPU.</p>
<p>The node and vertex threshold for a transform node to become a batch root can be overridden using the environment variables <tt>QSG_RENDERER_BATCH_NODE_THRESHOLD=[count]</tt> and <tt>QSG_RENDERER_BATCH_VERTEX_THRESHOLD=[count]</tt>. Overriding these flags will be mostly useful for platform vendors.</p>
<p><b>Note: </b>Beneath a batch root, one batch is created for each unique set of material state and geometry type.</p><a name="clipping"></a>
<h3>Clipping</h3>
<p>When setting <a href="qml-qtquick-item.html#clip-prop">Item::clip</a> to true, it will create a <a href="qsgclipnode.html">QSGClipNode</a> with a rectangle in its geometry. The default renderer will apply this clip by using scissoring in OpenGL. If the item is rotated by a non-90-degree angle, the OpenGL's stencil buffer is used. Qt Quick Item only supports setting a rectangle as clip through QML, but the scene graph API and the default renderer can use any shape for clipping.</p>
<p>When applying a clip to a subtree, that subtree needs to be rendered with a unique OpenGL state. This means that when <a href="qml-qtquick-item.html#clip-prop">Item::clip</a> is true, batching of that item is limited to its children. When there are many children, like a <a href="qml-qtquick-listview.html">ListView</a> or <a href="qml-qtquick-gridview.html">GridView</a>, or complex children, like a TextArea, this is fine. One should, however, use clip on smaller items with caution as it prevents batching. This includes button label, text field or list delegate and table cells.</p>
<a name="vertex-buffers"></a>
<h3>Vertex Buffers</h3>
<p>Each batch uses a vertex buffer object (VBO) to store its data on the GPU. This vertex buffer is retained between frames and updated when the part of the scene graph that it represents changes.</p>
<p>By default, the renderer will upload data into the VBO using <tt>GL_STATIC_DRAW</tt>. It is possible to select different upload strategy by setting the environment variable <tt>QSG_RENDERER_BUFFER_STRATEGY=[strategy]</tt>. Valid values are <tt>stream</tt> and <tt>dynamic</tt>. Changing this value is mostly useful for platform vendors.</p>
<a name="antialiasing"></a>
<h2>Antialiasing</h2>
<p>The scene graph supports two types of antialiasing. By default, primitives such as rectangles and images will be antialiased by adding more vertices along the edge of the primitives so that the edges fade to transparent. We call this method <i>vertex antialiasing</i>. If the user requests a multisampled OpenGL context, by setting a QSurfaceFormat with samples greater than <tt>0</tt> using QQuickWindow::setFormat(), the scene graph will prefer multisample based antialiasing (MSAA). The two techniques will affect how the rendering happens internally and have different limitations.</p>
<p>It is also possible to override the antialiasing method used by setting the environment variable <tt>QSG_ANTIALIASING_METHOD</tt> to either <tt>vertex</tt> or <tt>msaa</tt>.</p>
<p>Vertex antialiasing can produce seams between edges of adjacent primitives, even when the two edges are mathmatically the same. Multisample antialiasing does not.</p>
<a name="vertex-antialiasing"></a>
<h3>Vertex Antialiasing</h3>
<p>Vertex antialiasing can be enabled and disabled on a per-item basis using the <a href="qml-qtquick-item.html#antialiasing-prop">Item::antialiasing</a> property. It will work regardless of what the underlying hardware supports and produces higher quality antialiasing, both for normally rendered primitives and also for primitives captured into framebuffer objects, for instance using the <a href="qml-qtquick-shadereffectsource.html">ShaderEffectSource</a> type.</p>
<p>The downside to using vertex antialiasing is that each primitive with antialiasing enabled will have to be blended. In terms of batching, this means that the renderer needs to do more work to figure out if the primitive can be batched or not and due to overlaps with other elements in the scene, it may also result in less batching, which could impact performance.</p>
<p>On low-end hardware blending can also be quite expensive so for an image or rounded rectangle that covers most of the screen, the amount of blending needed for the interior of these primitives can result in significant performance loss as the entire primitive must be blended.</p>
<a name="multisample-antialiasing"></a>
<h3>Multisample Antialiasing</h3>
<p>Multisample antialiasing is a hardware feature where the hardware calculates a coverage value per pixel in the primitive. Some hardware can multisample at a very low cost, while other hardware may need both more memory and more GPU cycles to render a frame.</p>
<p>Using multisample antialiasing, many primitives, such as rounded rectangles and image elements can be antialiased and still be <i>opaque</i> in the scene graph. This means the renderer has an easier job when creating batches and can rely on early-z to avoid overdraw.</p>
<p>When multisample antialiasing is used, content rendered into framebuffer objects, need additional extensions to support multisampling of framebuffers. Typically <tt>GL_EXT_framebuffer_multisample</tt> and <tt>GL_EXT_framebuffer_blit</tt>. Most desktop chips have these extensions present, but they are less common in embedded chips. When framebuffer multisampling is not available in the hardware, content rendered into framebuffer objects will not be antialiased, including the content of a <a href="qml-qtquick-shadereffectsource.html">ShaderEffectSource</a>.</p>
<a name="performance"></a>
<h2>Performance</h2>
<p>As stated in the beginning, understanding the finer details of the renderer is not required to get good performance. It is written to optimize for common use cases and will perform quite well under almost any circumstance.</p>
<ul>
<li>Good performance comes from effective batching, with as little as possible of the geometry being uploaded again and again. By setting the environment variable <tt>QSG_RENDERER_DEBUG=render</tt>, the renderer will output statistics on how well the batching goes, how many batches, which batches are retained and which are opaque and not. When striving for optimal performance, uploads should happen only when really needed, batches should be fewer than 10 and at least 3-4 of them should be opaque.</li>
<li>The default renderer does not do any CPU-side viewport clipping nor occlusion detection. If something is not supposed to be visible, it should not be shown. Use <tt>Item::visible: false</tt> for items that should not be drawn. The primary reason for not adding such logic is that it adds additional cost which would also hurt applications that took care in behaving well.</li>
<li>Make sure the texture atlas is used. The Image and <a href="qml-qtquick-borderimage.html">BorderImage</a> items will use it unless the image is too large. For textures created in C++, pass <a href="qquickwindow.html#CreateTextureOption-enum">QQuickWindow::TextureCanUseAtlas</a> when calling QQuickWindow::createTexture(). By setting the environment variable <tt>QSG_ATLAS_OVERLAY</tt> all atlas textures will be colorized so they are easily identifiable in the application.</li>
<li>Use opaque primitives where possible. Opaque primitives are faster to process in the renderer and faster to draw on the GPU. For instance, PNG files will often have an alpha channel, even though each pixel is fully opaque. JPG files are always opaque. When providing images to an <a href="qquickimageprovider.html">QQuickImageProvider</a> or creating images with <a href="qquickwindow.html#createTextureFromImage">QQuickWindow::createTextureFromImage</a>(), let the image have QImage::Format_RGB32, when possible.</li>
<li>Be aware of that overlapping compond items, like in the illustration above, can not be batched.</li>
<li>Clipping breaks batching. Never use on a per-item basis, inside tables cells, item delegates or similar. Instead of clipping text, use eliding. Instead of clipping an image, create a <a href="qquickimageprovider.html">QQuickImageProvider</a> that returns a cropped image.</li>
<li>Batching only works for 16-bit indices. All built-in items use 16-bit indices, but custom geometry is free to also use 32-bit indices.</li>
<li>Some material flags prevent batching, the most limiting one being <a href="qsgmaterial.html#Flag-enum">QSGMaterial::RequiresFullMatrix</a> which prevents all batching.</li>
<li>Applications with a monochrome background should set it using <a href="qquickwindow.html#color-prop">QQuickWindow::setColor</a>() rather than using a top-level Rectangle item. <a href="qquickwindow.html#color-prop">QQuickWindow::setColor</a>() will be used in a call to <tt>glClear()</tt>, which is potentially faster.</li>
</ul>
<p>If an application performs poorly, make sure that rendering is actually the bottleneck. Use a profiler! The environment variable <tt>QSG_RENDER_TIMING=1</tt> will output a number of useful timing parameters which can be useful in pinpointing where a problem lies.</p>
</div>
<!-- @@@qtquick-visualcanvas-scenegraph-renderer.html -->
</div>
</div>
</div>
</div>
</div>
<div class="footer">
<p>
<acronym title="Copyright">©</acronym> 2013 Digia Plc and/or its
subsidiaries. Documentation contributions included herein are the copyrights of
their respective owners.<br> The documentation provided herein is licensed under the terms of the <a href="http://www.gnu.org/licenses/fdl.html">GNU Free Documentation License version 1.3</a> as published by the Free Software Foundation.<br> Digia, Qt and their respective logos are trademarks of Digia Plc in Finland and/or other countries worldwide. All other trademarks are property
of their respective owners. </p>
</div>
</body>
</html>
|