File: DeepIDsSpecification.rst

package info (click to toggle)
openexr 3.4.6%2Bds-2
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 41,948 kB
  • sloc: cpp: 166,793; ansic: 26,769; python: 2,990; sh: 581; makefile: 9
file content (407 lines) | stat: -rw-r--r-- 17,966 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407

..
  SPDX-License-Identifier: BSD-3-Clause
  Copyright Contributors to the OpenEXR Project.

OpenEXR Deep IDs Specification
##############################


Introduction
============

Deep IDs are primarily used in compositing applications to select
objects in images:

-  The 3D renderer stores multiple ids per
   pixel in the final OpenEXR file as well a scene manifest providing a mapping to a
   human-readable identifier, such as an object or material name.
-  The compositing application reads in the deep IDs and implements a
   UI for the artist to create a selection based on the manifest data.
-  The selected IDs will then be used to create a deep (per-sample) or
   shallow (per-pixel) mask for later use.

Typical uses include:

-  Grading selected objects.
-  Removing a certain percentage of particles from a snow or dust motes
   render.
-  Adding consistent variations to different individuals in a crowd or to different
   trees in a forest by applying an ID-based color correction.

Deep IDs can also be used for debugging:

-  To identify an object/material which renders incorrectly.
-  To collect all visible IDs over a sequence so as to prune scene
   objects that are never on screen.

Pros
----

-  Each pixel can contain an arbitrary number of IDs (0 included),
   allowing for perfect mattes.
-  Deep IDs are combined with transparency data to support anti-aliasing,
   motion blur and depth of field.
-  A deep image stores the color of the individual objects as well as their
   individual alpha, so an object extracted by ID will have the correct color
   and alpha values along the edges.

Cons
----

-  IDs cannot be directly inspected with a simple image viewer.
-  Hash collisions can make multiple objects map to the same ID.
-  DeepIDs require software to support deepscanline and/or deeptile OpenEXR image types.



Cryptomatte comparison
----------------------

Cryptomatte [1]_ is widely adopted ID scheme similar to that described here, with a few differences:

-  Cryptomatte uses a similar concept of a manifest, which is stored as JSON in a string attribute.
-  Cryptomatte allocates a fixed size array for IDs, imposing a maximum number of IDs in one pixel.
   Setting this too large requires a lot of memory; setting it too small can cause
   noise if an object is selected which has been discarded from some pixels.
-  Cryptomatte uses the ``scanlineimage`` or ``tiledimage`` EXR format, rather than ``deepscanline`` or ``deeptile``
   to simplify adoption by applications already supporting these EXR types.
   Similarly, IDs are stored in the ``R`` and ``B`` channels as FLOAT rather than UINT types,
   though are intended to be interpreted as UINT. Although these
   channels are named as color channels, care must be taken not to color manage or otherwise modify them,
   and tools which re-write Cryptomatte files should ensure they are written as FLOAT rather than HALF.
-  Cryptomatte allows slightly less than 2\ :sup:`32` different IDs; the Deep ID scheme allows for 2\ :sup:`32` or 2\ :sup:`64` different IDs.
   This reduces the chance of a hash collision.
-  Cryptomatte only stores *coverage* of each object. This can be used to generate a mask for an object to apply
   a selective grade. Because the depth ordering, transparency and color of each object is not stored, objects isolated
   using Cryptomatte will have contaminated edge colors and incorrect alpha values.

Comparisons of typical images suggest Cryptomatte and deep images with IDs are similar size on disk,
even though deep images carry more information,
but generally deep images are processed faster since they take less space when decompressed.

It is possible to convert a Deep ID image into a Cryptomatte image, which may be a path to adoption for
tools which read Cryptomatte but not Deep IDs. Conversion in the opposite direction is also possible,
but the resultant deep ID image can only be used for coverage computation.

As such, the Deep ID approach can be considered an evolution of Cryptomatte, supporting and extending all the features of Cryptomatte,
but with a more formal specification and flexible storage.


Deep ID Basics
==============


The deep OpenEXR files need to contain the following elements:

