Go to file
chyyran 0cb6f6a346 d3d11: be agnostic about deferred contexts
This brings the D3D11 API more in line with the D3D12 and Vulkan runtimes. To get the old behaviour with immediate contexts, just pass a NULL context.
2023-02-10 18:08:13 -05:00
.github/workflows doc: indicate need for Vulkan SDK 2023-02-06 02:20:11 -05:00
.idea capi: fix missing vulkan definitions 2023-01-13 19:17:57 -05:00
include d3d11: be agnostic about deferred contexts 2023-02-10 18:08:13 -05:00
librashader rt: make runtimes thread safe and document thread safety guarantees 2023-02-10 03:20:59 -05:00
librashader-capi d3d11: be agnostic about deferred contexts 2023-02-10 18:08:13 -05:00
librashader-common chore: Release 2023-02-08 21:22:33 -05:00
librashader-preprocess chore: Release 2023-02-08 21:22:33 -05:00
librashader-presets chore: Release 2023-02-08 21:22:33 -05:00
librashader-reflect chore: Release 2023-02-08 21:22:33 -05:00
librashader-runtime chore: Release 2023-02-08 21:22:33 -05:00
librashader-runtime-d3d11 d3d11: be agnostic about deferred contexts 2023-02-10 18:08:13 -05:00
librashader-runtime-d3d12 d3d11: be agnostic about deferred contexts 2023-02-10 18:08:13 -05:00
librashader-runtime-gl rt: make runtimes thread safe and document thread safety guarantees 2023-02-10 03:20:59 -05:00
librashader-runtime-vk rt: make runtimes thread safe and document thread safety guarantees 2023-02-10 03:20:59 -05:00
test d3d11: be agnostic about deferred contexts 2023-02-10 18:08:13 -05:00
.clang-format ld: finish librashader_ld 2023-01-14 01:38:59 -05:00
.gitattributes tests: add imgui example for dx11 2023-01-28 03:07:56 -05:00
.gitignore examples: add d3d11 c++ examples 2023-01-26 23:18:10 -05:00
.gitmodules examples: add d3d11 c++ examples 2023-01-26 23:18:10 -05:00
BROKEN_SHADERS.md test: add test for shader preprocess 2023-02-08 03:37:21 -05:00
Cargo.lock rt: make runtimes thread safe and document thread safety guarantees 2023-02-10 03:20:59 -05:00
Cargo.toml d3d12: take hello triangle from windows-rs 2023-02-01 02:08:50 -05:00
LICENSE-GPL.md doc: license under MPLv2 2022-11-21 20:49:18 -05:00
LICENSE.md doc: license under MPLv2 2022-11-21 20:49:18 -05:00
README.md rt: make runtimes thread safe and document thread safety guarantees 2023-02-10 03:20:59 -05:00
triangle.png d3d11: set the proper sampler for each pass input 2023-01-29 22:19:08 -05:00

librashader

Mega Bezel SMOOTH-ADV

Mega Bezel SMOOTH-ADV on DirectX 11

librashader (/ˈli:brəʃeɪdɚ/) is a preprocessor, compiler, and runtime for RetroArch 'slang' shaders, rewritten in pure Rust.

Latest Version Docs License Nightly rust

Supported Render APIs

librashader supports OpenGL 3, OpenGL 4.6, Vulkan, Direct3D 11, and Direct3D 12. Metal and WebGPU are not currently supported (but pull-requests are welcome). librashader does not support legacy render APIs such as older versions of OpenGL, or legacy versions of Direct3D.

API Status librashader feature
OpenGL 3.3+ gl
OpenGL 4.6 gl
Vulkan vk
Direct3D 11 d3d11
Direct3D 12 d3d12
Metal
WebGPU

✔ = Render API is supported — Render API is not supported

Usage

librashader provides both a Rust API under the librashader crate, and a C API. Both APIs are first-class and fully supported. The C API is geared more towards integration with existing projects. The Rust librashader crate exposes more of the internals if you wish to use parts of librashader piecemeal.

The librashader C API is best used by including librashader_ld.h in your project, which implements a loader that dynamically loads the librashader (librashader.so or librashader.dll) implementation in the search path.

Building

librashader requires the following build time dependencies


For Rust projects, simply add the crate to your Cargo.toml.

cargo add librashader

To build the C compatible dynamic library, cargo-post is required.

cargo post build --release --package librashader-capi

This will output a librashader.dll or librashader.so in the target folder.

C ABI Compatibility

As the recommended way of integrating librashader is by the librashader_ld single header library, ABI stability is important to ensure that updates to librashader do not break existing consumers.

As of 0.1.0-rc.3, the C ABI should be mostly stable. We reserve the right to make breaking changes before a numbered release without following semantic versioning.

Linking statically against librashader.h is possible, but is not officially supported. You will need to ensure linkage parameters are correct in order to successfully link with librashader.lib or librashader.a. The corrosion CMake package is highly recommended.

Examples

The following Rust examples show how to use each librashader runtime.

Some basic examples on using the C API are also provided in the librashader-capi-tests directory.

Compatibility

librashader implements the entire RetroArch shader pipeline and is highly compatible with existing shaders.

