<div class="gmail_quote">On Wed, Jun 6, 2012 at 3:31 PM, Boudewijn Rempt <span dir="ltr"><<a href="mailto:boud@valdyas.org" target="_blank">boud@valdyas.org</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
Here are some notes I prepared today. It's a long read, but please help me out by pointing out mistakes and options I've forgotten. I've got a feeling we might have to refactor our display code quite a bit to make this work correctly, and fast -- especially because in git master there's quite a<br>
bit of brokenness.<br>
<br>
Contents:<br>
<br>
* color management in Nuke<br>
* color management in Mari<br>
* color management in Krita<br>
** Master<br>
** Lut branch<br>
** With Ocio<br>
<br>
The goal is to make Krita function correctly again with all HDR colorspaces and make opengl and<br>
qpainter canvas support the following for HDR colorspaces:<br>
<br>
* exposure<br>
* gamma<br>
* lut selection<br>
* painting<br>
* color selection (didn't check how this is done in Nuke and Mari yet, probably not in the same way as in Krita, where you use the ordinary colors selectors to select a 8bit int rgba that gets converted using the exposure settng).<br>
<br>
------------------------------<u></u>-----------<br>
* Color Management in Nuke<br>
<br>
"The gain and gamma sliders let you adjust the displayed image, without<br>
affecting your final output. These controls are useful for tasks like spotting<br>
holes in mattes."<br>
<br>
"Press the Zebra Stripe button to apply stripes to all pixels outside the range<br>
0.0 to 1.0."<br>
<br>
"Input Process and Viewer Process operations can be used to modify the<br>
image from the viewed node before it is displayed on your monitor."<br>
<br>
Note: the getting started says Input Process is deprecated.<br>
<br>
"There are two predefined Viewer Processes, sRGB and rec709,"<br>
<br>
" Viewer Process are executed on the GPU. 1D LUT and 3D LUT (Vectorfield) have GPU implementations, so the built-in Viewer Processes will run on the GPU (unless gl buffer depth has been set to byte in the Viewer settings, in which case all processing is done on the CPU). "<br>
<br>
"Nuke includes the following predefined Viewer Process gizmos: sRGB and<br>
rec709. By default, sRGB is used because it is a best-guess default for a<br>
typical computer monitor."<br>
<br>
*Conclusion*: Nuke is pretty simple-minded. It doesn't care that the monitor might be calibrated or not, it just allows users to select "profiles" that change the look, without any care for whether the same project would look the same with a different monitor.<br>
<br></blockquote><div><br>I guess someone would get shot for the first sentence in Simon's place ;) Would be a bit strange if an application that costs a few thousand euro couldn't do that.<br><br>It appears to me that Nuke delegates that to the user. The calibration is done with LUTs instead of the ICC profiles. The calibration itself seems to be just another node.<br>
</div><blockquote class="gmail_quote" style="margin:0pt 0pt 0pt 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">
<br>
* Color Management in Mari<br>
<br>
Mari has both a dialog and a color magement toolbar. The color management toolbar seems identical with the list of features in the ocio manual, down to showing only luminance or individual rgb components.<br>
<br>
The toolbar has the following items:<br>
<br>
Set how colors are displayed using the ColorSpace toolbar:<br>
1. Toggle to enable Enable Color Management in the Color Manager<br>
2. Select a custom OpenColorIO (.ocio) configuration file by clicking<br>
3. Select the colorspace of the current scene using Input ColorSpace<br>
dropdown.<br>
4. Set the display device used to view the scene using the Display Device<br>
dropdown menu.<br>
5. Select the colorspace transform to apply to the scene from the View<br>
Transform dropdown menu.<br>
6. Use the Component dropdown menu to see the individual color<br>
components in the scene.<br>
7. Use the Gain controls to set the amount of exposure adjustment applied<br>
before the display transform by either entering a multiplier (exposure<br>
value), dragging on the slider, or adjusting the F-Stop value.<br>
8. Finally, set the amount of Gamma correction applied after the display<br>
transform by entering a gamma level or by dragging the gamma slider.<br>
<br>
There are also some things called "display filters" That sounds like it is a way to do color correction similar to GIMP -- but I might be confused about the way they mean "color correction" in the Mari manual.<br>
<br>
The mari color management toolbar is included in the ocio distribution, so it's pretty easy to study.<br>
<br>
Conclusion: we can implement, probably, the entire set of Mari functionality in the Lut docker with some changes to our canvas code. Enabling OCIO would then _disable_ what I think of as "real" color management.<br>
<br>
------------------------------<u></u>------------------------------<u></u>------------------<br>
<br>
* Color Management in Krita<br>
<br>
Krita currently follows a traditional icc-based colormanagement model. It uses lcms2 and ctlcs. Krita used to support lcms1 and a kubelka-munk (spectral) color engine. Krita can convert pixels between all the color engines it supports. Krita doesn't have an ocio-based color engine yet.<br>
<br>
Quick glossary:<br>
<br>
Colormodel : pixel arrangement (rgb, bgr, yuv etc, so many bytes per component)<br>
Profile : icc, ctlcs or other profile that describes what the pixel<br>
values mean and thus allows conversion<br>
Colorspace : combination of model and profile<br>
Engine : a plugin that provides colorspace and profile implementations.<br>
<br>
+ Krita doesn't support spot channels (yet)<br>
+ Every layer has a working colorspace. Layers in an image can all have different colorspaces.<br>
+ The image has a working colorspace: on composition, all layers are converted to the working colorspace and combined together<br>
+ The monitor has a profile: this describes the way the monitor deviates from other monitors and is necessary to make sure the same image looks the same on two systems.<br>
+ a colorspace can be marked to support HDR<br>
+ we want to be able to have a natural-feeling color selection method for HDR colorspaces<br>
<br>
** In Master<br>
<br>
+ ctlcs colorspaces are not marked as HDR: here exposure is applied during image recomposition by embedding the exposure information in the profile.<br>
+ the new lcms float colorspaces are marked as HDR, but the exposure shader is broken.<br>
+ ctlcs colorspaces show exposure even in the qpainter canvas (because exposure is applied during composition)<br>
+ painting on ctlcs float colorspaces works through converting the selected fg/bg color to float using exposure embedded in the ctlcs profile.<br>
<br>
** In the lut-docker branch<br>
<br>
** Display<br>
<br>
+ exposure and gamma are stored in KisCanvasResourceProvider<br>
<br>
*** OpenGL<br>
<br>
In the lut branch (master is broken for opengl as far as HDR is concerned, it is not used at all):<br>
<br>
if (image colorspace is integer):<br>
1. convert the image to 8 or 16 bit integer rgba textures using the monitor profile. Currently broken.<br>
2. show the textures<br>
<br>
elif (float):<br>
1. convert the image to 16 or 32 bit float rgba textures. This doesn't use the monitor profile (since that often isn't compatible with float), but the default profile given by the image colorspace. I.e., no real conversion.<br>
2. apply the exposure/gamma shader to the textures<br>
3. show the textures<br>
<br>
*** QPainter<br>
1. convert the image to 8 bit integer using the monitor profile<br>
2. scale the image<br>
3. display the image<br>
<br>
<br>
** OCIO<br>
<br>
+ Ocio has ocio2icc so, command line tool to generate an ICC “proofing” profile from a color space transform, which can be used in applications such as Photoshop. We don't use proofing transforms in Krita, do we?<br>
<br>
+ Ocio can be used in the display part of Krita, it can generate corrected textures or pixels. We could use Ocio in the lut-docker and if enabled make it replace the final step of conversion from the projection to the display, or we could make it a step in between.<br>
<br>
+ ocio can use the cpu or gpu to perform its transformations.<br>
<br>
This situation would make Krita function optionally like Mari for all practical purposes.<br>
<br>
** Possible Refactoring of Krita's display architecture<br>
<br>
* create the ocio-based lut docker, which is only active for float images<br>
* the lut docker allows the user to select any of the configured luts; this transformation will be applied on the projection. Then another transformation should be applied to correct the result for the monitor idiosyncracies.<br>
* create a step in both canvases where the image is filtered using ocio.<br>
</blockquote><div><br>I think you need to do more than just the lut docker. The image would have an ocio colorspace and we can't convert between icc based and ocic easily.<br><br>To make that easier we might limit the layer colospace to the image colospace to avoid conversion on lower levels too.<br>
</div><blockquote class="gmail_quote" style="margin:0pt 0pt 0pt 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">
** OpenGL<br>
<br>
This depends a lot on where we'll go with the opengl2 canvas. But ocio can proably replace our home-grown exposure shader.<br></blockquote><div><br>Just if Krita is running in ocio mode, lcms would still need a shader or so.<br>
</div><blockquote class="gmail_quote" style="margin:0pt 0pt 0pt 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">
We want to:<br>
<br>
* make it faster<br>
* support high channel depth monitors and cards<br>
* apply the monitor profile<br>
<br>
if (ocio and float):<br>
1. make ocio generate textures from the image with all settings<br>
2. how the deuce do we apply the monitor calibration profile? I don't know yet... (DisplayTransform?)<br>
elif (not ocio):<br>
1. convert image to the right texture format for the monitor capabilities using lcms<br>
2. upload the textures<br>
3. show the textures<br>
<br>
<br>
** QPainter<br>
<br>
if (not ocio):<br>
1. convert the image to 8 bit integer using the monitor profile<br>
2. scale the image<br>
3. display the image<br>
elif (ocio):<br>
if (float):<br>
1. convert image pixels using lcms to a known good space for ocio, if necessary.<br>
2. convert image pixels using ocio (exposure, gamma, other ocio tricks) to 8 bit rRGB rgba. Use the ocio cpu path.<br>
2a. make sure the ocio configuration is available as icc so we can create<br>
a correct Krita lmcs2-based colorspace<br>
3. convert converted pixels using lcms2 and the monitor profile to image<br>
4. pre-scale image<br>
5. show image<br></blockquote></div><br>I think if you go with lcms->ocio->lcms, that defeats the whole purpose of using ocio in the first place. I think we could just assume that somebody who uses ocio will use opengl canvas.<br>