-  Deep IDs stored using one or two ``Imf::UINT`` (``uint32_t``)
   channels:

   -  A single channel stores 32 bit hashes.
   -  A pair of channels store 64 bit hashes.

-  Manifest data stored in the file’s metadata as an attribute, or in a side-car file.

   -  **NOTE**:  OpenEXR 3.0+ provides a data structure
      and attribute for efficient storage. For more details see :ref:`idmanifest-label`


Sample storage
==============


Terminology
-----------

-  **Channel**: Every deep image contains many named channels (``R``,
   ``G``, ``B``, ``A``, ``Z``, ``id``, etc)
-  **Pixel**: A discrete image element containing 0 or more deep
   ``samples``.
-  **Sample**: A container storing the value of each channel at a
   specific depth.
-  **Id**: A unique numerical identifier mapping to an ``item``.
-  **Kind**: a named ``id`` category (model name, shader name, asset management URI, etc).
-  **Manifest**: a data structure mapping each id to one ``item``.
-  **Item**: an entity (object, material, etc) represented by:

   -  Artist friendly strings in the manifest, e.g. “Chair16” (one string of each ``kind``)
   -  A collection of ``samples`` matching a particular ``id`` in the image.

Principles
----------

1. Deep IDs have a single ``id`` of each ``kind`` per ``sample``, and
   every ``item`` in the image has a consistent ``id`` for all its
   ``samples``.
2. If two different ``items`` overlap the same space, they will be
   stored in separate ``samples`` (which themselves may or may not
   overlap).
3. Some ``id`` may not have an associated string, like ``particleid``,
   and still be useful (for example through image-based picking).
4. In complex cases like an instanced particle system, each ``sample``
   may have an ``instanceid``, ``particleid``, and ``id``/``objectid``.

Standard ID Channel names
-------------------------

============== ======================================
Name           Contents
============== ======================================
**id**         by default, an object identifier  [2]_
**objectid**   identifier of an object
**materialid** identifier of an material
**particleid** identifier of a particle
**instanceid** identifier of an instanced object
============== ======================================


To limit the risk of hash collision, a ``uint64_t`` bits can be encoded
with two ``uint32_t`` channels. The convention is then to suffix the
channel name with ``0`` or ``1`` to indicate the channels storing the
least and the most significant 32 bits respectively,
e.g. ``particleid0`` and  ``particleid1`` or ``particle.id0`` and ``particle.id1``

When sorted alphanumerically, the channel storing the most significant bits should appear immediately
after the channel storing the least significant bits. See appendix for details.

ID generation
-------------

Any non-uint32 identifier can be hashed to generate an id.

-  ``uint32_t`` hashes can be generated with ``MurmurHash3_x86_32``.
-  ``uint64_t`` hashes can be generated with the top 8 bytes of
   ``MurmurHash3_x64_128``.

OpenEXR 3.x offers two convenience functions to hash strings  [3]_:
``IDManifest::MurmurHash32`` and ``IDManifest::MurmurHash64``.

The Deep ID scheme does not require these hash schemes to be used. For example if an asset management
system already generates hashes, those can be used instead. It is also not required to use a hash scheme
to map between names and numeric identifiers. For example the IDs could be generated in the order that they
are stored in the source geometry file. Such an approach would avoid hash collisions,
but would not generate IDs consistently across different shots or scenes.


Multivariate IDs
----------------

Hashing more than one ``kind`` (i.e. object + material) limits storage
requirements without impairing the artist’s ability to generate
arbitrary mattes.

For examples hashing the object and material names together is common
practice. In that case, a single ``id`` will map to two ``kinds`` in the
manifest, providing more flexibility at the cost of a slightly increased
risk of hash collision.

Manifest data
-------------

The manifest contains the human-readable data corresponding to a given
hash. It is a big table of strings that may require more storage than
the actual image data. It can be stored using the following mechanisms:

.. _idmanifest-label:

OpenEXR idmanifest container
^^^^^^^^^^^^^^^^^^^^^^^^^^^^