Please report an issue if you run into a shader that works in RetroArch, but not under librashader.

  • Filter chains do not terminate at the backbuffer.
    • Unlike RetroArch, librashader does not have full knowledge of the entire rendering state and is designed to be pluggable at any point in your render pipeline. Instead, filter chains terminate at a caller-provided output surface and viewport. It is the caller's responsibility to blit the surface back to the backbuffer.
  • Shaders are compiled in parallel where possible. This should noticeably decrease preset compile times. Parallel shader compilation is not available to OpenGL.
  • HDR10 support is not part of any shader runtime and is not supported by librashader.
  • For performance reasons, mipmaps are never generated for the input texture. In theory, this means that presets with mipmap_input0 = "true" will not get a mipmapped input. In practice, no known shader presets set mipmap_input0 = "true".

Runtime specific differences

  • OpenGL
    • Copying of in-flight framebuffer contents to history is done via glBlitFramebuffer rather than drawing a quad into an intermediate FBO.
    • Sampler objects are used rather than glTexParameter.
    • Sampler inputs and outputs are not renamed. This is useful for debugging shaders in RenderDoc.
    • UBO and Push Constant Buffer sizes are padded to 16-byte boundaries.
  • OpenGL 4.6+
    • All caveats from the OpenGL 3.3+ section should be considered.
    • Should work on OpenGL 4.5 but this is not guaranteed. The OpenGL 4.6 runtime may eventually switch to using ARB_spirv_extensions for loading shaders, and this will not be marked as a breaking change.
    • The OpenGL 4.6 runtime uses Direct State Access to minimize changes to the OpenGL state. For GPUs released within the last 5 years, this may improve performance.
  • Vulkan
    • The Vulkan runtime uses VK_KHR_dynamic_rendering by default. This extension must be enabled at device creation. Explicit render passes can be used by configuring filter chain options, but may have reduced performance compared to dynamic rendering.
    • Allocations within the runtime are done through gpu-allocator rather than handled manually.
  • Direct3D 11
    • Framebuffer copies are done via ID3D11DeviceContext::CopySubresourceRegion rather than a CPU conversion + copy.
  • Direct3D 12
    • The Direct3D 12 runtime uses render passes. This feature has been available since Windows 10 version 1809, which was released in late 2018.
    • For maximum compatibility with shaders, a shader compile pipeline based on spirv-to-dxil is used, with the SPIRV-Cross HLSL pipeline used as a fallback. This brings shader compatibility beyond what the RetroArch Direct3D 12 runtime provides. The HLSL pipeline fallback may be removed in the future as spirv-to-dxil improves.

Most, if not all shader presets should work fine on librashader. The runtime specific differences should not affect the output, and are more a heads-up for integrating librashader into your project.

Quad vertices and rotations

All runtimes except OpenGL render with an identity matrix MVP and a VBO for with range [-1, 1]. The final pass uses a Quad VBO with range [0, 1] and the following projection matrix by default.

static DEFAULT_MVP: &[f32; 16] = &[
  2.0, 0.0, 0.0, 0.0,
  0.0, 2.0, 0.0, 0.0,
  0.0, 0.0, 0.0, 0.0,
  -1.0, -1.0, 0.0, 1.0,
];

As with RetroArch, a rotation on this MVP will be applied only on the final pass for these runtimes. This is the only way to pass orientation information to shaders.

The OpenGL runtime uses a VBO for range [0, 1] for all passes and the following MVP for all passes.

static GL_DEFAULT_MVP: &[f32; 16] = &[
    2.0, 0.0, 0.0, 0.0,
    0.0, 2.0, 0.0, 0.0,
    0.0, 0.0, 2.0, 0.0,
    -1.0, -1.0, 0.0, 1.0,
];

Thread safety

In general, it is safe to create a filter chain instance from a different thread, but drawing filter passes must be externally synchronized. The exceptions to filter chain creation are in OpenGL, where creating the filter chain instance is safe if and only if the thread local OpenGL context is initialized to the same context as the drawing thread, and in Direct3D 11, where filter chain creation is thread-unsafe if the ID3D11Device was created with D3D11_CREATE_DEVICE_SINGLETHREADED.

Writing a librashader Runtime

If you wish to contribute a runtime implementation not already available, see the librashader-runtime crate for helpers and shared logic used across all librashader runtime implementations. Using these helpers and traits will ensure that your runtime has consistent behaviour for uniform and texture semantics bindings with the existing librashader runtimes.

These types should not be exposed to the end user in the runtime's public API, and should be kept internal to the implementation of the runtime.

License

The core parts of librashader such as the preprocessor, the preset parser, the reflection library, and the runtimes, are all licensed under the Mozilla Public License version 2.0.

The librashader C API, i.e. its headers and definitions, not its implementation in librashader-capi, are more permissively licensed, and may allow you to use librashader in your permissively licensed or proprietary project.

To facilitate easier use of librashader in projects incompatible with MPL-2.0, librashader_ld implements a loader which thunks its calls to any librashader.so or librashader.dll library found in the load path. A non-MPL-2.0 compatible project may link against librashader_ld to use the librashader runtime, provided that librashader.so or librashader.dll are distributed under the restrictions of MPLv2.

Note that this means that if your project is unable to comply with the requirements of MPL-2.0, you can not distribute librashader.so or librashader.dll alongside your project. The end user must obtain the implementation of librashader themselves. For more information, see the MPL 2.0 FAQ.

At your discretion, you may instead choose to distribute librashader under the terms of GPLv3 rather than MPL-2.0.