OCIO and MaterialX¶
This notebook will cover one workflow of using OCIO to generate code for MaterialX. We will be using the API based on OCIO 2.2 and above (released in October 2022).
The aim of this notebook is to go over how OCIO can be used to generate "implementations". The longer term MaterialX aim is to generate functional node graphs. This book will cover the setup required for generation, but can only show how source code implementation generation can be performed at the curren time.
The workflow covered is the "green" parts in this overall workflow diagram:
If / when graph generation is possible source code implementations can be swapped out for graph implementations.
The breakdown is as follows:
library (cmlib
)
- OCIO setup
- Setting up OCIO configurations and getting available color spaces.
- Setting up OCIO "processors" for generating color transforms
- Generating source code implementations
- Creating MaterialX implementations and definition wrappers for
color3
andcolor4
variants. - Add definitions and implementations to the "standard" transform
For further OCI there is a fair bit of documentation available with a useful starting place here
Setup¶
OpenColorIO is available as a pre-built Python package on PyPi here
pip
can be used to install the package which is called PyOpenColorIO
User can also build OpenColorIO by cloning the GitHub repository.
For the purposes of generating color transform implementation, most of the build options can be disabled. For For building and installing the Python package, make sure that the appropriate build option is set (current OCIO_BUILD_PYTHON
, OCIO_INSTALL_EXT_PACKAGES
respectively).
An example set of options is given here:
-DOPENIMAGEIO_INCLUDE_DIR="" -DOPENIMAGEIO_LIBRARY="" -DOCIO_BUILD_DOCS=OFF -DBUILD_SHARED_LIBS=OFF -DOCIO_BUILD_TESTS=ON -DOCIO_BUILD_GPU_TESTS=OFF -DOCIO_BUILD_PYTHON=1 -DOCIO_BUILD_JAVA=0 -DOCIO_INSTALL_EXT_PACKAGES=ALL -DOCIO_BUILD_APPS=0 -DOCIO_BUILD_NUKE=0
A third alternative is to use a pre-built Python package which comes with another installation such as a DCC. A check should be made to not inadvertently use the incorrect version of the package.
For this book, we install from PyPi.
# Package install
#pip install OpenColorIO
Here we import PyOpenColorIO and MaterialX.
# Import OCIO package
import PyOpenColorIO as OCIO
import MaterialX as mx
print('OCIO version:', OCIO.GetVersion())
print('MaterialX version:', mx.getVersionString())
OCIO version: 2.4.1 MaterialX version: 1.39.2
# Get the OCIO built in configs
registry = OCIO.BuiltinConfigRegistry().getBuiltinConfigs()
This items return canned be scanned and the appropriate configuration instantiated using CreateFromBuiltInConfig()
In the following example we built a dictionary of configs along with the available color spaces.
# Create a dictionary of configs
configs = {}
for item in registry:
# The short_name is the URI-style name.
# The ui_name is the name to use in a user interface.
short_name, ui_name, isRecommended, isDefault = item
# Don't present built-in configs to users if they are no longer recommended.
if isRecommended:
# Create a config using the Cg config
config = OCIO.Config.CreateFromBuiltinConfig(short_name)
colorSpaces = None
if config:
colorSpaces = config.getColorSpaces()
if colorSpaces:
configs[short_name] = [config, colorSpaces]
# Print the configs
for config in configs:
print('Built-in config:', config)
csnames = configs[config][0].getColorSpaceNames()
print('- Number of color spaces: %d' % len(csnames))
#for csname in csnames:
# print(' -', csname)
Built-in config: cg-config-v2.2.0_aces-v1.3_ocio-v2.4 - Number of color spaces: 23 Built-in config: studio-config-v2.2.0_aces-v1.3_ocio-v2.4 - Number of color spaces: 54
A more direct way to get the desired config is to call CreateFomFile
with the appropriate built in path. In this case we get the ACES Cg Config
.`
acesCgConfigPath = 'ocio://cg-config-v1.0.0_aces-v1.3_ocio-v2.1'
builtinCfgC = OCIO.Config.CreateFromFile(acesCgConfigPath)
print('Built-in config:', builtinCfgC.getName())
csnames = builtinCfgC.getColorSpaceNames()
print('- Number of color spaces: %d' % len(csnames))
Built-in config: cg-config-v1.0.0_aces-v1.3_ocio-v2.1 - Number of color spaces: 14
Color Spaces¶
To check what color space identifiers can be used we print out each color space name along with any aliases by calling getAliases()
on each color space.
from IPython.display import display_markdown
title = '| Configuration | Color Space | Aliases |\n'
title = title + '| --- | --- | --- |\n'
rows = ''
for c in configs:
config = configs[c][0]
colorSpaces = configs[c][1]
for colorSpace in colorSpaces:
aliases = colorSpace.getAliases()
rows = rows + '| ' + c + ' | ' + colorSpace.getName() + ' | ' + ', '.join(aliases) + ' |\n'
md = '<details><summary>Color Spaces</summary>\n\n' + title + rows + '</details>'
display_markdown(md, raw=True)
Color Spaces
Configuration | Color Space | Aliases |
---|---|---|
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | sRGB - Display | srgb_display |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | Display P3 - Display | displayp3_display |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | Rec.1886 Rec.709 - Display | rec1886_rec709_display |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | Rec.2100-PQ - Display | rec2100_pq_display |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | ST2084-P3-D65 - Display | st2084_p3d65_display |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | P3-D65 - Display | p3d65_display |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | ACES2065-1 | aces2065_1, aces, ACES - ACES2065-1, lin_ap0, lin_ap0_scene |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | ACEScc | ACES - ACEScc, acescc_ap1 |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | ACEScct | ACES - ACEScct, acescct_ap1 |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | ACEScg | ACES - ACEScg, lin_ap1, lin_ap1_scene |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | sRGB Encoded Rec.709 (sRGB) | srgb_encoded_rec709_srgb, Utility - sRGB - Texture, srgb_texture, srgb_rec709_scene, Input - Generic - sRGB - Texture, sRGB - Texture, srgb_tx |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | Gamma 1.8 Encoded Rec.709 | g18_encoded_rec709, Utility - Gamma 1.8 - Rec.709 - Texture, g18_rec709, Gamma 1.8 Rec.709 - Texture, g18_rec709_tx, g18_rec709_scene |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | Gamma 2.2 Encoded Rec.709 | g22_encoded_rec709, Utility - Gamma 2.2 - Rec.709 - Texture, g22_rec709, Gamma 2.2 Rec.709 - Texture, g22_rec709_tx, g22_rec709_scene |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | Gamma 2.4 Encoded Rec.709 | g24_encoded_rec709, g24_rec709, rec709_display, Utility - Rec.709 - Display, Gamma 2.4 Rec.709 - Texture, g24_rec709_tx |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | sRGB Encoded P3-D65 | srgb_encoded_p3d65, srgb_p3d65, srgb_displayp3, srgb_p3d65_scene, sRGB Encoded P3-D65 - Texture, srgb_encoded_p3d65_tx |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | Gamma 2.2 Encoded AdobeRGB | g22_encoded_adobergb, adobergb, g22_adobergb_scene |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | sRGB Encoded AP1 | srgb_encoded_ap1, srgb_ap1, srgb_ap1_scene, sRGB Encoded AP1 - Texture, srgb_encoded_ap1_tx |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | Gamma 2.2 Encoded AP1 | g22_encoded_ap1, g22_ap1, Gamma 2.2 AP1 - Texture, g22_ap1_tx, g22_ap1_scene |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear Rec.709 (sRGB) | lin_rec709_srgb, Utility - Linear - Rec.709, lin_rec709, lin_rec709_scene, lin_srgb, Utility - Linear - sRGB |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear P3-D65 | lin_p3d65, Utility - Linear - P3-D65, lin_displayp3, lin_p3d65_scene, Linear Display P3 |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear AdobeRGB | lin_adobergb, Utility - Linear - Adobe RGB, lin_adobergb_scene |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear Rec.2020 | lin_rec2020, Utility - Linear - Rec.2020, lin_rec2020_scene |
cg-config-v2.2.0_aces-v1.3_ocio-v2.4 | Raw | Utility - Raw, none |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | sRGB - Display | srgb_display |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Display P3 - Display | displayp3_display |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Rec.1886 Rec.709 - Display | rec1886_rec709_display |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Rec.1886 Rec.2020 - Display | rec1886_rec2020_display |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Rec.2100-HLG - Display | rec2100_hlg_display |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Rec.2100-PQ - Display | rec2100_pq_display |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | ST2084-P3-D65 - Display | st2084_p3d65_display |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | P3-D60 - Display | p3d60_display |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | P3-D65 - Display | p3d65_display |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | P3-DCI - Display | p3_dci_display |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | ACES2065-1 | aces2065_1, aces, ACES - ACES2065-1, lin_ap0, lin_ap0_scene |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | ACEScc | ACES - ACEScc, acescc_ap1 |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | ACEScct | ACES - ACEScct, acescct_ap1 |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | ACEScg | ACES - ACEScg, lin_ap1, lin_ap1_scene |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | ADX10 | Input - ADX - ADX10 |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | ADX16 | Input - ADX - ADX16 |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Apple Log | apple_log |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear ARRI Wide Gamut 3 | lin_arri_wide_gamut_3, Input - ARRI - Linear - ALEXA Wide Gamut, lin_alexawide |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | ARRI LogC3 (EI800) | arri_logc3_ei800, Input - ARRI - V3 LogC (EI800) - Wide Gamut, logc3ei800_alexawide |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | ARRI LogC4 | arri_logc4 |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear ARRI Wide Gamut 4 | lin_arri_wide_gamut_4, lin_awg4 |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | BMDFilm WideGamut Gen5 | bmdfilm_widegamut_gen5 |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | DaVinci Intermediate WideGamut | davinci_intermediate_widegamut |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | CanonLog2 CinemaGamut D55 | canonlog2_cinemagamut_d55, Input - Canon - Canon-Log2 - Cinema Gamut Daylight, canonlog2_cgamutday |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | CanonLog3 CinemaGamut D55 | canonlog3_cinemagamut_d55, Input - Canon - Canon-Log3 - Cinema Gamut Daylight, canonlog3_cgamutday |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear CinemaGamut D55 | lin_cinemagamut_d55, Input - Canon - Linear - Canon Cinema Gamut Daylight, lin_canoncgamutday |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear V-Gamut | lin_vgamut, Input - Panasonic - Linear - V-Gamut |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | V-Log V-Gamut | vlog_vgamut, Input - Panasonic - V-Log - V-Gamut |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear REDWideGamutRGB | lin_redwidegamutrgb, Input - RED - Linear - REDWideGamutRGB, lin_rwg |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Log3G10 REDWideGamutRGB | log3g10_redwidegamutrgb, Input - RED - REDLog3G10 - REDWideGamutRGB, rl3g10_rwg |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear S-Gamut3 | lin_sgamut3, Input - Sony - Linear - S-Gamut3 |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear S-Gamut3.Cine | lin_sgamut3cine, Input - Sony - Linear - S-Gamut3.Cine |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear Venice S-Gamut3 | lin_venice_sgamut3, Input - Sony - Linear - Venice S-Gamut3 |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear Venice S-Gamut3.Cine | lin_venice_sgamut3cine, Input - Sony - Linear - Venice S-Gamut3.Cine |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | S-Log3 S-Gamut3 | slog3_sgamut3, Input - Sony - S-Log3 - S-Gamut3 |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | S-Log3 S-Gamut3.Cine | slog3_sgamut3cine, Input - Sony - S-Log3 - S-Gamut3.Cine, slog3_sgamutcine |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | S-Log3 Venice S-Gamut3 | slog3_venice_sgamut3, Input - Sony - S-Log3 - Venice S-Gamut3 |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | S-Log3 Venice S-Gamut3.Cine | slog3_venice_sgamut3cine, Input - Sony - S-Log3 - Venice S-Gamut3.Cine, slog3_venice_sgamutcine |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear BMD WideGamut Gen5 | lin_bmd_widegamut_gen5 |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear DaVinci WideGamut | lin_davinci_widegamut |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | sRGB Encoded Rec.709 (sRGB) | srgb_encoded_rec709_srgb, Utility - sRGB - Texture, srgb_texture, srgb_rec709_scene, Input - Generic - sRGB - Texture, sRGB - Texture, srgb_tx |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Gamma 1.8 Encoded Rec.709 | g18_encoded_rec709, Utility - Gamma 1.8 - Rec.709 - Texture, g18_rec709, Gamma 1.8 Rec.709 - Texture, g18_rec709_tx, g18_rec709_scene |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Gamma 2.2 Encoded Rec.709 | g22_encoded_rec709, Utility - Gamma 2.2 - Rec.709 - Texture, g22_rec709, Gamma 2.2 Rec.709 - Texture, g22_rec709_tx, g22_rec709_scene |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Gamma 2.4 Encoded Rec.709 | g24_encoded_rec709, g24_rec709, rec709_display, Utility - Rec.709 - Display, Gamma 2.4 Rec.709 - Texture, g24_rec709_tx |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Camera Rec.709 | camera_rec709, Utility - Rec.709 - Camera, rec709_camera |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | sRGB Encoded P3-D65 | srgb_encoded_p3d65, srgb_p3d65, srgb_displayp3, srgb_p3d65_scene, sRGB Encoded P3-D65 - Texture, srgb_encoded_p3d65_tx |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Gamma 2.2 Encoded AdobeRGB | g22_encoded_adobergb, adobergb, g22_adobergb_scene |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | sRGB Encoded AP1 | srgb_encoded_ap1, srgb_ap1, srgb_ap1_scene, sRGB Encoded AP1 - Texture, srgb_encoded_ap1_tx |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Gamma 2.2 Encoded AP1 | g22_encoded_ap1, g22_ap1, Gamma 2.2 AP1 - Texture, g22_ap1_tx, g22_ap1_scene |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear Rec.709 (sRGB) | lin_rec709_srgb, Utility - Linear - Rec.709, lin_rec709, lin_rec709_scene, lin_srgb, Utility - Linear - sRGB |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear P3-D65 | lin_p3d65, Utility - Linear - P3-D65, lin_displayp3, lin_p3d65_scene, Linear Display P3 |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear AdobeRGB | lin_adobergb, Utility - Linear - Adobe RGB, lin_adobergb_scene |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Linear Rec.2020 | lin_rec2020, Utility - Linear - Rec.2020, lin_rec2020_scene |
studio-config-v2.2.0_aces-v1.3_ocio-v2.4 | Raw | Utility - Raw, none |
Supported Color Spaces in MaterialX¶
MaterialX currently uses color space names for :
- Color space (
colorspace
attribute) tagging for input images (filename
attributes) and colors (color3
andcolor4
types). - Node category identifiers for node definitions of the form
<from color space>_<to color space name>
to specify color space conversion nodes. (Node definitions are support as of MaterialX 1.38.7)
Note that any valid color space name can be used for input tagging.
At time of writing only specific color space conversions are supported via node definitions and hence can perform code injection during shader code generation.
Any color space information can still be passed as meta-data to the code generated (e.g. as is possible with the OSL
generator).
Further note that only certain aliases which are valid MaterialX identifiers are recognized in this context. For example g18_rec709
is used for color space Gamma 1.8 Rec.709 - Texture
and lin_rec709
is used for color space Linear Rec.709 (sRGB)
.
OCIO Shader Code Generation¶
It is possible to generate code color space transforms for certain code generation targets.
This is done by:
- Calling
getProcessor()
on the config with desired "source" and "destination" color spaces for the transform. - Creating a CPU or GPU processor
- Set the appropriate target language
- Getting the shader code using
getShaderText()
def generateShaderCode(config, sourceColorSpace, destColorSpace, language):
cshaderCodee = ''
if not config:
return shaderCode
# Create a processor for a pair of colorspaces (namely to go to linear)
processor = None
try:
processor = config.getProcessor(sourceColorSpace, destColorSpace)
except:
return shaderCode
gpuProcessor = None
if processor:
processor = processor.getOptimizedProcessor(OCIO.OPTIMIZATION_ALL)
gpuProcessor = processor.getDefaultGPUProcessor()
if gpuProcessor:
shaderDesc = OCIO.GpuShaderDesc.CreateShaderDesc()
if shaderDesc:
shaderDesc.setLanguage(language)
gpuProcessor.extractGpuShaderInfo(shaderDesc)
shaderCode = shaderDesc.getShaderText()
return shaderCode
# Use GLSL as the shader language to produce, and linear as the target color space
language = OCIO.GpuLanguage.GPU_LANGUAGE_GLSL_4_0
targetColorSpace = 'lin_rec709'
# Go through all the config and create code for each transform
title = '| Source | Target | Code |\n'
title = title + '| --- | --- | --- |\n'
rows = ''
testedSources = set()
for c in configs:
config = OCIO.Config.CreateFromBuiltinConfig(c)
colorSpaces = config.getColorSpaces()
for colorSpace in colorSpaces:
colorSpaceName = colorSpace.getName()
# Skip if the colorspace is already tested
if colorSpaceName in testedSources:
continue
testedSources.add(colorSpaceName)
code = generateShaderCode(config, colorSpace.getName(), targetColorSpace, language)
code = code.replace('\n', '<br>')
code = '<code>' + code + '</code>'
rows = rows + '| ' + colorSpace.getName() + ' | ' + targetColorSpace + ' | ' + code + '|\n'
md = '<details><summary>Transform Code for GLSL</summary>\n\n' + title + rows + '</details>'
display_markdown(md, raw=True)
Transform Code for GLSL
Source | Target | Code |
---|---|---|
sRGB - Display | lin_rec709 |
|
Display P3 - Display | lin_rec709 |
|
Rec.1886 Rec.709 - Display | lin_rec709 |
|
Rec.2100-PQ - Display | lin_rec709 |
|
ST2084-P3-D65 - Display | lin_rec709 |
|
P3-D65 - Display | lin_rec709 |
|
ACES2065-1 | lin_rec709 |
|
ACEScc | lin_rec709 |
|
ACEScct | lin_rec709 |
|
ACEScg | lin_rec709 |
|
sRGB Encoded Rec.709 (sRGB) | lin_rec709 |
|
Gamma 1.8 Encoded Rec.709 | lin_rec709 |
|
Gamma 2.2 Encoded Rec.709 | lin_rec709 |
|
Gamma 2.4 Encoded Rec.709 | lin_rec709 |
|
sRGB Encoded P3-D65 | lin_rec709 |
|
Gamma 2.2 Encoded AdobeRGB | lin_rec709 |
|
sRGB Encoded AP1 | lin_rec709 |
|
Gamma 2.2 Encoded AP1 | lin_rec709 |
|
Linear Rec.709 (sRGB) | lin_rec709 |
|
Linear P3-D65 | lin_rec709 |
|
Linear AdobeRGB | lin_rec709 |
|
Linear Rec.2020 | lin_rec709 |
|
Raw | lin_rec709 |
|
Rec.1886 Rec.2020 - Display | lin_rec709 |
|
Rec.2100-HLG - Display | lin_rec709 |
|
P3-D60 - Display | lin_rec709 |
|
P3-DCI - Display | lin_rec709 |
|
ADX10 | lin_rec709 |
|
ADX16 | lin_rec709 |
|
Apple Log | lin_rec709 |
|
Linear ARRI Wide Gamut 3 | lin_rec709 |
|
ARRI LogC3 (EI800) | lin_rec709 |
|
ARRI LogC4 | lin_rec709 |
|
Linear ARRI Wide Gamut 4 | lin_rec709 |
|
BMDFilm WideGamut Gen5 | lin_rec709 |
|
DaVinci Intermediate WideGamut | lin_rec709 |
|
CanonLog2 CinemaGamut D55 | lin_rec709 |
|
CanonLog3 CinemaGamut D55 | lin_rec709 |
|
Linear CinemaGamut D55 | lin_rec709 |
|
Linear V-Gamut | lin_rec709 |
|
V-Log V-Gamut | lin_rec709 |
|
Linear REDWideGamutRGB | lin_rec709 |
|
Log3G10 REDWideGamutRGB | lin_rec709 |
|
Linear S-Gamut3 | lin_rec709 |
|
Linear S-Gamut3.Cine | lin_rec709 |
|
Linear Venice S-Gamut3 | lin_rec709 |
|
Linear Venice S-Gamut3.Cine | lin_rec709 |
|
S-Log3 S-Gamut3 | lin_rec709 |
|
S-Log3 S-Gamut3.Cine | lin_rec709 |
|
S-Log3 Venice S-Gamut3 | lin_rec709 |
|
S-Log3 Venice S-Gamut3.Cine | lin_rec709 |
|
Linear BMD WideGamut Gen5 | lin_rec709 |
|
Linear DaVinci WideGamut | lin_rec709 |
|
Camera Rec.709 | lin_rec709 |
|
Integrating OCIO with MaterialX¶
We will pick an example transform to go over details on mapping from OCIO to MaterialX.
The first thing of note is OCIO function signatures
Currently all signatures transform 4 channel color inputs while MaterialX supports both 3 and 4 channel variants. This can be easily handled by adding pre and post conversion nodes, or by creating variant function signatures. The former is more robust and more in line with the proposed direction to have all OCIO transforms to be represented as graphs.
The signature name is not unique. This can be handled as OCIO provides mechanism to override the function names using
setFunctionName
andsetResourcePrefix
.
Following the current MaterialX convention we use the signature notation:
mx_<sourceName>_to_<targetname>_<type>
where type
is either color3
or color4
for 3 or 4 channel variants.
We add in two new utilities:
createTransformName
which will generate the unique function namesetShaderDescriptionParameters
which overrides the function name but also adds a prefix to uniquely identify dependent resources.
These are then used in a new code generation variation called generateShaderCode2()
which has additionally been modified to return the number of dependent texture resources, which can be queried from the shader
descriptor via the GpuShaderDesc.getTextures()
iterator.
def createTransformName(sourceSpace, targetSpace, typeName):
transformFunctionName = "mx_" + mx.createValidName(sourceSpace) + "_to_" + targetSpace + "_" + typeName
return transformFunctionName
def setShaderDescriptionParameters(shaderDesc, sourceSpace, targetSpace, typeName):
transformFunctionName = createTransformName(sourceSpace, targetSpace, typeName)
shaderDesc.setFunctionName(transformFunctionName)
shaderDesc.setResourcePrefix(transformFunctionName)
def generateShaderCode2(config, sourceColorSpace, destColorSpace, language):
shaderCode = ''
textureCount = 0
if not config:
return shaderCode, textureCount
# Create a processor for a pair of colorspaces (namely to go to linear)
processor = None
try:
processor = config.getProcessor(sourceColorSpace, destColorSpace)
except:
print('Failed to generated code for transform: %s -> %s' % (sourceColorSpace, destColorSpace))
return shaderCode, textureCount
if processor:
processor = processor.getOptimizedProcessor(OCIO.OPTIMIZATION_ALL)
gpuProcessor = processor.getDefaultGPUProcessor()
if gpuProcessor:
shaderDesc = OCIO.GpuShaderDesc.CreateShaderDesc()
if shaderDesc:
try:
shaderDesc.setLanguage(language)
if shaderDesc.getLanguage() == language:
setShaderDescriptionParameters(shaderDesc, sourceColorSpace, destColorSpace, "color4")
gpuProcessor.extractGpuShaderInfo(shaderDesc)
shaderCode = shaderDesc.getShaderText()
for t in shaderDesc.getTextures():
textureCount += 1
except OCIO.Exception as err:
print(err)
return shaderCode, textureCount
OCIO Resource Dependencies¶
Resource dependencies is a second major issue to examine.
In the example below we convert two different source color spaces.
One is "self-contained" in that there are no support functions being produced (ACES2065-1
),
while the second adds additional function and resources. Note that we maintain uniqueness of these additions by using setFunctionName
and setResourcePrefix
respectively.
Example 1: Self-contained¶
sourceColorSpace = 'ACES2065-1' # "acescg"
textureCount = 0
code = ''
code, textureCount = generateShaderCode2(builtinCfgC, sourceColorSpace, targetColorSpace, language)
if code:
code = code.replace("// Declaration of the OCIO shader function\n",
"// " + sourceColorSpace + " to " + targetColorSpace + " function. Texture count: %d\n" % textureCount)
code = '```c++\n' + code + '\n```'
display_markdown(code, raw=True)
// ACES2065-1 to lin_rec709 function. Texture count: 0
vec4 mx_ACES2065_1_to_lin_rec709_color4(vec4 inPixel)
{
vec4 outColor = inPixel;
// Add Matrix processing
{
vec4 res = vec4(outColor.rgb.r, outColor.rgb.g, outColor.rgb.b, outColor.a);
vec4 tmp = res;
res = mat4(2.5216861867438798, -0.27647991422992202, -0.015378064966034201, 0., -1.1341309882397199, 1.37271908766826, -0.152975335867399, 0., -0.38755519850416398, -0.096239173438334005, 1.16835340083343, 0., 0., 0., 0., 1.) * tmp;
outColor.rgb = vec3(res.x, res.y, res.z);
outColor.a = res.w;
}
return outColor;
}
Example 2: Secondary Dependencies¶
sourceColorSpace = 'ACEScc' # "acescg"
code, textureCount = generateShaderCode2(builtinCfgC, sourceColorSpace, targetColorSpace, language)
if code:
code = code.replace("// Declaration of the OCIO shader function\n",
"// " + sourceColorSpace + " to " + targetColorSpace + " function. Texture count: %d\n" % textureCount)
code = '```c++\n' + code + '\n```\n'
md = '<details><summary>Secondary Dependency Sample Code</summary>\n\n' + code + '</details>'
display_markdown(md, raw=True)
Secondary Dependency Sample Code
// Declaration of all variables
uniform sampler2D mx_ACEScc_to_lin_rec709_color4_lut1d_0Sampler;
// Declaration of all helper methods
vec2 mx_ACEScc_to_lin_rec709_color4_lut1d_0_computePos(float f)
{
float dep = clamp(f, 0.0, 1.0) * 4095.;
vec2 retVal;
retVal.y = floor(dep / 4095.);
retVal.x = dep - retVal.y * 4095.;
retVal.x = (retVal.x + 0.5) / 4096.;
retVal.y = (retVal.y + 0.5) / 2.;
return retVal;
}
// ACEScc to lin_rec709 function. Texture count: 1
vec4 mx_ACEScc_to_lin_rec709_color4(vec4 inPixel)
{
vec4 outColor = inPixel;
// Add Range processing
{
outColor.rgb = outColor.rgb * vec3(0.53763440860215062, 0.53763440860215062, 0.53763440860215062) + vec3(0.19354838709677422, 0.19354838709677422, 0.19354838709677422);
outColor.rgb = max(vec3(0., 0., 0.), outColor.rgb);
outColor.rgb = min(vec3(1., 1., 1.), outColor.rgb);
}
// Add LUT 1D processing for mx_ACEScc_to_lin_rec709_color4_lut1d_0
{
outColor.r = texture(mx_ACEScc_to_lin_rec709_color4_lut1d_0Sampler, mx_ACEScc_to_lin_rec709_color4_lut1d_0_computePos(outColor.r)).r;
outColor.g = texture(mx_ACEScc_to_lin_rec709_color4_lut1d_0Sampler, mx_ACEScc_to_lin_rec709_color4_lut1d_0_computePos(outColor.g)).r;
outColor.b = texture(mx_ACEScc_to_lin_rec709_color4_lut1d_0Sampler, mx_ACEScc_to_lin_rec709_color4_lut1d_0_computePos(outColor.b)).r;
}
// Add Matrix processing
{
vec4 res = vec4(outColor.rgb.r, outColor.rgb.g, outColor.rgb.b, outColor.a);
vec4 tmp = res;
res = mat4(0.69545224135745187, 0.044794563372037716, -0.0055258825581135443, 0., 0.14067869647029416, 0.85967111845642163, 0.0040252103059786595, 0., 0.16386906217225405, 0.0955343181715404, 1.0015006722521349, 0., 0., 0., 0., 1.) * tmp;
outColor.rgb = vec3(res.x, res.y, res.z);
outColor.a = res.w;
}
// Add Range processing
{
outColor.rgb = max(vec3(0., 0., 0.), outColor.rgb);
}
// Add Matrix processing
{
vec4 res = vec4(outColor.rgb.r, outColor.rgb.g, outColor.rgb.b, outColor.a);
vec4 tmp = res;
res = mat4(2.5216861867438798, -0.27647991422992202, -0.015378064966034201, 0., -1.1341309882397199, 1.37271908766826, -0.152975335867399, 0., -0.38755519850416398, -0.096239173438334005, 1.16835340083343, 0., 0., 0., 0., 1.) * tmp;
outColor.rgb = vec3(res.x, res.y, res.z);
outColor.a = res.w;
}
return outColor;
}
Issues With Texture Resources¶
From an integration point of view any introduction of texture lookups requires resource declarations in the code. (such as the uniform sampler2D mx_ACEScc_to_lin_rec709_color4_ocio_lut1d_0Sampler;
sampler declaration).
The only way to handle these is to have additional logic added for code insertion of color transforms, such that the shader function declarations and resources can be inserted into the code independently. The current MaterialX code generation logic does not otherwise support this using the "default color system".
Note : An experiment was attempted previously but does not align with the current proposal to have stand-alone node definitions. It was thus abandoned. ( For those interested the full code with code changes can be found here). Here 1D lookups (LUTS) were specified as input arrays and code generation created 1D textures dynamically based on the array inputs. 3D lookups were not handled.
From the point of view of creating node graphs, any implementation resource dependencies means it cannot be cleanly wrapped up into a self-contained node definition and implementation.
For now these can be "skipped" until such time as they are require, or the implementation changes to avoid using these.
Finding Transforms Using Texture Resources¶
We can re-iterate through all of the transforms of interest, and find these transforms using the following code.
Note that the code generation is not necessary but is written this way to reuse the existing utility
generateShaderCode2
).
# Scan through all the color spaces on the configs to check for texture resource usage.
testedSources = set()
for c in configs:
config = OCIO.Config.CreateFromBuiltinConfig(c)
colorSpaces = config.getColorSpaces()
for colorSpace in colorSpaces:
colorSpaceName = colorSpace.getName()
# Skip if the colorspace is already tested
if colorSpaceName in testedSources:
continue
testedSources.add(colorSpaceName)
# Test for texture resource usage
code, textureCount = generateShaderCode2(config, colorSpace.getName(), targetColorSpace, language)
if textureCount:
print('- Transform "%s" to "%s" requires %d texture resources' % (colorSpace.getName(), targetColorSpace, textureCount))
- Transform "Rec.2100-PQ - Display" to "lin_rec709" requires 1 texture resources
- Transform "ST2084-P3-D65 - Display" to "lin_rec709" requires 1 texture resources - Transform "ACEScc" to "lin_rec709" requires 1 texture resources - Transform "Rec.2100-HLG - Display" to "lin_rec709" requires 1 texture resources - Transform "ADX10" to "lin_rec709" requires 1 texture resources - Transform "ADX16" to "lin_rec709" requires 1 texture resources - Transform "Apple Log" to "lin_rec709" requires 1 texture resources - Transform "CanonLog2 CinemaGamut D55" to "lin_rec709" requires 1 texture resources - Transform "CanonLog3 CinemaGamut D55" to "lin_rec709" requires 1 texture resources
Target Language Support¶
At time of writing the target languages supported by OCIO and MaterialX differ. This includes non-core support such as MDL
and current versions of OSL
. Also as no logical operators are provided as with MaterialX, targets such as Vex
which parses and maps MaterialX nodes as operators is not easy to do.
OCIO and MaterialX recently added in Metal
language support. It is to be checked if there would be any issues with the additional struct
wrappers required for this language as it is uncommon for MaterialX code generation to call into a struct
function at the current time.
sourceColorSpace = "acescg"
language = OCIO.GpuLanguage.GPU_LANGUAGE_MSL_2_0
code, textureCount = generateShaderCode2(builtinCfgC, sourceColorSpace, targetColorSpace, language)
if code:
code = code.replace("// Declaration of the OCIO shader function\n", "// " + sourceColorSpace + " to " + targetColorSpace + " function\n")
code = '```c++\n' + code + '\n```\n'
md = '<details><summary>MSL struct usage</summary>\n\n' + code + '</details>'
display_markdown(md, raw=True)
MSL struct usage
// Declaration of class wrapper
struct mx_acescg_to_lin_rec709_color4mx_acescg_to_lin_rec709_color4
{
mx_acescg_to_lin_rec709_color4mx_acescg_to_lin_rec709_color4(
)
{
}
// acescg to lin_rec709 function
float4 mx_acescg_to_lin_rec709_color4(float4 inPixel)
{
float4 outColor = inPixel;
// Add Matrix processing
{
float4 res = float4(outColor.rgb.r, outColor.rgb.g, outColor.rgb.b, outColor.a);
float4 tmp = res;
res = float4x4(1.7050509926579815, -0.1302564175070435, -0.024003356804618042, 0., -0.62179212065700562, 1.1408047365754048, -0.1289689760649709, 0., -0.0832588720009797, -0.010548319068357653, 1.1529723328695858, 0., 0., 0., 0., 1.) * tmp;
outColor.rgb = float3(res.x, res.y, res.z);
outColor.a = res.w;
}
return outColor;
}
// Close class wrapper
};
float4 mx_acescg_to_lin_rec709_color4(
float4 inPixel)
{
return mx_acescg_to_lin_rec709_color4mx_acescg_to_lin_rec709_color4(
).mx_acescg_to_lin_rec709_color4(inPixel);
}
Using version 2.3 to access OSL
there appears to be additional issues with the code
generated as additional utility functions may be inserted which are not renamed to avoid collisions.
For example functions called max()
, pow()
etc are added which are outside the scope of the main shader declaration for which do not seem to be included in the logic for unique function name. As well as include additional include files which should be part of the OSL distribution. These need to be stripped out to embed this code as part of a larger OSL shader which already includes these functions to avoid name clashes.
An issue has been logged for this.
As OSL
integrations will generally perform color management outside of the shader, it is to be seen if this is important enough to address.
if OCIO.GpuLanguage.LANGUAGE_OSL_1:
sourceColorSpace = "acescg"
language = OCIO.GpuLanguage.LANGUAGE_OSL_1
code, textureCount = generateShaderCode2(builtinCfgC, sourceColorSpace, targetColorSpace, language)
if code:
# Bit of ugly patching to make the main function name consistent.
transformName = createTransformName(sourceColorSpace, targetColorSpace, 'color4')
code = code.replace('OSL_' + transformName, '__temp_name__')
code = code.replace(transformName, transformName + '_impl')
code = code.replace('__temp_name__', transformName)
code = code.replace("// Declaration of the OCIO shader function\n", "// " + sourceColorSpace + " to " + targetColorSpace + " function\n")
code = '```c++\n' + code + '\n```\n'
md = '<details><summary>OSL dependent function / includes code</summary>\n\n' + code + '</details>'
display_markdown(md, raw=True)
OSL dependent function / includes code
/* All the includes */
#include "vector4.h"
#include "color4.h"
/* All the generic helper methods */
vector4 __operator__mul__(matrix m, vector4 v)
{
return vector4(v.x * m[0][0] + v.y * m[0][1] + v.z * m[0][2] + v.w * m[0][3],
v.x * m[1][0] + v.y * m[1][1] + v.z * m[1][2] + v.w * m[1][3],
v.x * m[2][0] + v.y * m[2][1] + v.z * m[2][2] + v.w * m[2][3],
v.x * m[3][0] + v.y * m[3][1] + v.z * m[3][2] + v.w * m[3][3]);
}
vector4 __operator__mul__(color4 c, vector4 v)
{
return vector4(c.rgb.r, c.rgb.g, c.rgb.b, c.a) * v;
}
vector4 __operator__mul__(vector4 v, color4 c)
{
return v * vector4(c.rgb.r, c.rgb.g, c.rgb.b, c.a);
}
vector4 __operator__sub__(color4 c, vector4 v)
{
return vector4(c.rgb.r, c.rgb.g, c.rgb.b, c.a) - v;
}
vector4 __operator__add__(vector4 v, color4 c)
{
return v + vector4(c.rgb.r, c.rgb.g, c.rgb.b, c.a);
}
vector4 __operator__add__(color4 c, vector4 v)
{
return vector4(c.rgb.r, c.rgb.g, c.rgb.b, c.a) + v;
}
vector4 pow(color4 c, vector4 v)
{
return pow(vector4(c.rgb.r, c.rgb.g, c.rgb.b, c.a), v);
}
vector4 max(vector4 v, color4 c)
{
return max(v, vector4(c.rgb.r, c.rgb.g, c.rgb.b, c.a));
}
/* The shader implementation */
shader mx_acescg_to_lin_rec709_color4(color4 inColor = {color(0), 1}, output color4 outColor = {color(0), 1})
{
// acescg to lin_rec709 function
color4 mx_acescg_to_lin_rec709_color4_impl(color4 inPixel)
{
color4 outColor = inPixel;
// Add Matrix processing
{
vector4 res = vector4(outColor.rgb.r, outColor.rgb.g, outColor.rgb.b, outColor.a);
vector4 tmp = res;
res = matrix(1.7050509926579815, -0.62179212065700562, -0.0832588720009797, 0., -0.1302564175070435, 1.1408047365754048, -0.010548319068357653, 0., -0.024003356804618042, -0.1289689760649709, 1.1529723328695858, 0., 0., 0., 0., 1.) * tmp;
outColor.rgb = vector(res.x, res.y, res.z);
outColor.a = res.w;
}
return outColor;
}
outColor = mx_acescg_to_lin_rec709_color4_impl(inColor);
}
Creating MaterialX Node Definitions / Implementation¶
Given source code for now, it is still possible to create implementations and node definitions. If in the future the implementations can become MaterialX node graphs then the definition interface can still be used.
To create a new definition:
- The source and target color space is used to derive:
- a transform name: using the previously described
createTransformName()
utility - a node name by replace the 'mx_' function name with the "standard" 'ND_' prefix used for definition
- a node category
- a transform name: using the previously described
- Use
addNodeDef()
API to add a new definition - Set node group to be
colortransform
which is the recommended group for new color transforms. - Add a single input and output of the appropriate type (
color3
orcolor4
). - Set a default value on the input. For this code we assume the defaul to be opaque black.
Thie logic is encapsulated in a new generateMaterialXDefinition()
utility function.
def generateMaterialXDefinition(doc, sourceColorSpace, targetColorSpace, inputName, type):
# Create a definition
transformName = createTransformName(sourceColorSpace, targetColorSpace, type)
nodeName = transformName.replace('mx_', 'ND_')
comment = doc.addChildOfCategory('comment')
comment.setDocString(' Color space %s to %s transform. Generated via OCIO. ' % (sourceColorSpace, targetColorSpace))
definition = doc.addNodeDef(nodeName, 'color4')
category = sourceColorSpace + '_to_' + targetColorSpace
definition.setNodeString(category)
definition.setNodeGroup('colortransform')
defaultValueString = '0.0 0.0 0.0 1.0'
defaultValue = mx.createValueFromStrings(defaultValueString, 'color4')
input = definition.addInput(inputName, type)
input.setValue(defaultValue)
output = definition.getOutput('out')
output.setAttribute('default', 'in')
return definition
Another utility called writeShaderCode()
is used to write the code to file following the "standard" MaterialX naming convention.
def writeShaderCode(code, transformName, extension, target):
# Write source code file
filename = mx.FilePath('./data') / mx.FilePath(transformName + '.' + extension)
print('Write target[%s] source file %s' % (target,filename.asString()))
f = open(filename.asString(), 'w')
f.write(code)
f.close()
The implementation creation logic is encapsulated in a createMaterialXImplementation()
utility function which appends a new implementation to an existing Document.
The transform name is assumged to be precreated using createTransformName()
.
This is used to create a unique implementation Element name, and source code filename. We decied to use a file on disk as it is not possible to inline 1 or more function functions created by OCIO.
The implementation is associated with an existing node definition which is passed in and a target is explicit set to indicate which shading language (target) the implementation is for.
def createMaterialXImplementation(doc, definition, transformName, extension, target):
'''
Create a new implementation in a document for a given definition.
'''
implName = transformName + '_' + target
filename = transformName + '.' + extension
implName = implName.replace('mx_', 'IM_')
# Check if implementation already exists
impl = doc.getImplementation(implName)
if impl:
print('Implementation already exists: %s' % implName)
return impl
comment = doc.addChildOfCategory('comment')
comment.setDocString(' Color space %s to %s transform. Generated via OCIO for target: %s'
% (sourceColorSpace, targetColorSpace, target))
impl = doc.addImplementation(implName)
impl.setFile(filename)
impl.setFunction(transformName)
impl.setTarget(target)
impl.setNodeDef(definition)
return impl
Finally a small utility is added to write the document to disk.
def writeDocument(doc, filename):
print('Write MaterialX file:', filename.asString())
mx.writeToXmlFile(doc, filename)
Using these utilities we:
- Create separate definition and implementation Documents.
- Generate shader code for
GLSL
,MSL
, andOSL`` for the same color transform. (
OSL` is available in version 2.3). - Create a new definition for that transform
- Create a new implementation for each shader code result
# Example to create:
# - source code for a given transform for 2 shader targets
# - A definition wrapper for the source
# - An implementations per source code. All implementations are associated with single definition
definitionDoc = mx.createDocument()
definition = None
implDoc = mx.createDocument()
sourceColorSpace = "acescg"
type = 'color4'
transformName = createTransformName(sourceColorSpace, targetColorSpace, type)
# All code has the same input name
# It is possible to use a different name than the name used in the generated function ('inPixel')
#IN_PIXEL_STRING = 'inPixel'
IN_PIXEL_STRING = 'in'
# Pick a source and target color space
sourceColorSpace = 'acescg'
targetColorSpace = 'lin_rec709'
# List of MaterialX target language, source code extensions, and OCIO GPU languages
generationList = [#['genmsl', 'metal', OCIO.GpuLanguage.GPU_LANGUAGE_MSL_2_0],
['genglsl', 'glsl', OCIO.GpuLanguage.GPU_LANGUAGE_GLSL_4_0] ]
if OCIO.GpuLanguage.LANGUAGE_OSL_1:
generationList.append(['genosl', 'osl', OCIO.GpuLanguage.LANGUAGE_OSL_1])
for gen in generationList:
target = gen[0]
extension = gen[1]
language = gen[2]
code, textureCount = generateShaderCode2(builtinCfgC, sourceColorSpace, targetColorSpace, language)
if code:
# Emit the source code file
writeShaderCode(code, transformName, extension, target)
# Create the definition once
if not definition:
definition = generateMaterialXDefinition(definitionDoc, sourceColorSpace, targetColorSpace,
IN_PIXEL_STRING, type)
# Create the implementation
createMaterialXImplementation(implDoc, definition, transformName, extension, target)
Write target[genglsl] source file .\data\mx_acescg_to_lin_rec709_color4.glsl Write target[genosl] source file .\data\mx_acescg_to_lin_rec709_color4.osl
The resulting MaterialX wrappers are then written to disk as follows:
Variant Creation¶
Given a node definition for the color4
variant is is possible to create a functional graph and corresponding definition for a color3
variant. The graph simple convertes from color3
to color4
before connecting to the transform and then converts back to color3
for output.
color4Name = definition.getName()
color3Name = color4Name.replace('color4', 'color3')
color3Def = definitionDoc.addNodeDef(color3Name)
color3Def.copyContentFrom(definition)
c3input = color3Def.getInput(IN_PIXEL_STRING)
c3input.setType('color3')
c3input.setValue(mx.createValueFromStrings('0.0 0.0 0.0', 'color3'))
ngName = color3Def.getName().replace('ND_', 'NG_')
ng = definitionDoc.addNodeGraph(ngName)
c4instance = ng.addNodeInstance(definition)
c4instance.addInputsFromNodeDef()
c4instanceIn = c4instance.getInput(IN_PIXEL_STRING)
c3to4 = ng.addNode('convert', 'c3to4', 'color4')
c3to4Input = c3to4.addInput('in', 'color3')
c4to3 = ng.addNode('convert', 'c4to3', 'color3')
c4to3Input = c4to3.addInput('in', 'color4')
ngout = ng.addOutput('out', 'color3')
#ngin = ng.addInput('in', 'color3')
ng.setNodeDef(color3Def)
c4instanceIn.setNodeName(c3to4.getName())
c4to3Input.setNodeName(c4instance.getName())
ngout.setNodeName(c4to3.getName())
c3to4Input.setInterfaceName(IN_PIXEL_STRING)
result = mx.writeToXmlString(definitionDoc)
display_markdown('#### Generated Color3 Variant\n' + '```xml\n' + result + '```\n', raw=True)
valid, log = definitionDoc.validate()
if not valid:
print('Document created is invalid:', log)
filename = mx.FilePath('./data') / mx.FilePath(definition.getName() + '_2.' + 'mtlx')
print('Write MaterialX definition variant file:', filename.asString())
mx.writeToXmlFile(definitionDoc, filename)
Generated Color3 Variant¶
<?xml version="1.0"?>
<materialx version="1.39">
<!-- Color space acescg to lin_rec709 transform. Generated via OCIO. -->
<nodedef name="ND_acescg_to_lin_rec709_color4" node="acescg_to_lin_rec709" nodegroup="colortransform">
<output name="out" type="color4" default="in" />
<input name="in" type="color4" value="0, 0, 0, 1" />
</nodedef>
<nodedef name="ND_acescg_to_lin_rec709_color3" node="acescg_to_lin_rec709" nodegroup="colortransform">
<output name="out" type="color3" />
<input name="in" type="color3" value="0, 0, 0" />
</nodedef>
<nodegraph name="NG_acescg_to_lin_rec709_color3" nodedef="ND_acescg_to_lin_rec709_color3">
<acescg_to_lin_rec709 name="node1" type="color4" nodedef="ND_acescg_to_lin_rec709_color4">
<input name="in" type="color4" value="0, 0, 0, 1" nodename="c3to4" />
</acescg_to_lin_rec709>
<convert name="c3to4" type="color4">
<input name="in" type="color3" interfacename="in" />
</convert>
<convert name="c4to3" type="color3">
<input name="in" type="color4" nodename="node1" />
</convert>
<output name="out" type="color3" nodename="c4to3" />
</nodegraph>
</materialx>
Document created is invalid: Node input has too many bindings: <input name="in" type="color4" value="0, 0, 0, 1" nodename="c3to4"> Write MaterialX definition variant file: .\data\ND_acescg_to_lin_rec709_color4_2.mtlx
MaterialX Standard Library Inclusion¶
It is possible to add a new color space transform to the "standard" MaterialX library locations. This can be done for local testing for in some cases when additional search paths are not supported.
Introduced with version 1.38.7, the location of definitions and implementations can be found in the cmlib
folder under the installation libraries
location. Note that target specific source file directories were removed as all implementations are now target independent node graphs.
To include source file implementations the appropriate target
folders can be added in.
Under that folder an implementation file can be added for each target along with associated source files. The new definition can be added to `cmlib_defs.mtlx.
See the Creating Definitions book for more on how set up new definitions.
The standard library folder structure is partially shown below. The bolded items would be the ones of interest. The stdlib
folder is also shown to show how each folder follows the same naming and hierarchy conventions.
- cmlib // Color space transform library
- cmlib_defs.mtlx // definitions file
- cmlib_ng.mtlx // functional node graphs file
- genglsl // GLSL implementation folder
- cmlib_genglsl_impl.mtlx // Implementation declarations
- GLSL files reside here
- genmsl // MSL implementation folder
- cmlib_genmsl_impl.mtlx // Implementation declarations
- MSL files reside here
- stdlib // "Standard" node library
- genglsl
- stdlib_genglsl_impl.mtlx
- genmdl
- genmsl
- stdlib_genmsl_impl.mtlx
- genosl
- targets
- genglsl
Future Exploration¶
At time of writing (September 2024), the NanoColor initiative is underway. The current plan is to allow pre-processing of transforms to create MaterialX definitions with graph implementation. See the next section for some unofficial prototyping.
As there is the intent to provide Javascript bindings it will be interesting to see how this will interact with Web libraries and how it can work with the glTF Texture Procedurals extension.
NodeGraph Prototype¶
The following is some prototype code to extract out the list of transforms and attempt to create corresponding MaterialX nodes. It currently only handles matrix operations.
As a first step instead of extracting out code the list of transforms
is extracted. A GPU processor or CPU processor can be used. (Not sure if there is any difference which processor is used). This logic is given in the generateTransformGraph
utility
def generateTransformGraph(config, sourceColorSpace, destColorSpace):
if not config:
return None
# Create a processor for a pair of colorspaces (namely to go to linear)
processor = None
groupTransform = None
try:
processor = config.getProcessor(sourceColorSpace, destColorSpace)
except:
print('Failed to get processor for: %s -> %s' % (sourceColorSpace, destColorSpace))
return groupTransform
if processor:
processor = processor.getOptimizedProcessor(OCIO.OPTIMIZATION_ALL)
groupTransform = processor.createGroupTransform()
return groupTransform
Next we use this to extract out the list of transforms and iterate over them to pull out information per transform.
The basic 'template' for creating a new MaterialX definition is a single new nodedef
with one input and one output. In this case we use color3
as the input and output type.
For each transform a new corresponding MaterialX node is created. For now the logic only handles matrix tranforms. These are chained together incrementally. All computations work on vector3
data.
All nodes are encapsulated into a functional
nodegraph implementation with the appropriate conversion to/from color3
to vector3
is performed.
groupTransform = generateTransformGraph(builtinCfgC, sourceColorSpace, targetColorSpace)
result = f'{groupTransform}'
display_markdown('#### Extracted Transform\n' + '```xml\n' + result + '```\n', raw=True)
# To add. Proper testing of unsupported transforms...
invalidTransforms = [ OCIO.TransformType.TRANSFORM_TYPE_LUT3D, OCIO.TransformType.TRANSFORM_TYPE_LUT1D,
OCIO.TransformType.TRANSFORM_TYPE_RANGE,
OCIO.TransformType.TRANSFORM_TYPE_GRADING_PRIMARY ]
# Create a document, a nodedef and a functional graph.
graphDoc = mx.createDocument()
outputType = 'color3'
xformName = sourceColorSpace + '_to_' + targetColorSpace + '_' + outputType
nd = graphDoc.addNodeDef('ND_' + xformName )
nd.setAttribute('node', xformName)
ndInput = nd.addInput('in', 'color3')
ndInput.setValue(mx.createValueFromStrings('0.0 0.0 0.0', 'color3'))
ng = graphDoc.addNodeGraph('NG_' + xformName)
ng.setAttribute('nodedef', nd.getName())
convertNode = ng.addNode('convert', 'asVec', 'vector3')
converInput = convertNode.addInput('in', 'color3')
converInput.setInterfaceName('in')
print(f'Transform from: {sourceColorSpace} to {targetColorSpace}')
print(f'Number of transforms: {groupTransform.__len__()}')
previousNode = None
# Iterate and create appropriate nodes and connections
for i in range(groupTransform.__len__()):
transform = groupTransform.__getitem__(i)
# Get type of transform
transformType = transform.getTransformType()
if transformType in invalidTransforms:
print(f'- Transform[{i}]: {transformType} contains an unsupported transform type')
continue
#print(f'- Transform[{i}]: {transformType}')
if transformType == OCIO.TransformType.TRANSFORM_TYPE_MATRIX:
matrixNode = ng.addNode('transform', ng.createValidChildName(f'matrixTransform'), 'vector3')
# Route output from previous node as input of current node
inInput = matrixNode.addInput('in', 'vector3')
if previousNode:
inInput.setAttribute('nodename', previousNode)
else:
if i==0:
inInput.setAttribute('nodename', 'asVec')
else:
inInput.setValue(mx.createValueFromStrings('0.0 0.0 0.0', 'vector3'))
# Set matrix value
matInput = matrixNode.addInput('mat', 'matrix33')
matrixValue = transform.getMatrix()
# Extract 3x3 matrix from 4x4 matrix
matrixValue = matrixValue[0:3] + matrixValue[4:7] + matrixValue[8:11]
matrixValue = ', '.join([str(x) for x in matrixValue])
#print(' - Matrix:', matrixValue)
matInput.setAttribute('value', matrixValue)
# Add offset value
offsetValue = transform.getOffset()
offsetValue = ', '.join([str(x) for x in offsetValue])
#print(' - Offset:', offsetValue)
# Add a add vector3 to graph
previousNode = matrixNode.getName()
#elif transformType == OCIO.TransformType.TRANSFORM_TYPE_LOG:
# print(' - Base:', transform.getBase())
# Create an output for the last node if any
convertNode2 = ng.addNode('convert', 'asColor', 'color3')
converInput2 = convertNode2.addInput('in', 'vector3')
converInput2.setAttribute('nodename', previousNode)
if previousNode:
out = ng.addOutput(ng.createValidChildName('out'), 'color3')
out.setAttribute('nodename', 'asColor')
# Write the graph document
print('---------------------------')
print('Write OCIO transform graph file:', xformName + '.' + 'mtlx')
filename = mx.FilePath('./data') / mx.FilePath(xformName + '.' + 'mtlx')
mx.writeToXmlFile(graphDoc, filename)
result = mx.writeToXmlString(graphDoc)
display_markdown('#### Generated Transform Graph\n' + '```xml\n' + result + '```\n', raw=True)
Extracted Transform¶
<GroupTransform direction=forward, transforms=
<MatrixTransform direction=forward, fileindepth=unknown, fileoutdepth=unknown, matrix=1.705050992657982 -0.6217921206570056 -0.0832588720009797 0 -0.1302564175070435 1.140804736575405 -0.01054831906835765 0 -0.02400335680461804 -0.1289689760649709 1.152972332869586 0 0 0 0 1, offset=0 0 0 0>>```
Transform from: acescg to lin_rec709 Number of transforms: 1 --------------------------- Write OCIO transform graph file: acescg_to_lin_rec709_color3.mtlx
Generated Transform Graph¶
<?xml version="1.0"?>
<materialx version="1.39">
<nodedef name="ND_acescg_to_lin_rec709_color3" node="acescg_to_lin_rec709_color3">
<output name="out" type="color3" />
<input name="in" type="color3" value="0, 0, 0" />
</nodedef>
<nodegraph name="NG_acescg_to_lin_rec709_color3" nodedef="ND_acescg_to_lin_rec709_color3">
<convert name="asVec" type="vector3">
<input name="in" type="color3" interfacename="in" />
</convert>
<transform name="matrixTransform" type="vector3">
<input name="in" type="vector3" nodename="asVec" />
<input name="mat" type="matrix33" value="1.7050509926579815, -0.6217921206570056, -0.0832588720009797, -0.1302564175070435, 1.1408047365754048, -0.010548319068357653, -0.024003356804618042, -0.1289689760649709, 1.1529723328695858" />
</transform>
<convert name="asColor" type="color3">
<input name="in" type="vector3" nodename="matrixTransform" />
</convert>
<output name="out" type="color3" nodename="asColor" />
</nodegraph>
</materialx>