An experimental Python-based GPU shading embedded domain-specific language (EDSL)
Project description
Metashade
What is Metashade?
Metashade is an experimental Python-based GPU shading embedded domain-specific language (EDSL). When a Metashade script executes, it generates code in a target shading language. Currently, limited but useful subsets of HLSL and GLSL are supported, and more targets are possible in the future.
To see Metashade in action, check out the glTF demo at https://github.com/metashade/metashade-glTFSample or the tests which are run by CI:
There's also a prototype MaterialX integration that allows Metashade to generate MaterialX node implementations. See the metashade.mtlx package and the mtlx tests for examples.
For a detailed discussion of the motivation for Metashade and its design, please see the presentation on Google Slides or watch this presentation from Academy Software Foundation Open Source Days 2024.
Rationale
- Programming at a more abstract level than the target language:
- Metaprogramming - think C++ templates but with greater flexibility. Like any other Python code, Metashade code is polymorphic at generation time. This approach can replace the traditional ubershader practice, effectively replacing the C preprocessor with Python.
- Stricter typing - e.g. a 3D point and an RGB color can be represented with different Metashade types, backed by the same data type in HLSL.
- Multi-language/cross-platform support.
Cross-compilation (e.g. with SPIRV-Cross) is definitely an alternative but the code generation approach should offer higher flexibility around:
- more divergent languages, e.g. HLSL vs OSL;
- language dialects;
- integration required by the specific host application (a shader fragment with an interface defined in metadata, an effect file etc.), which is hard to accomplish with cross-compilation because it typically operates on final, full shaders with a defined entry point.
- Easy integration with content pipeline and build system scripts written in Python, and the vast Python ecosystem in general.
What does it look like?
The following Metashade Python code
@export
def D_Ggx(sh, NdotH: 'Float', fAlphaRoughness: 'Float') -> 'Float':
"""
GGX/Trowbridge-Reitz Normal Distribution Function.
Args:
NdotH: Dot product of surface normal and half-vector
fAlphaRoughness: Roughness parameter (perceptualRoughness^2)
Returns:
NDF value
"""
sh.fASqr = fAlphaRoughness * fAlphaRoughness
sh.fF = (NdotH * sh.fASqr - NdotH) * NdotH + sh.Float(1.0)
sh.return_(
(sh.fASqr / (sh.Float(math.pi) * sh.fF * sh.fF)).saturate()
)
generates the following HLSL output:
// GGX/Trowbridge-Reitz Normal Distribution Function.
//
// Args:
// NdotH: Dot product of surface normal and half-vector
// fAlphaRoughness: Roughness parameter (perceptualRoughness^2)
//
// Returns:
// NDF value
//
float D_Ggx(float NdotH, float fAlphaRoughness)
{
float fASqr = fAlphaRoughness * fAlphaRoughness;
float fF = (((NdotH * fASqr) - NdotH) * NdotH) + 1.0;
return saturate(fASqr / ((3.141592653589793 * fF) * fF));
}
How does it work?
Popular Pythonic GPU EDSLs like Nvidia Warp, Taichi, Numba and OpenAI’s Triton rely on Python's introspection to capture the Python AST and transpile it to the target language. This approach can only support a subset of Python syntax that maps onto the target language.
In contrast, Metashade generates target code dynamically, during the execution of Python code, modeling the state of the shader being generated in objects called generators. This requires some idiosyncratic Python syntax but in return we get the full power of Python at generation time. Python's run time becomes the shader's design time, and it becomes a metaprogramming language, independent of the target language and replacing mechanisms like the C Preprocessor, generics and templates.
This offers the following benefits:
- Easy-to-use metaprogramming. Imperative metaprogramming is possible (by comparison, C++ templates are a pure-functional language).
- The whole stack is debuggable by the application programmer.
- Codegen can interact with the outside world (file system or user input). E.g. the glTF demo loads glTF assets and generates shaders based on their contents.
- Codegen can integrate with arbitrary Python code. E.g. the glTF demo uses the third-party pygltflib to parse glTF assets.
- It's easy to build abstractions on top of basic codegen.
Creating a generator
Before Metashade can generate anything, a generator object has to be created for a specific target shading language profile, with an output file (or a file-like stream object) passed as a constructor argument, e.g.
from metashade.hlsl.sm6 import ps_6_0
from metashade.glsl import frag
def generate(sh):
# Polymorphic shader code for multiple targets
pass
with open('ps.hlsl', 'w') as ps_file:
sh = ps_6_0.Generator(ps_file)
generate(sh)
with open('fs.glsl', 'w') as frag_file:
sh = frag.Generator(frag_file)
generate(sh)
Note that, by convention, the generator object is always named sh (for "shader").
This helps Metashade code be polymorphic with regard to different target profiles.
E.g. code with the same logic can be generated for an HLSL pixel shader and a GLSL compute shader.
Generating C-like scopes and local variables
Metashade uses Python variables to represent variables in target C-like shading languages, but there are obviously major differences in their behavior, namely:
- C-like languages have different scoping rules than Python.
- In Python, variables are always assigned by reference and the same variable can point to different objects, possibly of different types in its lifetime. Variables in C-like shading languages, in contrast, are typed statically and are assigned by value.
The following Python mechanisms are used in Metashade to emulate the C-like semantics at Python code's run time.
with statements
Metashade uses Python's with statements to emulate C-like scopes.
For example, in a function definition, the sh.function object starts a new scope by modifying the internal state of generator sh when its special __enter__ and __exit__ methods are called.
__getattr__ and __setattr__
To model the semantics of target variables, Metashade exposes them syntactically as member variables on the generator object (hence the idiosyncratic sh.x syntax).
This is implemented with the special __getattr__ and __setattr__ functions, which overload member accesses.
With these, we have complete control over the variable's lifetimes and typing.
For example, we can capture the variable's name with __setattr__() without the need for Python introspection.
We can also easily check in __setattr__() if the user is trying to reinitialize the variable with a different type, and we can raise an exception in __getattr__() if the user tries to access a variable that's gone out of scope.
The __getattr__()/__setattr__() mechanism is also used for other features, such as accessing struct members and vector elements.
Operator overloading
Just like in C++, Python's operators can be overloaded with arbitrary logic. Metashade uses this feature to model expressions in the target language - something along the lines of
def __add__(self, other):
return Expression(f"{self} + {other}")
This also opens up possibilities for injecting custom logic, such as stronger type checks, independent from the target language.
E.g. sh.RgbF can’t be added to sh.Point3f even though they're both backed by the float3 built-in type in HLSL.
For added syntactic sugar, the __floordiv__ operator is overloaded to generate comments in the target language (see below).
Some examples
The following Python code
sh.rgba = sh.RgbaF(rgb = (0, 1, 0), a = 0)
sh // 'Swizzling - the destination type is deduced'
sh // "a-la `auto` in C++"
sh.color = sh.rgba.rgb
sh // 'Write masking'
sh.color.r = 1
sh // 'Intrinsics example'
sh.N = sh.N.normalize()
sh // 'Dot product == Python 3 matmul'
sh // '(a.k.a. "walrus") operator'
sh.NdotL = sh.N @ sh.L
# The walrus operator is also used to
# combine textures and samplers
combined_sampler = sh.g_tColor @ sh.g_sColor
sh // 'Sample the texture'
sh.rgbaSample = combined_sampler(sh.uv)
generates the following HLSL:
float4 rgba = float4(float3(0, 1, 0), 0);
// Swizzling - the destination type is deduced
// a-la `auto` in C++
float3 color = rgba.rgb;
// Write masking
color.r = 1;
// Intrinsics example
N = normalize(N);
// Dot product == Python 3 matmul
// (a.k.a. "walrus") operator
float NdotL = dot(N, L);
// Sample the texture
float4 rgbaSample = g_tColor.Sample(g_sColor, uv);
Conditional statements
Metashade models if and else statements of the target C-like languages with Python's with statements, for example:
with sh.if_(sh.g_f4A.x):
sh.result.color = sh.g_f4B
with sh.else_():
sh.result.color = sh.g_f4D
sh.return_(sh.result)
generates the following HLSL:
if (g_f4A.x)
{
result.color = g_f4B;
}
else
{
result.color = g_f4D;
}
return result;
Please note that the native Python if and else can be used in Metashade to implement design-time logic, similar to #ifdef or if constexpr() in C++.
This is another example of how Metashade explicitly separates design-time and run-time code.
For example, the following code is generated conditionally if the input geometry has vertex tangents:
if hasattr(sh.vsIn, 'Tobj'):
sh.vsOut.Tw = sh.g_WorldXf.xform(sh.vsIn.Tobj.xyz).xyz.normalize()
sh.vsOut.Bw = sh.vsOut.Nw.cross(sh.vsOut.Tw) * sh.vsIn.Tobj.w
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file metashade-0.6.0a1.tar.gz.
File metadata
- Download URL: metashade-0.6.0a1.tar.gz
- Upload date:
- Size: 58.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.13.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
f8744e8856124a733d3a1700781c56c10a5b27a5951e54d935668ec90dee1e0b
|
|
| MD5 |
f4426c17c8ef968f2582e3b67df3da39
|
|
| BLAKE2b-256 |
deabea5975972c09b114ca23101e01ce474797f33968b6f4031704ae495e554d
|
File details
Details for the file metashade-0.6.0a1-py3-none-any.whl.
File metadata
- Download URL: metashade-0.6.0a1-py3-none-any.whl
- Upload date:
- Size: 75.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.13.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
f8d3f745f61109aca3cb164a5db72c1160b0604179c424c5688a83b630170ae8
|
|
| MD5 |
8fd941a5c14b72fe8863e506323c04a6
|
|
| BLAKE2b-256 |
635e64717047306ffab6e6b1a8c6afdcaf7d0092d63c83bd4bc82630da087c12
|