Vulkan & OpenGL Threaded CAD Scene Sample
We have released various Vulkan samples on our professional graphics repository. In this post we present some details of the “threaded cadscene sample”.
The sample allows comparing various rendering approaches using core OpenGL, extended OpenGL via bindless and NV_command_list as well as Vulkan. It currently does make use of NVIDIA specific extensions to use Vulkan within an OpenGL context and display a Vulkan image.
The content being rendered in the sample is a CAD model which is made of many parts that have few triangles. Having such low complexity per draw-call can very often result into being CPU bound. The sample is a fork of the public cadscene OpenGL sample and introduces the usage of multiple CPU threads.
By default a performance graph is drawn that illustrates the relative costs for rendering. The top green bar represents GPU time, and the lower blue bar CPU time. The bar that extends to the maximum is the primary bottle-neck. Be aware that the GPU costs may be appear higher than actual, when very CPU bound as discussed below in the performance section. When cloning the model multiple times, the scene complexity is increased, the sample does not use instancing for this, but generates actual bindings/drawcalls for the new objects. An animation effect can be activated to “extrude” the model by modifying the matrices on the GPU.
Shademodes
Solid: just triangles are drawn Solid with edges: triangles and edge outlines on top (using polygonoffset to push triangles back). When no global sorting (see later) is performed, it means we toggle between the two modes for every object. Alongside the mode changes is a shader change, so that wireframe is unlit. The shader change for that effect is not really necessary, and toggling so often is not optimal, but it creates some artificial load for the graphics api.When the “min statechanges” option is enabled, we will draw first solid then edges, effectively reducing the number of shader changes dramatically.
Renderers
Upon renderer activation the scene is traversed and encoded into a list of “drawitems”. Optionally this list can be globally sorted once with the “min statechanges” option. All renderers operate from the same “drawitems” list.
By default all renderers also use the same state and binding sequences. VBO/IBO bind per geometry change, a uniform buffer range bind per material and a matrix change.
Variants:
- default: we traverse the list and either directly render from it, or build temporary command-buffers every frame.
- re-use: command-buffers are built only once and then re-used when rendering. This typically yields lowest CPU costs.
-
MT: multi-threaded, makes use of N worker-threads to build the command-buffers. The list is processed in chunks of workingset many items. Each thread grabs an available chunk. The global drawing order may be different every frame.
- worker process: the generated command-buffers are processed by the worker-threads directly (but protected by mutex).
- main process: command-buffers are passed to the main thread for processing (no mutex for processing step).
OpenGL Renderers
-
gl core
All matrices and materials are stored in big buffer objects, which allows us to efficiently bind the required sub-range for a drawcall via glBindBufferRange(GL_UNIFORM_BUFFER, usageSlot, buffer, index * itemSize, itemSize). (Only uses GL Core techniques) -
gl nvbindless
Similar as above but using glBufferAddressRange calls passing 64-bit GPU addresses instead of GL object handles. -
gl re-use nvcmd buffer
Uses NV_command_list to encode sequence of drawing and binding commands. They are stored in a GL buffer object and rendered from that. -
gl re-use nvcmd compiled
The commands are not stored in a buffer but in the compiled commandlist object, allowing further optimizations especially for state changes like shaders. -
gl MT nvcmd main process
Persistent mapped buffers are used for storing the commands. The pointers are passed to the worker-threads where the command-buffers are filled. Those threads do not have a GL context. Rendering is done from the main thread. Different buffers are used every other frame to avoid fences.
Vulkan Renderers
-
vk re-use cmd
The entire scene is encoded in a single big command-buffer, and re-used every frame. -
vk re-use obj-level cmd
Every object in the scene has its own small secondary command-buffer. This means less optimal state transitions, as each command-buffer must be self contained, there is no state inheritance (other than the rendertarget being used). At render time all the secondaries are referenced by a primary command-buffer that is built per-frame. Given the very few per-object commands, this serves rather as experiment and is not a recommended rendering method. -
vk MT cmd worker process
Each thread has FRAMES CommandBufferPools, which are cycled through. At the beginning the pool is reset and command-buffers are generated from it in chunks. Using another pool every frame avoids the use of fences. ```USE_THREADED_SECONDARIES``` define controls whether the threaded command-buffers are secondaries (default). -
vk MT cmd main process
Same as above just that the command-buffers are passed to main thread for submission/addition to the primary command-buffer.
Performance
Preliminary Results: The Vulkan driver is still very new and the performance numbers are therefore not final. The benchmark was run on a Quadro M6000, with a first generation i7-860 CPU.
Important Note on GPU timings The GPU time is measured via timestamps. If the "begin" timestamp is in a different driver submission to the GPU (aka pushbuffer) than the "end" timestamp, then CPU bottlenecks will skew the GPU timings. The CPU time between submissions will impact the reported GPU time. This may be the case when very many GPU commands are created per-frame, although it typically affects OpenGL more than Vulkan.
To get more exact timings about the relative cost of each section, the application does measure the CPU time for the "flush" (Vulkan vkQueueSubmit, OpenGL glFlush) as well. The application does three flushes when animation is active, and two otherwise.
Solid, Material-Grouped, 4 model copies
In this scenario we generate around 44k drawcalls all using the same shader. The cost of uniform and geometry buffer binds as well as drawcalls (average ~50 triangles) dominates.
renderer | GPU time | CPU time [ms] |
---|---|---|
gl core | 4.1 | 7.8 |
gl nvbindless | 2.7 | 4.7 |
vk cmd 1 thread | 1.0 | 1.8 |
vk cmd 2 threads | 1.0 | 1.0 |
gl nvcmd 1 thread | 1.2 | 0.9 |
gl nvcmd 2 threads | 1.2 | 0.6 |
vk re-use cmd | 1.0 | 0.03 |
vk re-use obj-level cmd | 3.6 | 1.6 |
gl re-use nvcmd buffer | 1.0 | 0.05 |
gl re-use nvcmd compiled | 1.0 | 0.05 |
Our CPU costs are nicely reduced with Vulkan and scale with more threads. By re-using command buffers we can remove the CPU costs almost entirely. We can also see that the re-use of many tiny command buffers is not recommended (in this scenario around 10k secondary buffers per-frame).
The NV_command_list can be a tad faster than Vulkan on the CPU-side, as it benefits of having the API designed directly for the hardware in mind. There is no function calls and the data is formatted in a hardware friendly way. It is expected that Vulkan's performance will improve or that similar mechanisms will be exposed.
Solid with Edges, Material-Grouped, 2 model copies
Now we render edges on top of the surfaces. We still generate around 44k drawcalls but half are for triangles, the other for lines. Within every object we toggle between the two and change the shader (not optimal but serves as stress test). The cost of those shader changes now dominates.
renderer | GPU time | CPU time [ms] |
---|---|---|
gl core | skewed! | 45.0 |
gl nvbindless | skewed! | 41.0 |
vk cmd 1 thread | 3.5 | 1.8 |
vk cmd 2 threads | 3.5 | 1.0 |
gl nvcmd 1 thread | 3.5 | 1.5 |
gl nvcmd 2 threads | 3.5 | 1.5 |
vk re-use cmd | 3.4 | 0.6 |
vk re-use obj-level cmd | 3.6 | 1.0 |
gl re-use nvcmd buffer | 3.0 | 1.1 |
gl re-use nvcmd compiled | 2.8 | 0.05 |
In this scenario we can see that the modern principles of Vulkan (and NV_command_list) are on a whole new level when it comes to CPU efficiency (and NVIDIA's Quadro OpenGL driver is well-optimized). The shader toggles are just way too costly for a classic graphics API approach.
The NV_command_list doesn't scale with CPU threads as the shader state switching is handled on the main submission thread in a serial fashion. Vulkan, however, still can scale by using more threads.
Performance Conclusion
Overall Vulkan is a big improvement over unextended OpenGL, especially re-using CommandBuffers can result in greatly reduced CPU cost. In this particular sample NV_command_list is often very similar to Vulkan, which does show that OpenGL with proprietary extensions can be very fast. In pure draw-call limited scenarios core OpenGL's MultiDrawIndirect can also be very efficient as seen in another sample.
It should be noted that this sample only uses two pipeline objects (or state-objects in OpenGL) and the rest of the scene's resources are static as well. That means vulkan cannot make use of its greatest strength: generating and validating resources in parallel. Therefore, benchmarks in samples like this are very directed at a certain problem and only a snapshot of an APIs capability.