Since OpenEXR 3.0, there is a new standard ``idmanifest`` attribute
using a ``CompressedIDManifest`` metadata type, specially designed to
transport manifest data efficiently [4]_. It is optimized to reduce the storage space required,
and is the most standard approach.

The utility ``exrmanifest`` outputs the manifest of EXR images as plain text.

Note that use of the ``idmanifest`` attribute is not limited to the Deep ID scheme.
Other schemes - such as Cryptomatte - could also use the attribute to encode manifest
and other related data, to save storage space.
(Indeed, the idmanifest was in part derived from Cryptomatte’s metadata scheme for this purpose)


OpenEXR string container
^^^^^^^^^^^^^^^^^^^^^^^^

The manifest can be stored in ``string`` or ``stringvector`` attributes,
but this is not very efficient and may significantly increase file size.

Side-car files
^^^^^^^^^^^^^^

Alternatively, the manifest may be stored in a separate file, with an OpenEXR attributes,
a database or a file naming convention used to associate one or more OpenEXR files
with the corresponding sidecar file. Sidecar files can be advantageous because
they can be shared between different images, and also updated as more content is being rendered.

Such schemes are not supported by the OpenEXR library, nor are they defined here,
since that is outside the scope of the OpenEXR file format specification.
Although sidecar files may be appropriate for temporary usage, it is strongly recommended
that the embedded manifest is used in OpenEXR images which are to be shared between different companies
or for archival usage.



Deep Compositing and Deep IDs
=============================


Deep Compositing workflows allow objects to be rendered in separate passes and then merged together as a post process.
For example, it is possible to render a volumetric element such as a smoke, and merge other objects into it.
Deep images containing such volume renders may need to become very large in order to capture all the detail required for the merge to work correctly.
Deep IDs are a powerful addition to deep compositing workflows, as they allow for greater control of individual elements within a single render.
For this to work effectively, the ID passes must be included in the same deep image as that used for deep compositing.
It is tempting to generate two separate deep images for each rendered object, one containing color and one containing the IDs,
particularly since this is a common approach with Cryptomatte workflow.
However, creating separate files complicates deep compositing with deep IDs, since it is harder to associate each individual object's ID with its color.
Very little extra storage is required to add an ID channel to a deep image.

Deep IDs are still useful where full deep compositing workflows are not used, to allow color correction of individual objects.
In that case, the deep image need not be large. In the case of a volume, a single sample can be used to indicate the volume's ID, color and transparency.
Renderers can produce these efficient passes by combining together adjacent deep samples that have the same set of id channels.
The object color (RGB) channels can also be omitted to reduce file size at the cost of loss of fidelity along object edges.

Tools which do not support full deep compositing workflows could still support deep images for the limited use of ID selection.
For example an image editing package may have a "load selected objects from deep image" tool which allows the desired object to be selected,
then processes that into a regular non-deep image or mask for editing.

Example code
============

OpenEXR provides two example tools, ``deepidexample`` and ``deepidselect``.
Compiled tools will be found in the ``src/examples`` folder in the build directory. They are not installed.


DeepIDExample
-------------

``deepidexample`` creates a deep image with multiple objects (two different shapes at one of three sizes),
in one of seven colors. It is intended as a tool for generating test sequences and as an example of code
that generates an image with deep IDs and a manifest.

``deepidexample`` can generate a sequence of frames, to help test that the IDs are consistently
generated and selected. Specify ``--frame`` for the frame number. The animation cycles every 100 frames.
This ``bash`` command generates a sequence of frames:

.. code:: bash

     for x in `seq 1000 1100` ; do ./deepidexample --frame $x output.$x.deep.exr ; done

Run ``deepidexample`` to see further options.


DeepIDSelect
------------

``deepidselect`` selects individual objects within a deep file, and outputs just those objects.
It is intended to serve as an example of parsing idmanifests to find compile a list of IDs which
match a given substring, and using those ids to identify samples. Its usage is not limited solely
to files created by deepidexample; it should handle files with arbitrary channel names and manifests.
deepidselect supports the ``id64`` scheme with the ``--64`` flag.

