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
|
/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#include "WebGLQuery.h"
#include "GLContext.h"
#include "mozilla/dom/WebGL2RenderingContextBinding.h"
#include "mozilla/StaticPrefs_webgl.h"
#include "nsContentUtils.h"
#include "WebGLContext.h"
namespace mozilla {
////
static GLuint GenQuery(gl::GLContext* gl) {
GLuint ret = 0;
gl->fGenQueries(1, &ret);
return ret;
}
WebGLQuery::WebGLQuery(WebGLContext* webgl)
: WebGLContextBoundObject(webgl),
mGLName(GenQuery(mContext->gl)),
mTarget(0),
mActiveSlot(nullptr) {}
WebGLQuery::~WebGLQuery() {
if (!mContext) return;
mContext->gl->fDeleteQueries(1, &mGLName);
}
////
static GLenum TargetForDriver(const gl::GLContext* gl, GLenum target) {
switch (target) {
case LOCAL_GL_ANY_SAMPLES_PASSED:
case LOCAL_GL_ANY_SAMPLES_PASSED_CONSERVATIVE:
break;
default:
return target;
}
if (gl->IsSupported(gl::GLFeature::occlusion_query_boolean)) return target;
if (gl->IsSupported(gl::GLFeature::occlusion_query2))
return LOCAL_GL_ANY_SAMPLES_PASSED;
return LOCAL_GL_SAMPLES_PASSED;
}
void WebGLQuery::BeginQuery(GLenum target, RefPtr<WebGLQuery>& slot) {
mTarget = target;
mActiveSlot = &slot;
*mActiveSlot = this;
////
const auto& gl = mContext->gl;
const auto driverTarget = TargetForDriver(gl, mTarget);
gl->fBeginQuery(driverTarget, mGLName);
}
void WebGLQuery::EndQuery() {
*mActiveSlot = nullptr;
mActiveSlot = nullptr;
mCanBeAvailable = false;
////
const auto& gl = mContext->gl;
const auto driverTarget = TargetForDriver(gl, mTarget);
gl->fEndQuery(driverTarget);
}
Maybe<double> WebGLQuery::GetQueryParameter(GLenum pname) const {
switch (pname) {
case LOCAL_GL_QUERY_RESULT_AVAILABLE:
case LOCAL_GL_QUERY_RESULT:
break;
default:
mContext->ErrorInvalidEnumInfo("pname", pname);
return Nothing();
}
if (!mTarget) {
mContext->ErrorInvalidOperation("Query has never been active.");
return Nothing();
}
if (mActiveSlot) {
mContext->ErrorInvalidOperation("Query is still active.");
return Nothing();
}
// End of validation
////
const auto& gl = mContext->gl;
uint64_t val = 0;
switch (pname) {
case LOCAL_GL_QUERY_RESULT_AVAILABLE:
gl->fGetQueryObjectuiv(mGLName, pname, (GLuint*)&val);
return Some(static_cast<bool>(val));
case LOCAL_GL_QUERY_RESULT:
switch (mTarget) {
case LOCAL_GL_TIME_ELAPSED_EXT:
case LOCAL_GL_TIMESTAMP_EXT:
if (mContext->Has64BitTimestamps()) {
gl->fGetQueryObjectui64v(mGLName, pname, &val);
break;
}
[[fallthrough]];
default:
gl->fGetQueryObjectuiv(mGLName, LOCAL_GL_QUERY_RESULT, (GLuint*)&val);
break;
}
switch (mTarget) {
case LOCAL_GL_ANY_SAMPLES_PASSED:
case LOCAL_GL_ANY_SAMPLES_PASSED_CONSERVATIVE:
case LOCAL_GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN:
case LOCAL_GL_TIME_ELAPSED_EXT:
case LOCAL_GL_TIMESTAMP_EXT:
return Some(val);
}
MOZ_CRASH("Bad `mTarget`.");
default:
MOZ_CRASH("Bad `pname`.");
}
}
void WebGLQuery::QueryCounter() {
const GLenum target = LOCAL_GL_TIMESTAMP_EXT;
if (mTarget && target != mTarget) {
mContext->ErrorInvalidOperation("Queries cannot change targets.");
return;
}
mTarget = target;
mCanBeAvailable = false;
const auto& gl = mContext->gl;
gl->fQueryCounter(mGLName, mTarget);
}
} // namespace mozilla
|