OES_sample_variables
Name
OES_sample_variables
Name Strings
GL_OES_sample_variables
Contact
Daniel Koch, NVIDIA Corporation (dkoch 'at' nvidia.com)
Contributors
Pat Brown, NVIDIA
Eric Werness, NVIDIA
Graeme Leese, Broadcom
Contributors to ARB_gpu_shader5
Contributors to ARB_sample_shading
Members of the OpenGL ES Working Group
Notice
Copyright (c) 2011-2019 The Khronos Group Inc. Copyright terms at
http://www.khronos.org/registry/speccopyright.html
Specification Update Policy
Khronos-approved extension specifications are updated in response to
issues and bugs prioritized by the Khronos OpenGL ES Working Group. For
extensions which have been promoted to a core Specification, fixes will
first appear in the latest version of that core Specification, and will
eventually be backported to the extension document. This policy is
described in more detail at
https://www.khronos.org/registry/OpenGL/docs/update_policy.php
Status
Complete.
Ratified by the Khronos Board of Promoters on 2014/03/14.
Version
Last Modified Date: January 10, 2019
Revision: 10
Number
OpenGL ES Extension #170
Dependencies
OpenGL ES 3.0 and GLSL ES 3.00 required.
This extension is written against the OpenGL ES 3.0.2 (April 8, 2013)
and the OpenGL ES Shading Language Specification Revision 4
(March 6, 2013) specifications.
This extension interacts with OES_sample_shading.
This extension interacts with OES_shader_multisample_interpolation.
This extension interacts with OpenGL ES 3.1.
Overview
This extension allows fragment shaders more control over multisample
rendering. The mask of samples covered by a fragment can be read by
the shader and individual samples can be masked out. Additionally
fragment shaders can be run on individual samples and the sample's
ID and position read to allow better interaction with multisample
resources such as textures.
In multisample rendering, an implementation is allowed to assign the
same sets of fragment shader input values to each sample, which then
allows the optimization where the shader is only evaluated once and
then distributed to the samples that have been determined to be
covered by the primitive currently being rasterized. This extension
does not change how values are interpolated, but it makes some details
of the current sample available. This means that where these features
are used (gl_SampleID and gl_SamplePosition), implementations must
run the fragment shader for each sample.
In order to obtain per-sample interpolation on fragment inputs, either
OES_sample_shading or OES_shader_multisample_interpolation must
be used in conjunction with the features from this extension.
New Procedures and Functions
None.
New Tokens
None.
Additions to Chapter 2 of the OpenGL ES 3.0 Specification (OpenGL ES Operation)
None.
Additions to Chapter 3 of the OpenGL ES 3.0 Specification (Rasterization)
Modify section 3.9.2, Shader Execution, p. 162
(Add the following paragraphs to the section Shader Inputs, p. 164, after
the paragraph about gl_FrontFacing)
The built-in read-only variable gl_SampleID is filled with the
sample number of the sample currently being processed. This variable
is in the range zero to gl_NumSamples minus one, where
gl_NumSamples is the
total number of samples in the framebuffer, or one if rendering to a
non-multisample framebuffer. Using gl_SampleID in a fragment shader
causes the entire shader to be executed per-sample. When rendering to a
non-multisample buffer,
gl_SampleID will always be zero. gl_NumSamples is the sample count
of the framebuffer regardless of whether the framebuffer is multisampled
or not.
The built-in read-only variable gl_SamplePosition contains the
position of the current sample within the multi-sample draw buffer.
The x and y components of gl_SamplePosition contain the sub-pixel
coordinate of the current sample and will have values in the range
[0, 1]. The sub-pixel coordinate of the center of the pixel is
always (0.5, 0.5). Using this variable in a fragment shader
causes the entire shader to be executed per-sample. When rendering to a
non-multisample buffer,
gl_SamplePosition will always be (0.5, 0.5).
The built-in variable gl_SampleMaskIn is an integer array holding
bitfields indicating the set of fragment samples covered by the primitive
corresponding to the fragment shader invocation. The number of elements
in the array is ceil(gl_MaxSamples/32), where gl_MaxSamples is the
the value of MAX_SAMPLES, the maximum number of color samples supported
by the implementation. Bit <n> of element <w> in the
array is set if and only if the sample numbered <w>*32+<n> is considered
covered for this fragment shader invocation. When rendering to a
non-multisample buffer, all
bits are zero except for bit zero of the first array element. That bit
will be one if the pixel is covered and zero otherwise. Bits in the
sample mask corresponding to covered samples that will be killed due to
SAMPLE_COVERAGE or SAMPLE_MASK will not be set (section 4.1.3).
When per-sample shading is active due to the use of a fragment input
qualified by "sample" or due to the use of the gl_SampleID or
gl_SamplePosition variables, only the bit for the current sample is
set in gl_SampleMaskIn.
When OpenGL ES API state specifies multiple fragment shader invocations
for a given fragment, the bit corresponding to each covered sample will
be set in exactly one fragment shader invocation.
Modify section Shader Outputs, p. 165
(Replace the second sentence of the first paragraph with the following)
These outputs are split into two categories, user-defined outputs and the
built-in outputs gl_FragColor, gl_FragData[n] (both available only in
OpenGL ES Shading Language version 1.00), gl_FragDepth and gl_SampleMask.
(Insert the following paragraph after the first paragraph of the section)
The built-in integer array gl_SampleMask can be used to change the
sample coverage for a fragment from within the shader. The number
of elements in the array is ceil(gl_MaxSamples/32), where
gl_MaxSamples is the value of MAX_SAMPLES, the maximum number of
color samples supported by the implementation.
If bit <n> of element <w> in the array is set to zero, sample
<w>*32+<n> should be considered uncovered for the purposes of
multisample fragment operations (Section 4.1.3). Modifying the
sample mask in this way may exclude covered samples from being
processed further at a per-fragment granularity. However, setting
sample mask bits to one will never enable samples not covered by the
original primitive. If the fragment shader is being executed at
any frequency other than per-fragment, bits of the sample mask not
corresponding to the current fragment shader invocation are ignored.
Additions to Chapter 4 of the OpenGL ES 3.0.2 Specification (Per-Fragment Operations and the Framebuffer)
Modify Section 4.1.3, Multisample Fragment Operations, p. 170
(modify first paragraph of section) This step modifies fragment alpha and
coverage values based on the values of SAMPLE_ALPHA_TO_COVERAGE,
SAMPLE_COVERAGE, SAMPLE_COVERAGE_VALUE,
SAMPLE_COVERAGE_INVERT, and an output sample mask optionally written by
the fragment shader. No changes to the fragment alpha or coverage values
are made at this step if the value of
SAMPLE_BUFFERS is not one.
(insert new paragraph before the paragraph on SAMPLE_COVERAGE, p. 171)
Next, if a fragment shader is active and statically assigns to the
built-in output variable gl_SampleMask, the fragment coverage is ANDed
with the bits of the sample mask. The initial values for elements of
gl_SampleMask are undefined. Bits in each array element that are not
written due to flow control or partial writes (i.e., bit-wise operations)
will continue to have undefined values. The value of those bits ANDed with
the fragment coverage is undefined. If no fragment shader is active, or
if the active fragment shader does not statically assign values to
gl_SampleMask, the fragment coverage is not modified.
Additions to Chapter 5 of the OpenGL ES 3.0.2 Specification (Special Functions)
None.
Additions to Chapter 6 of the OpenGL ES 3.0.2 Specification (State and State Requests)
None.
Modifications to The OpenGL ES Shading Language Specification, Version 3.00.04
Including the following line in a shader can be used to control the
language features described in this extension:
#extension GL_OES_sample_variables
A new preprocessor #define is added to the OpenGL ES Shading Language:
#define GL_OES_sample_variables 1
Add to section 7.2 "Fragment Shader Special Variables"
(add the following to the list of built-in variables that are accessible
from a fragment shader)
in lowp int gl_SampleID;
in mediump vec2 gl_SamplePosition;
in highp int gl_SampleMaskIn[(gl_MaxSamples+31)/32];
out highp int gl_SampleMask[(gl_MaxSamples+31)/32];
(add the following descriptions of the new variables)
The input variable gl_SampleID is filled with the
sample number of the sample currently being processed. This
variable is in the range 0 to gl_NumSamples-1, where gl_NumSamples is
the total number of samples in the framebuffer, or one if rendering to
a non-multisample framebuffer. Any static use of gl_SampleID in a
fragment shader causes the entire shader to be executed per-sample.
The input variable gl_SamplePosition contains the
position of the current sample within the multi-sample draw
buffer. The x and y components of gl_SamplePosition contain the
sub-pixel coordinate of the current sample and will have values in
the range 0.0 to 1.0. Any static use of this variable in a fragment
shader causes the entire shader to be executed per-sample.
For the both the input array gl_SampleMaskIn[] and the output
array gl_SampleMask[], bit B of mask M (gl_SampleMaskIn[M]
or gl_SampleMask[M]) corresponds to sample 32*M+B. These arrays
have ceil(gl_MaxSamples/32) elements, where gl_MaxSamples is
the maximum number of color samples supported by the implementation.
The input variable gl_SampleMaskIn indicates the set of samples covered
by the primitive generating the fragment during multisample rasterization.
It has a sample bit set if and only if the sample is considered covered for
this fragment shader invocation.
The output array gl_SampleMask[] sets the sample mask for
the fragment being processed. Coverage for the current fragment will
be the logical AND of the coverage mask and the output
gl_SampleMask. If the fragment shader
statically assigns a value to gl_SampleMask, the sample mask will
be undefined for any array elements of any fragment shader
invocations that fails to assign a value. If a shader does not
statically assign a value to gl_SampleMask, the sample mask has no
effect on the processing of a fragment.
Add to section 7.3 Built-in Constants
const mediump int gl_MaxSamples = 4;
Add to Section 7.4 Built-in Uniform State
(Add the following prototype to the list of built-in uniforms
accessible from a fragment shader:)
uniform lowp int gl_NumSamples;
Additions to the AGL/GLX/WGL/EGL Specifications
None
Dependencies on OES_sample_shading
If OES_sample_shading is not supported ignore any mention of API state
that forces multiple shader invocations per fragment.
Dependencies on OES_shader_multisample_interpolation
If OES_shader_multisample_interpolation is not supported ignore any mention of the
"sample" qualifier keyword for fragment inputs.
Dependencies on OpenGL ES 3.1
If OpenGL ES 3.1 is not supported, ignore references to SAMPLE_MASK.
Errors
None.
New State
None.
New Implementation Dependent State
None.
Issues
(0) This extension is based on ARB_sample_shading. What are the major
differences?
1- rebased against ES 3.0
2- various editing for consistency to GL 4.4/GLSL 440 specs
3- added precision qualifiers for GLSL builtins
4- removed mention of SAMPLE_ALPHA_TO_ONE
5- replaced mention of "color and texture coordinates" with more
generic language about fragment shader inputs.
6- removed mention of multisample enable.
7- added gl_SampleMaskIn from ARB_gpu_shader5
8- replace the term 'evaluated' with 'executed' (Issue 3)
9- removed mention of sizing gl_SampleMask[] (Issue 4)
10- added gl_MaxSamples shading language constant.
For historical issues, please see ARB_sample_shading and
ARB_gpu_shader5.
(1) OpenGL has a MULTISAMPLE enable that was not included in OpenGL ES.
Should we add it into this extension or base it purely on if the target
surface is multisample?
DISCUSSION:
GL (4.4) says:
"Multisample rasterization is enabled or disabled by calling Enable or
Disable with the symbolic constant MULTISAMPLE."
GL ES (3.0.2) says:
"Multisample rasterization cannot be enabled or disabled after a GL
context is created."
RESOLVED. Multisample rasterization should be based on the target
surface properties. Will not pick up the explicit multisample
enable, but the language for ES3.0.2 doesn't sound right either.
Bug 10690 tracks this and it should be fixed in later versions
of the ES3.0 specification.
(2) ES requires vec2s in a fragment shader to be declared with a precision
qualifiers, what precision should be used for gl_SamplePosition?
RESOLVED: mediump should be used since lowp might be implemented with
fixed point and be unable to exactly represent [0.5, 0.5].
(3) Is it reasonable to run shaders per-sample when interpolation is still
per-fragment?
RESOLVED: Yes. This allows a useful way of interacting with
multi-sample resources so it is included. To avoid confusion between
between per-sample interpolation and per-sample execution, we'll
use the term "executed" instead of "evaluated".
(4) ARB_sample_shaders says that "gl_SampleMask[] must be sized either
implicitly or explicitly in the fragment shader to be the same size
described above." ES doesn't have implicitly sized arrays.
Does this need to be explicitly declared in a shader or should it be
predeclared by the implementation? If predeclared, should it be an
error to redeclare it in the shader?
RESOLVED: In practice, one couldn't detect a difference between an
implicitly sized array and one that is automatically sized correctly
by a builtin declaration. In ES it is considered to be declared
(correctly sized) by the implementation when necessary and thus no
specific statement is required. As with all built-ins it is an
error for a shader to redeclare it.
(5) How does one know the size of the gl_SampleMaskIn/gl_SampleMask
arrays?
RESOLVED: The GL spec states that the size of the arrays is
ceil(<s>/32) where <s> is the maximum number of color samples
in the implementation. <s> is thus the equivalent of MAX_SAMPLES
which is the upper bound on the number of supported sample
of any format. As a convenience we add the built-in shading
language constant gl_MaxSamples to mirror this API
constant in the shading language and the size of the arrays is
defined in terms of this constant.
(6) Should the shading language built-ins have OES suffixes?
RESOLVED: No. Per Bug 11637, the WG made a policy decision
that GLSL ES identifiers imported without semantic change
or subsetting as OES extensions from core GLSL do not carry
suffixes. The #extension mechanism must still be used to
enable the appropriate extension before the functionality can
be used.
Revision History
Rev. Date Author Changes
---- ---------- -------- -----------------------------------------
10 2019-01-10 Jon Leech Clarify the requirements on gl_SampleMaskIn
(internal API issue #45).
9 2014-02-12 dkoch remove GLSL suffixes per Issue 6.
8 2014-01-30 dkoch rename to OES, clean editing notes
7 2013-12-11 dkoch correct names of interacting extensions
6 2013-10-24 dkoch add gl_MaxSampleOES builtin constant and Issue 5
5 2013-10-22 dkoch Clarifications from Ian Romanick
4 2013-10-03 dkoch Added dependency on texture_storage_multisample
3 2013-10-03 dkoch Resolved all issues.
Changed gl_SamplePosition to mediump.
Changed the term "evaluated" to "executed".
Removed language about sizing gl_SampleMask.
2 2013-09-08 dkoch Added interactions for SampleMaskIn, deps.
Misc small editorial updates.
Added issue 4, unresolved issue 3.
1 2013-09-03 gleese Extracted from OES_sample_shading and
OES_shader_multisample_interpolation