In basic usage, specify ``input.deep.exr (matches) output.deep.exr``

``matches`` is a list of one or more separate strings. All objects whose names contain any of the
given substring will be included in output.deep.exr (it is a logical OR of the arguments)
The ``--and`` can be used to force matching of (one or more of) the following match as well as the previous.
For example, ``blue --and circle`` will match any object which is both blue, and a circle.
``blue green --and big small --and circle`` will match blue or green objects,
and which are big or small, and which are circles.
This could also be read as `( blue or green ) and ( big or small ) and ( circle )`

Each match can be limited to a given component name by specifying ``component:match``.
For example ``model:bl`` will match objects whose model is ``blob`` but not ones whose material is ``blue``.
Specifying a channel name followed by a number will select the object by number, rather than by name.
For example, ``particleid:12`` will select the object with particle ID 12.
(Note that this feature means it is not possible to have a purely numeric substring match with this tool)

``--mask`` outputs a shallow single channel image which indicating the relative coverage of each pixel
for the selected object. For schemes where the deep image only contains ID (and alpha) information,
but does not store color, this can be used to grade only the selected object.
Edge contamination may be observed along transparent edges of a selected object, if an object behind it is not selected.

To keep the code simple, ``deepidselect`` is only a minimal example of string matching against ID manifests.
For example, it doesn't support regular expressions, or more advanced Boolean logic including negative matches.


Appendix
========


64 to 2 x 32 bits conversion and back
-------------------------------------

To limit the risk of hash collision, a ``uint64_t`` can be encoded in 2
``uint32_t`` channels, like ``materialid`` and ``materialid2``, using
little-endian byte ordering.

.. code:: cpp

   #include <cstdint>
   #include <iostream>
   #include <iomanip>

   int main()
   {
       using namespace std;

       // uint 64 input
       uint64_t x = 0x12345678'87654321ULL;
       cout << setw(20) << "uint64 input: " << hex << x << endl;

       // Convert one uint 64 -> two uint 32
       uint32_t lo = uint32_t(x);
       uint32_t hi = uint32_t(x >> 32);
       cout << setw(20) << "uint32 low: " << hex << lo << "  high: " << hi << endl;

       // Convert two uint32 -> one uint64
       uint64_t y = (uint64_t(hi) << 32) | lo;
       cout << setw(20) << "uint64 recombined: " << hex << y << endl;
   }

Output:

::

         uint64 input: 1234567887654321
           uint32 low: 87654321  high: 12345678
    uint64 recombined: 1234567887654321

Computing a shallow mask from Deep IDs
--------------------------------------

A shallow mask is a pixel-level mask that can be used with non-deep
compositing operators.

Here is the pseudo-code to correctly compute an ID selection mask for a
single pixel:

.. code:: python

   total_combined_alpha = 0.0
   mask_alpha = 0.0
   sorted_pixel = sort_pixel_front_to_back(input_pixel)

   foreach(sample in sorted_pixel):
       if id_is_in_selection(sample.id):
           mask_alpha += sample.alpha * (1.0 - total_combined_alpha)
       total_combined_alpha += sample.alpha * (1.0 - total_combined_alpha)

   if total_combined_alpha == 0.0:
       return 0.0
   else:
       return mask_alpha / total_combined_alpha

.. [1]
   See `Cryptomatte <https://github.com/Psyop/Cryptomatte>`__ on github

.. [2]
   See `OpenEXR reserved channel
   names <https://openexr.com/en/latest/TechnicalIntroduction.html#deep-data-special-purpose-channels-and-reserved-channel-names>`__.

.. [3]
   See
   `ImfIDManifest <https://github.com/AcademySoftwareFoundation/openexr/blob/main/src/lib/OpenEXR/ImfIDManifest.h>`__

.. [4]
   For more details of the compression scheme used by idmanifest, see `A scheme for storing object ID manifests in openEXR images` In Proceedings of the 8th Annual Digital Production Symposium (DigiPro '18). Association for Computing Machinery, New York, NY, USA, Article 9, 1–8. https://doi.org/10.1145/3233085.3233086