EXT_occlusion_query_boolean
Name
EXT_occlusion_query_boolean
Name Strings
GL_EXT_occlusion_query_boolean
Contributors
All those who have contributed to the definition of occlusion
query functionality in the OpenGL ARB and OpenGL ES workgroups,
upon which this extension spec is entirely dependent.
Contact
Benj Lipchak, Apple (lipchak 'at' apple.com)
Status
Complete
Version
Date: July 22, 2011
Revision: 2
Number
OpenGL ES Extension #100
Dependencies
Written based on the wording of the OpenGL ES 2.0.25 Full Specification
(November 2, 2010).
Overview
This extension defines a mechanism whereby an application can
query whether any pixels (or, more precisely, samples) are drawn
by a primitive or group of primitives.
The primary purpose of such a query (hereafter referred to as an
"occlusion query") is to determine the visibility of an object.
Typically, the application will render the major occluders in the
scene, then perform an occlusion query for each detail object in
the scene. On subsequent frames, the previous results of the
occlusion queries can be used to decide whether to draw an object
or not.
New Procedures and Functions
void GenQueriesEXT(sizei n, uint *ids);
void DeleteQueriesEXT(sizei n, const uint *ids);
boolean IsQueryEXT(uint id);
void BeginQueryEXT(enum target, uint id);
void EndQueryEXT(enum target);
void GetQueryivEXT(enum target, enum pname, int *params);
void GetQueryObjectuivEXT(uint id, enum pname, uint *params);
New Tokens
Accepted by the <target> parameter of BeginQueryEXT, EndQueryEXT,
and GetQueryivEXT:
ANY_SAMPLES_PASSED_EXT 0x8C2F
ANY_SAMPLES_PASSED_CONSERVATIVE_EXT 0x8D6A
Accepted by the <pname> parameter of GetQueryivEXT:
CURRENT_QUERY_EXT 0x8865
Accepted by the <pname> parameter of GetQueryObjectivEXT and
GetQueryObjectuivEXT:
QUERY_RESULT_EXT 0x8866
QUERY_RESULT_AVAILABLE_EXT 0x8867
Additions to Chapter 2 of the OpenGL ES 2.0 Specification (OpenGL ES Operation)
Add a new section "Asynchronous Queries" between sections 2.12 and 2.13
and renumber subsequent sections:
"2.13 Asynchronous Queries
Asynchronous queries provide a mechanism to return information about the
processing of a sequence of GL commands. There is one query type
supported by the GL. Occlusion queries (see section 4.1.6) set a boolean
to true when any fragments or samples pass the depth test.
The results of asynchronous queries are not returned by the GL immediately
after the completion of the last command in the set; subsequent commands
can be processed while the query results are not complete. When available,
the query results are stored in an associated query object. The commands
described in section 6.1.6 provide mechanisms to determine when query
results are available and return the actual results of the query. The
name space for query objects is the unsigned integers, with zero reserved
by the GL.
Each type of query supported by the GL has an active query object name. If
the active query object name for a query type is non-zero, the GL is
currently tracking the information corresponding to that query type and the
query results will be written into the corresponding query object. If the
active query object for a query type name is zero, no such information is
being tracked.
A query object is created and made active by calling
void BeginQueryEXT(enum target, uint id);
<target> indicates the type of query to be performed; valid values of
<target> are defined in subsequent sections. If the query object name <id>
has not been created, the name is marked as used and associated with a new
query object of the type specified by <target>. Otherwise <id> must be the
name of an existing query object of that type.
BeginQueryEXT fails and an INVALID_OPERATION error is generated if <id>
is not a name returned from a previous call to GenQueriesEXT, or if such
a name has since been deleted with DeleteQueriesEXT.
BeginQueryEXT sets the active query object name for the query type given
by <target> to <id>. If BeginQueryEXT is called with an <id> of zero, if
the active query object name for <target> is non-zero (for the targets
ANY_SAMPLES_PASSED_EXT and ANY_SAMPLES_PASSED_CONSERVATIVE_EXT, if the
active query for either target is non-zero), if <id> is the name of an
existing query object whose type does not match <target>, or if <id> is the
active query object name for any query type, the error INVALID_OPERATION is
generated.
The command
void EndQueryEXT(enum target);
marks the end of the sequence of commands to be tracked for the query type
given by <target>. The active query object for <target> is updated to
indicate that query results are not available, and the active query object
name for <target> is reset to zero. When the commands issued prior to
EndQueryEXT have completed and a final query result is available, the
query object active when EndQueryEXT is called is updated by the GL. The
query object is updated to indicate that the query results are available
and to contain the query result. If the active query object name for
<target> is zero when EndQueryEXT is called, the error INVALID_OPERATION
is generated.
The command
void GenQueriesEXT(sizei n, uint *ids);
returns <n> previously unused query object names in <ids>. These names are
marked as used, for the purposes of GenQueriesEXT only, but no object is
associated with them until the first time they are used by BeginQueryEXT.
Query objects are deleted by calling
void DeleteQueriesEXT(sizei n, const uint *ids);
<ids> contains <n> names of query objects to be deleted. After a query
object is deleted, its name is again unused. Unused names in <ids> are
silently ignored, as is the value zero. If an active query object is
deleted its name immediately becomes unused, but the underlying object is
not deleted until it is no longer active (see section C.1).
Query objects contain two pieces of state: a single bit indicating whether
a query result is available, and an integer containing the query result
value. The number of bits used to represent the query result is
implementation-dependent and may vary by query object type. In the initial
state of a query object, the result is available and its value is zero.
The necessary state for each query type is an unsigned integer holding the
active query object name (zero if no query object is active), and any state
necessary to keep the current results of an asynchronous query in progress.
Only a single type of occlusion query can be active at one time, so the
required state for occlusion queries is shared."
Additions to Chapter 3 of the OpenGL ES 2.0 Specification (Rasterization)
None
Additions to Chapter 4 of the OpenGL ES 2.0 Specification (Per-Fragment Operations and the Frame Buffer)
Add a new section "Occlusion Queries" between sections 4.1.5 and
4.1.6 and renumber subsequent sections:
"4.1.6 Occlusion Queries
Occlusion queries use query objects to track the number of fragments or
samples that pass the depth test. An occlusion query can be started and
finished by calling BeginQueryEXT and EndQueryEXT, respectively, with a
target of ANY_SAMPLES_PASSED_EXT or ANY_SAMPLES_PASSED_CONSERVATIVE_EXT.
When an occlusion query is started with the target
ANY_SAMPLES_PASSED_EXT, the samples-boolean state maintained by the GL is
set to FALSE. While that occlusion query is active, the samples-boolean
state is set to TRUE if any fragment or sample passes the depth test. When
the occlusion query finishes, the samples-boolean state of FALSE or TRUE is
written to the corresponding query object as the query result value, and
the query result for that object is marked as available. If the target of
the query is ANY_SAMPLES_PASSED_CONSERVATIVE_EXT, an implementation may
choose to use a less precise version of the test which can additionally set
the samples-boolean state to TRUE in some other implementation dependent
cases."
Additions to Chapter 5 of the OpenGL ES 2.0 Specification (Special Functions)
None
Additions to Chapter 6 of the OpenGL ES 2.0 Specification (State and State Requests)
Add a new section "Asynchronous Queries" between sections 6.1.5 and
6.1.6 and renumber subsequent sections:
"6.1.6 Asynchronous Queries
The command
boolean IsQueryEXT(uint id);
returns TRUE if <id> is the name of a query object. If <id> is zero,
or if <id> is a non-zero value that is not the name of a query
object, IsQueryEXT returns FALSE.
Information about a query target can be queried with the command
void GetQueryivEXT(enum target, enum pname, int *params);
<target> identifies the query target, and must be one of
ANY_SAMPLES_PASSED_EXT or ANY_SAMPLES_PASSED_CONSERVATIVE_EXT.
<pname> must be CURRENT_QUERY_EXT. The name of the currently active
query for <target>, or zero if no query is active, will be placed in
<params>.
The state of a query object can be queried with the command
void GetQueryObjectuivEXT(uint id, enum pname, uint *params);
If <id> is not the name of a query object, or if the query object
named by <id> is currently active, then an INVALID_OPERATION error is
generated. <pname> must be QUERY_RESULT_EXT or QUERY_RESULT_AVAILABLE_EXT.
If <pname> is QUERY_RESULT_EXT, then the query object's result value
is returned as a single integer in <params>. If the value is so large in
magnitude that it cannot be represented with the requested type, then the
nearest value representable using the requested type is returned.
There may be an indeterminate delay before the above query returns. If
<pname> is QUERY_RESULT_AVAILABLE_EXT, FALSE is returned if such a delay
would be required; otherwise TRUE is returned. It must always be true that
if any query object returns a result available of TRUE, all queries of the
same type issued prior to that query must also return TRUE.
Querying the state for any given query object forces that occlusion query
to complete within a finite amount of time. Repeatedly querying the
QUERY_RESULT_AVAILABLE_EXT state for any given query object is guaranteed
to return true eventually. Note that multiple queries to the same occlusion
object may result in a significant performance loss. For better performance
it is recommended to wait N frames before querying this state. N is
implementation dependent but is generally between one and three.
If multiple queries are issued using the same object name prior to calling
GetQueryObjectuivEXT, the result and availability information returned
will always be from the last query issued. The results from any queries
before the last one will be lost if they are not retrieved before starting
a new query on the same <target> and <id>."
Additions to Appendix C of the OpenGL ES 2.0 Specification (Shared Object and Multiple Contexts)
Change the first sentence of the first paragraph of section C.1.2 to read:
"When a buffer, texture, renderbuffer, or query is deleted, ..."
Add the following sentence to the end of section C.1.2:
"A query object is in use so long as it is the active query object for a
query type and index, as described in section 2.13."
Errors
The error INVALID_OPERATION is generated if BeginQueryEXT is called
where <id> is not a name returned from a previous call to GenQueriesEXT,
or if such a name has since been deleted with DeleteQueriesEXT.
The error INVALID_OPERATION is generated if BeginQueryEXT is called
where <id> is zero.
The error INVALID_OPERATION is generated if BeginQueryEXT is called
where <id> is the name of an existing query object whose type does not
match <target>.
The error INVALID_OPERATION is generated if BeginQueryEXT is called
where <id> is the active query object name for any query type.
The error INVALID_OPERATION is generated if BeginQueryEXT is called
when the active query object name for either ANY_SAMPLES_PASSED_EXT or
ANY_SAMPLES_PASSED_CONSERVATIVE_EXT is non-zero.
The error INVALID_OPERATION is generated if EndQueryEXT is called
when the active query object name for <target> is zero.
The error INVALID_OPERATION is generated if GetQueryObjectuivEXT is
called where <id> is not the name of a query object.
The error INVALID_OPERATION is generated if GetQueryObjectuivEXT is
called where <id> is the name of a currently active query object.
The error INVALID_VALUE is generated if GenQueriesEXT is called where
<n> is negative.
The error INVALID_VALUE is generated if DeleteQueriesEXT is called
where <n> is negative.
The error INVALID_ENUM is generated if BeginQueryEXT, EndQueryEXT,
or GetQueryivEXT is called where <target> is not
ANY_SAMPLES_PASSED_EXT or ANY_SAMPLES_PASSED_CONSERVATIVE_EXT.
The error INVALID_ENUM is generated if GetQueryivEXT is called where
<pname> is not CURRENT_QUERY_EXT.
The error INVALID_ENUM is generated if GetQueryObjectuivEXT is called
where <pname> is not QUERY_RESULT_EXT or QUERY_RESULT_AVAILABLE_EXT.
New State
(table 6.18, p. 233)
Int'l
Get Value Type Get Command Value Description Sec
-------------------------- ---- -------------------- ----- ---------------------- -----
- B - FALSE query active 4.1.6
CURRENT_QUERY_EXT Z+ GetQueryivEXT 0 active query ID 4.1.6
QUERY_RESULT_EXT B GetQueryObjectuivEXT FALSE samples-passed 4.1.6
QUERY_RESULT_AVAILABLE_EXT B GetQueryObjectuivEXT FALSE query result available 4.1.6
Issues
(1) What should the enum be called?
RESOLVED: The enum should be called ANY_SAMPLES_PASSED as in
ARB_occlusion_query2 to retain compatibility between the two
extensions.
(2) Can application-provided names be used as query object names?
ARB_occlusion_query allows application-provided names, but this
was later removed in core OpenGL.
RESOLVED: No, we will follow core OpenGL on this.
(3) Should calling GenQueries or DeleteQueries when a query is
active produce an error?
This behavior is in ARB_occlusion_query but was
removed in OpenGL 3.0.
RESOLVED: Not an error. Calling DeleteQueries marks the name
as no longer used, but the object is not deleted until it is no
longer in use (i.e. no longer active).
(4) What is the interaction with multisample?
RESOLVED: The query result is set to true if at least one
sample passes the depth test.
(5) Exactly what stage in the pipeline are we counting samples at?
RESOLVED: We are counting immediately after _both_ the depth and
stencil tests, i.e., samples that pass both. Note that the depth
test comes after the stencil test, so to say that it is the
number that pass the depth test is sufficient; though it is often
conceptually helpful to think of the depth and stencil tests as
being combined, because the depth test's result impacts the
stencil operation used.
(6) Is it guaranteed that occlusion queries return in order?
RESOLVED: Yes.
If occlusion test X occurred before occlusion query Y, and the driver
informs the app that occlusion query Y is done, the app can infer that
occlusion query X is also done.
(7) Will polling a query for QUERY_RESULT_AVAILABLE without a Flush
possibly cause an infinite loop?
RESOLVED: No.
(8) Should there be a "target" parameter to BeginQuery?
RESOLVED: Yes. This distinguishes the boolean queries
defined by this extension (and ARB_occlusion_query2) from
the counter queries defined by ARB_occlusion_query.
(9) Are query objects shareable between multiple contexts?
RESOLVED: No. Query objects are lightweight and we normally share
large data across contexts. Also, being able to share query objects
across contexts is not particularly useful. In order to do the async
query across contexts, a query on one context would have to be finished
before the other context could query it.
(10) Should there be a limit on how many queries can be outstanding?
RESOLVED: No. If an implementation has an internal limit, it can
flush the pipeline when it runs out.
(11) Can an implementation sometimes return a conservative result,
i.e. return true even though no samples were drawn?
RESOLVED: Yes, but only when explicitly enabled by the
application.
Allowing such results with no restrictions effectively makes
the functionality of the extension optional, which decreases
its value. Precise restrictions are presumably hard to
specify.
One situation where this restriction could be relevant is if
an implementation performs a conservative early depth test at
a lower precision and wants to base the occlusion query result
on that whenever the early depth test can be used.
(12) Should the restrictions in issue 11 be explicitly enabled
by the application in order to be in effect?
RESOLVED: Yes.
The restrictions could be enabled by a hint call or by using
a different enum in the BeginQuery call.
This would enable the application to choose whether it wants a
precise (but possibly slow) version or an approximate (but
possibly faster) version.
(13) Can the restrictions in issue 18 be applied nondeterministically?
An implementation might benefit from taking the decision of
whether to apply a particular restriction on a case by case
basis. Some of these decisions could depend on
nondeterministic effects such as memory bus timing.
RESOLVED: No. This would violate the GL repeatability
principle.
(14) How does an application request that the result is allowed to
be conservative (as defined in issue 11)?
RESOLVED: It is specified as a separate query target,
ANY_SAMPLES_PASSED_CONSERVATIVE.
Revision History
Date: 5/03/2011 Revision: 1 (Benj Lipchak) - Initial draft based on XXX_occlusion_query_boolean
Date: 7/22/2011 Revision: 2 (Benj Lipchak) - Rename from APPLE to EXT