blob: 7e596a8539289958a6375c426e68956fe7ed6c13 [file] [log] [blame]
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001//
Geoff Langeeba6e12014-02-03 13:12:30 -05002// Copyright (c) 2002-2014 The ANGLE Project Authors. All rights reserved.
apatrick@chromium.org144f2802012-07-12 01:42:34 +00003// Use of this source code is governed by a BSD-style license that can be
4// found in the LICENSE file.
5//
6
7// Context.cpp: Implements the gl::Context class, managing all GL state and performing
8// rendering operations. It is the GLES2 specific implementation of EGLContext.
9
Geoff Lang2b5420c2014-11-19 14:20:15 -050010#include "libANGLE/Context.h"
apatrick@chromium.org144f2802012-07-12 01:42:34 +000011
Jamie Madillb9293972015-02-19 11:07:54 -050012#include <iterator>
13#include <sstream>
14
Geoff Lang0b7eef72014-06-12 14:10:47 -040015#include "common/platform.h"
Jamie Madillb9293972015-02-19 11:07:54 -050016#include "common/utilities.h"
Geoff Lang2b5420c2014-11-19 14:20:15 -050017#include "libANGLE/Buffer.h"
Jamie Madillb9293972015-02-19 11:07:54 -050018#include "libANGLE/Compiler.h"
Jamie Madill9dd0cf02014-11-24 11:38:51 -050019#include "libANGLE/Display.h"
Geoff Lang2b5420c2014-11-19 14:20:15 -050020#include "libANGLE/Fence.h"
21#include "libANGLE/Framebuffer.h"
22#include "libANGLE/FramebufferAttachment.h"
Geoff Lang2b5420c2014-11-19 14:20:15 -050023#include "libANGLE/Program.h"
Geoff Lang2b5420c2014-11-19 14:20:15 -050024#include "libANGLE/Query.h"
Jamie Madillb9293972015-02-19 11:07:54 -050025#include "libANGLE/Renderbuffer.h"
Geoff Lang2b5420c2014-11-19 14:20:15 -050026#include "libANGLE/ResourceManager.h"
27#include "libANGLE/Sampler.h"
Jamie Madill9dd0cf02014-11-24 11:38:51 -050028#include "libANGLE/Surface.h"
Geoff Lang2b5420c2014-11-19 14:20:15 -050029#include "libANGLE/Texture.h"
30#include "libANGLE/TransformFeedback.h"
31#include "libANGLE/VertexArray.h"
32#include "libANGLE/formatutils.h"
33#include "libANGLE/validationES.h"
34#include "libANGLE/renderer/Renderer.h"
shannon.woods@transgaming.com486d9e92013-02-28 23:15:41 +000035
Geoff Langf6db0982015-08-25 13:04:00 -040036namespace
37{
38
39void MarkTransformFeedbackBufferUsage(gl::TransformFeedback *transformFeedback)
40{
Geoff Lang1a683462015-09-29 15:09:59 -040041 if (transformFeedback && transformFeedback->isActive() && !transformFeedback->isPaused())
Geoff Langf6db0982015-08-25 13:04:00 -040042 {
43 for (size_t tfBufferIndex = 0; tfBufferIndex < transformFeedback->getIndexedBufferCount();
44 tfBufferIndex++)
45 {
46 const OffsetBindingPointer<gl::Buffer> &buffer =
47 transformFeedback->getIndexedBuffer(tfBufferIndex);
48 if (buffer.get() != nullptr)
49 {
50 buffer->onTransformFeedback();
51 }
52 }
53 }
54}
Jamie Madill46e6c7a2016-01-18 14:42:30 -050055
56// Attribute map queries.
57EGLint GetClientVersion(const egl::AttributeMap &attribs)
58{
59 return attribs.get(EGL_CONTEXT_CLIENT_VERSION, 1);
60}
61
62GLenum GetResetStrategy(const egl::AttributeMap &attribs)
63{
64 EGLenum attrib = attribs.get(EGL_CONTEXT_OPENGL_RESET_NOTIFICATION_STRATEGY_EXT,
65 EGL_NO_RESET_NOTIFICATION_EXT);
66 switch (attrib)
67 {
68 case EGL_NO_RESET_NOTIFICATION:
69 return GL_NO_RESET_NOTIFICATION_EXT;
70 case EGL_LOSE_CONTEXT_ON_RESET:
71 return GL_LOSE_CONTEXT_ON_RESET_EXT;
72 default:
73 UNREACHABLE();
74 return GL_NONE;
75 }
76}
77
78bool GetRobustAccess(const egl::AttributeMap &attribs)
79{
80 return (attribs.get(EGL_CONTEXT_OPENGL_ROBUST_ACCESS_EXT, EGL_FALSE) == EGL_TRUE);
81}
82
83bool GetDebug(const egl::AttributeMap &attribs)
84{
85 return (attribs.get(EGL_CONTEXT_OPENGL_DEBUG, EGL_FALSE) == EGL_TRUE);
86}
87
88bool GetNoError(const egl::AttributeMap &attribs)
89{
90 return (attribs.get(EGL_CONTEXT_OPENGL_NO_ERROR_KHR, EGL_FALSE) == EGL_TRUE);
91}
92
Geoff Langf6db0982015-08-25 13:04:00 -040093} // anonymous namespace
94
apatrick@chromium.org144f2802012-07-12 01:42:34 +000095namespace gl
96{
daniel@transgaming.comca1ac1f2013-01-11 04:13:05 +000097
Corentin Wallez51706ea2015-08-07 14:39:22 -040098Context::Context(const egl::Config *config,
Corentin Wallez51706ea2015-08-07 14:39:22 -040099 const Context *shareContext,
100 rx::Renderer *renderer,
Jamie Madill46e6c7a2016-01-18 14:42:30 -0500101 const egl::AttributeMap &attribs)
102 : ValidationContext(GetClientVersion(attribs),
Jamie Madillf25855c2015-11-03 11:06:18 -0500103 mState,
104 mCaps,
105 mTextureCaps,
106 mExtensions,
107 nullptr,
Jamie Madill46e6c7a2016-01-18 14:42:30 -0500108 mLimitations,
109 GetNoError(attribs)),
110 mCompiler(nullptr),
Jamie Madillf25855c2015-11-03 11:06:18 -0500111 mRenderer(renderer),
Jamie Madill46e6c7a2016-01-18 14:42:30 -0500112 mClientVersion(GetClientVersion(attribs)),
Corentin Walleze3b10e82015-05-20 11:06:25 -0400113 mConfig(config),
Jamie Madill46e6c7a2016-01-18 14:42:30 -0500114 mClientType(EGL_OPENGL_ES_API),
115 mHasBeenCurrent(false),
116 mContextLost(false),
117 mResetStatus(GL_NO_ERROR),
118 mResetStrategy(GetResetStrategy(attribs)),
119 mRobustAccess(GetRobustAccess(attribs)),
120 mCurrentSurface(nullptr),
121 mResourceManager(nullptr)
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000122{
Jamie Madill46e6c7a2016-01-18 14:42:30 -0500123 ASSERT(!mRobustAccess); // Unimplemented
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000124
Jamie Madill46e6c7a2016-01-18 14:42:30 -0500125 initCaps(mClientVersion);
Geoff Langc0b9ef42014-07-02 10:02:37 -0400126
Jamie Madill46e6c7a2016-01-18 14:42:30 -0500127 mState.initialize(mCaps, mExtensions, mClientVersion, GetDebug(attribs));
Régis Fénéon83107972015-02-05 12:57:44 +0100128
Shannon Woods53a94a82014-06-24 15:20:36 -0400129 mFenceNVHandleAllocator.setBaseHandle(0);
Geoff Lang7dca1862013-07-30 16:30:46 -0400130
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000131 if (shareContext != NULL)
132 {
133 mResourceManager = shareContext->mResourceManager;
134 mResourceManager->addRef();
135 }
136 else
137 {
daniel@transgaming.com370482e2012-11-28 19:32:13 +0000138 mResourceManager = new ResourceManager(mRenderer);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000139 }
140
Jamie Madillc185cb82015-04-28 12:39:08 -0400141 mData.resourceManager = mResourceManager;
142
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000143 // [OpenGL ES 2.0.24] section 3.7 page 83:
144 // In the initial state, TEXTURE_2D and TEXTURE_CUBE_MAP have twodimensional
145 // and cube map texture state vectors respectively associated with them.
146 // In order that access to these initial textures not be lost, they are treated as texture
147 // objects all of whose names are 0.
148
Geoff Lang691e58c2014-12-19 17:03:25 -0500149 Texture *zeroTexture2D = new Texture(mRenderer->createTexture(GL_TEXTURE_2D), 0, GL_TEXTURE_2D);
Jamie Madilldedd7b92014-11-05 16:30:36 -0500150 mZeroTextures[GL_TEXTURE_2D].set(zeroTexture2D);
Jamie Madilldedd7b92014-11-05 16:30:36 -0500151
Geoff Lang691e58c2014-12-19 17:03:25 -0500152 Texture *zeroTextureCube = new Texture(mRenderer->createTexture(GL_TEXTURE_CUBE_MAP), 0, GL_TEXTURE_CUBE_MAP);
Jamie Madilldedd7b92014-11-05 16:30:36 -0500153 mZeroTextures[GL_TEXTURE_CUBE_MAP].set(zeroTextureCube);
Geoff Lang76b10c92014-09-05 16:28:14 -0400154
155 if (mClientVersion >= 3)
156 {
157 // TODO: These could also be enabled via extension
Geoff Lang691e58c2014-12-19 17:03:25 -0500158 Texture *zeroTexture3D = new Texture(mRenderer->createTexture(GL_TEXTURE_3D), 0, GL_TEXTURE_3D);
Jamie Madilldedd7b92014-11-05 16:30:36 -0500159 mZeroTextures[GL_TEXTURE_3D].set(zeroTexture3D);
Geoff Lang76b10c92014-09-05 16:28:14 -0400160
Geoff Lang691e58c2014-12-19 17:03:25 -0500161 Texture *zeroTexture2DArray = new Texture(mRenderer->createTexture(GL_TEXTURE_2D_ARRAY), 0, GL_TEXTURE_2D_ARRAY);
Jamie Madilldedd7b92014-11-05 16:30:36 -0500162 mZeroTextures[GL_TEXTURE_2D_ARRAY].set(zeroTexture2DArray);
Geoff Lang76b10c92014-09-05 16:28:14 -0400163 }
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000164
Jamie Madille6382c32014-11-07 15:05:26 -0500165 mState.initializeZeroTextures(mZeroTextures);
166
Jamie Madill57a89722013-07-02 11:57:03 -0400167 bindVertexArray(0);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000168 bindArrayBuffer(0);
169 bindElementArrayBuffer(0);
Geoff Lang76b10c92014-09-05 16:28:14 -0400170
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000171 bindRenderbuffer(0);
172
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +0000173 bindGenericUniformBuffer(0);
Shannon Woodsf3acaf92014-09-23 18:07:11 -0400174 for (unsigned int i = 0; i < mCaps.maxCombinedUniformBlocks; i++)
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +0000175 {
176 bindIndexedUniformBuffer(0, i, 0, -1);
177 }
178
shannon.woods%transgaming.com@gtempaccount.com51171882013-04-13 03:39:10 +0000179 bindCopyReadBuffer(0);
180 bindCopyWriteBuffer(0);
shannon.woods%transgaming.com@gtempaccount.comc926e5f2013-04-13 03:39:18 +0000181 bindPixelPackBuffer(0);
182 bindPixelUnpackBuffer(0);
shannon.woods%transgaming.com@gtempaccount.com51171882013-04-13 03:39:10 +0000183
Geoff Lang1a683462015-09-29 15:09:59 -0400184 if (mClientVersion >= 3)
185 {
186 // [OpenGL ES 3.0.2] section 2.14.1 pg 85:
187 // In the initial state, a default transform feedback object is bound and treated as
188 // a transform feedback object with a name of zero. That object is bound any time
189 // BindTransformFeedback is called with id of zero
Geoff Lang1a683462015-09-29 15:09:59 -0400190 bindTransformFeedback(0);
191 }
Geoff Langc8058452014-02-03 12:04:11 -0500192
Jamie Madill83f349e2015-09-23 09:50:36 -0400193 mCompiler = new Compiler(mRenderer, getData());
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000194}
195
196Context::~Context()
197{
Geoff Lang7dd2e102014-11-10 15:19:26 -0500198 mState.reset();
Geoff Lang21329412014-12-02 20:50:30 +0000199
Corentin Wallez37c39792015-08-20 14:19:46 -0400200 for (auto framebuffer : mFramebufferMap)
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000201 {
Corentin Wallez37c39792015-08-20 14:19:46 -0400202 // Default framebuffer are owned by their respective Surface
Geoff Langf6227922015-09-04 11:05:47 -0400203 if (framebuffer.second != nullptr && framebuffer.second->id() != 0)
Corentin Wallez37c39792015-08-20 14:19:46 -0400204 {
205 SafeDelete(framebuffer.second);
206 }
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000207 }
208
Corentin Wallez80b24112015-08-25 16:41:57 -0400209 for (auto fence : mFenceNVMap)
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000210 {
Corentin Wallez80b24112015-08-25 16:41:57 -0400211 SafeDelete(fence.second);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000212 }
213
Corentin Wallez80b24112015-08-25 16:41:57 -0400214 for (auto query : mQueryMap)
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000215 {
Geoff Langf0aa8422015-09-29 15:08:34 -0400216 if (query.second != nullptr)
217 {
218 query.second->release();
219 }
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000220 }
221
Corentin Wallez80b24112015-08-25 16:41:57 -0400222 for (auto vertexArray : mVertexArrayMap)
Jamie Madill57a89722013-07-02 11:57:03 -0400223 {
Corentin Wallez80b24112015-08-25 16:41:57 -0400224 SafeDelete(vertexArray.second);
Jamie Madill57a89722013-07-02 11:57:03 -0400225 }
226
Corentin Wallez80b24112015-08-25 16:41:57 -0400227 for (auto transformFeedback : mTransformFeedbackMap)
Geoff Langc8058452014-02-03 12:04:11 -0500228 {
Geoff Lang36167ab2015-12-07 10:27:14 -0500229 if (transformFeedback.second != nullptr)
230 {
231 transformFeedback.second->release();
232 }
Geoff Langc8058452014-02-03 12:04:11 -0500233 }
234
Jamie Madilldedd7b92014-11-05 16:30:36 -0500235 for (auto &zeroTexture : mZeroTextures)
Geoff Lang76b10c92014-09-05 16:28:14 -0400236 {
Jamie Madilldedd7b92014-11-05 16:30:36 -0500237 zeroTexture.second.set(NULL);
Geoff Lang76b10c92014-09-05 16:28:14 -0400238 }
239 mZeroTextures.clear();
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000240
Corentin Wallez51706ea2015-08-07 14:39:22 -0400241 if (mCurrentSurface != nullptr)
242 {
243 releaseSurface();
244 }
245
Jamie Madill1e9ae072014-11-06 15:27:21 -0500246 if (mResourceManager)
247 {
248 mResourceManager->release();
249 }
Geoff Lang492a7e42014-11-05 13:27:06 -0500250
251 SafeDelete(mCompiler);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000252}
253
daniel@transgaming.comad629872012-11-28 19:32:06 +0000254void Context::makeCurrent(egl::Surface *surface)
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000255{
Jamie Madill77a72f62015-04-14 11:18:32 -0400256 ASSERT(surface != nullptr);
257
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000258 if (!mHasBeenCurrent)
259 {
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000260 initRendererString();
Geoff Langcec35902014-04-16 10:52:36 -0400261 initExtensionStrings();
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000262
Shannon Woods53a94a82014-06-24 15:20:36 -0400263 mState.setViewportParams(0, 0, surface->getWidth(), surface->getHeight());
264 mState.setScissorParams(0, 0, surface->getWidth(), surface->getHeight());
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000265
266 mHasBeenCurrent = true;
267 }
268
Jamie Madill1b94d432015-08-07 13:23:23 -0400269 // TODO(jmadill): Rework this when we support ContextImpl
270 mState.setAllDirtyBits();
271
Corentin Wallez51706ea2015-08-07 14:39:22 -0400272 if (mCurrentSurface)
273 {
274 releaseSurface();
275 }
Jamie Madill18fdcbc2015-08-19 18:12:44 +0000276 surface->setIsCurrent(true);
Corentin Wallez37c39792015-08-20 14:19:46 -0400277 mCurrentSurface = surface;
Jamie Madill18fdcbc2015-08-19 18:12:44 +0000278
Corentin Wallez37c39792015-08-20 14:19:46 -0400279 // Update default framebuffer, the binding of the previous default
280 // framebuffer (or lack of) will have a nullptr.
Jamie Madillc1c1cdc2015-04-30 09:42:26 -0400281 {
Corentin Wallez37c39792015-08-20 14:19:46 -0400282 Framebuffer *newDefault = surface->getDefaultFramebuffer();
283 if (mState.getReadFramebuffer() == nullptr)
284 {
285 mState.setReadFramebufferBinding(newDefault);
286 }
287 if (mState.getDrawFramebuffer() == nullptr)
288 {
289 mState.setDrawFramebufferBinding(newDefault);
290 }
291 mFramebufferMap[0] = newDefault;
Jamie Madillc1c1cdc2015-04-30 09:42:26 -0400292 }
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000293}
294
Jamie Madill77a72f62015-04-14 11:18:32 -0400295void Context::releaseSurface()
296{
Corentin Wallez37c39792015-08-20 14:19:46 -0400297 ASSERT(mCurrentSurface != nullptr);
298
299 // Remove the default framebuffer
Corentin Wallez51706ea2015-08-07 14:39:22 -0400300 {
Corentin Wallez37c39792015-08-20 14:19:46 -0400301 Framebuffer *currentDefault = mCurrentSurface->getDefaultFramebuffer();
302 if (mState.getReadFramebuffer() == currentDefault)
303 {
304 mState.setReadFramebufferBinding(nullptr);
305 }
306 if (mState.getDrawFramebuffer() == currentDefault)
307 {
308 mState.setDrawFramebufferBinding(nullptr);
309 }
310 mFramebufferMap.erase(0);
Corentin Wallez51706ea2015-08-07 14:39:22 -0400311 }
312
Corentin Wallez51706ea2015-08-07 14:39:22 -0400313 mCurrentSurface->setIsCurrent(false);
314 mCurrentSurface = nullptr;
Jamie Madill77a72f62015-04-14 11:18:32 -0400315}
316
daniel@transgaming.comf688c0d2012-10-31 17:52:57 +0000317// NOTE: this function should not assume that this context is current!
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000318void Context::markContextLost()
319{
320 if (mResetStrategy == GL_LOSE_CONTEXT_ON_RESET_EXT)
321 mResetStatus = GL_UNKNOWN_CONTEXT_RESET_EXT;
322 mContextLost = true;
323}
324
325bool Context::isContextLost()
326{
327 return mContextLost;
328}
329
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000330GLuint Context::createBuffer()
331{
332 return mResourceManager->createBuffer();
333}
334
335GLuint Context::createProgram()
336{
337 return mResourceManager->createProgram();
338}
339
340GLuint Context::createShader(GLenum type)
341{
Jamie Madill006cbc52015-09-23 16:47:54 -0400342 return mResourceManager->createShader(mRenderer->getRendererLimitations(), type);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000343}
344
345GLuint Context::createTexture()
346{
347 return mResourceManager->createTexture();
348}
349
350GLuint Context::createRenderbuffer()
351{
352 return mResourceManager->createRenderbuffer();
353}
354
Geoff Lang882033e2014-09-30 11:26:07 -0400355GLsync Context::createFenceSync()
Jamie Madillcd055f82013-07-26 11:55:15 -0400356{
357 GLuint handle = mResourceManager->createFenceSync();
358
Cooper Partind8e62a32015-01-29 15:21:25 -0800359 return reinterpret_cast<GLsync>(static_cast<uintptr_t>(handle));
Jamie Madillcd055f82013-07-26 11:55:15 -0400360}
361
Jamie Madill57a89722013-07-02 11:57:03 -0400362GLuint Context::createVertexArray()
363{
Geoff Lang36167ab2015-12-07 10:27:14 -0500364 GLuint vertexArray = mVertexArrayHandleAllocator.allocate();
365 mVertexArrayMap[vertexArray] = nullptr;
366 return vertexArray;
Jamie Madill57a89722013-07-02 11:57:03 -0400367}
368
Jamie Madilldc356042013-07-19 16:36:57 -0400369GLuint Context::createSampler()
370{
371 return mResourceManager->createSampler();
372}
373
Geoff Langc8058452014-02-03 12:04:11 -0500374GLuint Context::createTransformFeedback()
375{
Geoff Lang36167ab2015-12-07 10:27:14 -0500376 GLuint transformFeedback = mTransformFeedbackAllocator.allocate();
377 mTransformFeedbackMap[transformFeedback] = nullptr;
378 return transformFeedback;
Geoff Langc8058452014-02-03 12:04:11 -0500379}
380
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000381// Returns an unused framebuffer name
382GLuint Context::createFramebuffer()
383{
384 GLuint handle = mFramebufferHandleAllocator.allocate();
385
386 mFramebufferMap[handle] = NULL;
387
388 return handle;
389}
390
Jamie Madill33dc8432013-07-26 11:55:05 -0400391GLuint Context::createFenceNV()
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000392{
Jamie Madill33dc8432013-07-26 11:55:05 -0400393 GLuint handle = mFenceNVHandleAllocator.allocate();
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000394
Kenneth Russellcaa549c2014-10-10 17:52:59 -0700395 mFenceNVMap[handle] = new FenceNV(mRenderer->createFenceNV());
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000396
397 return handle;
398}
399
400// Returns an unused query name
401GLuint Context::createQuery()
402{
403 GLuint handle = mQueryHandleAllocator.allocate();
404
405 mQueryMap[handle] = NULL;
406
407 return handle;
408}
409
410void Context::deleteBuffer(GLuint buffer)
411{
412 if (mResourceManager->getBuffer(buffer))
413 {
414 detachBuffer(buffer);
415 }
Jamie Madill893ab082014-05-16 16:56:10 -0400416
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000417 mResourceManager->deleteBuffer(buffer);
418}
419
420void Context::deleteShader(GLuint shader)
421{
422 mResourceManager->deleteShader(shader);
423}
424
425void Context::deleteProgram(GLuint program)
426{
427 mResourceManager->deleteProgram(program);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000428}
429
430void Context::deleteTexture(GLuint texture)
431{
432 if (mResourceManager->getTexture(texture))
433 {
434 detachTexture(texture);
435 }
436
437 mResourceManager->deleteTexture(texture);
438}
439
440void Context::deleteRenderbuffer(GLuint renderbuffer)
441{
442 if (mResourceManager->getRenderbuffer(renderbuffer))
443 {
444 detachRenderbuffer(renderbuffer);
445 }
Jamie Madill893ab082014-05-16 16:56:10 -0400446
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000447 mResourceManager->deleteRenderbuffer(renderbuffer);
448}
449
Jamie Madillcd055f82013-07-26 11:55:15 -0400450void Context::deleteFenceSync(GLsync fenceSync)
451{
452 // The spec specifies the underlying Fence object is not deleted until all current
453 // wait commands finish. However, since the name becomes invalid, we cannot query the fence,
454 // and since our API is currently designed for being called from a single thread, we can delete
455 // the fence immediately.
Minmin Gong794e0002015-04-07 18:31:54 -0700456 mResourceManager->deleteFenceSync(static_cast<GLuint>(reinterpret_cast<uintptr_t>(fenceSync)));
Jamie Madillcd055f82013-07-26 11:55:15 -0400457}
458
Jamie Madill57a89722013-07-02 11:57:03 -0400459void Context::deleteVertexArray(GLuint vertexArray)
460{
Geoff Lang36167ab2015-12-07 10:27:14 -0500461 auto iter = mVertexArrayMap.find(vertexArray);
462 if (iter != mVertexArrayMap.end())
Geoff Lang50b3fe82015-12-08 14:49:12 +0000463 {
Geoff Lang36167ab2015-12-07 10:27:14 -0500464 VertexArray *vertexArrayObject = iter->second;
465 if (vertexArrayObject != nullptr)
466 {
467 detachVertexArray(vertexArray);
468 delete vertexArrayObject;
469 }
Geoff Lang50b3fe82015-12-08 14:49:12 +0000470
Geoff Lang36167ab2015-12-07 10:27:14 -0500471 mVertexArrayMap.erase(iter);
472 mVertexArrayHandleAllocator.release(vertexArray);
Jamie Madill57a89722013-07-02 11:57:03 -0400473 }
474}
475
Jamie Madilldc356042013-07-19 16:36:57 -0400476void Context::deleteSampler(GLuint sampler)
477{
478 if (mResourceManager->getSampler(sampler))
479 {
480 detachSampler(sampler);
481 }
482
483 mResourceManager->deleteSampler(sampler);
484}
485
Geoff Langc8058452014-02-03 12:04:11 -0500486void Context::deleteTransformFeedback(GLuint transformFeedback)
487{
Jamie Madill5fd0b2d2015-01-05 13:38:44 -0500488 auto iter = mTransformFeedbackMap.find(transformFeedback);
Geoff Langc8058452014-02-03 12:04:11 -0500489 if (iter != mTransformFeedbackMap.end())
490 {
Geoff Lang36167ab2015-12-07 10:27:14 -0500491 TransformFeedback *transformFeedbackObject = iter->second;
492 if (transformFeedbackObject != nullptr)
493 {
494 detachTransformFeedback(transformFeedback);
495 transformFeedbackObject->release();
496 }
497
Geoff Lang50b3fe82015-12-08 14:49:12 +0000498 mTransformFeedbackMap.erase(iter);
Geoff Lang36167ab2015-12-07 10:27:14 -0500499 mTransformFeedbackAllocator.release(transformFeedback);
Geoff Langc8058452014-02-03 12:04:11 -0500500 }
501}
502
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000503void Context::deleteFramebuffer(GLuint framebuffer)
504{
505 FramebufferMap::iterator framebufferObject = mFramebufferMap.find(framebuffer);
506
507 if (framebufferObject != mFramebufferMap.end())
508 {
509 detachFramebuffer(framebuffer);
510
511 mFramebufferHandleAllocator.release(framebufferObject->first);
512 delete framebufferObject->second;
513 mFramebufferMap.erase(framebufferObject);
514 }
515}
516
Jamie Madill33dc8432013-07-26 11:55:05 -0400517void Context::deleteFenceNV(GLuint fence)
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000518{
Jamie Madill33dc8432013-07-26 11:55:05 -0400519 FenceNVMap::iterator fenceObject = mFenceNVMap.find(fence);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000520
Jamie Madill33dc8432013-07-26 11:55:05 -0400521 if (fenceObject != mFenceNVMap.end())
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000522 {
Jamie Madill33dc8432013-07-26 11:55:05 -0400523 mFenceNVHandleAllocator.release(fenceObject->first);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000524 delete fenceObject->second;
Jamie Madill33dc8432013-07-26 11:55:05 -0400525 mFenceNVMap.erase(fenceObject);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000526 }
527}
528
529void Context::deleteQuery(GLuint query)
530{
531 QueryMap::iterator queryObject = mQueryMap.find(query);
532 if (queryObject != mQueryMap.end())
533 {
534 mQueryHandleAllocator.release(queryObject->first);
535 if (queryObject->second)
536 {
537 queryObject->second->release();
538 }
539 mQueryMap.erase(queryObject);
540 }
541}
542
Geoff Lang70d0f492015-12-10 17:45:46 -0500543Buffer *Context::getBuffer(GLuint handle) const
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000544{
545 return mResourceManager->getBuffer(handle);
546}
547
Geoff Lang48dcae72014-02-05 16:28:24 -0500548Shader *Context::getShader(GLuint handle) const
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000549{
550 return mResourceManager->getShader(handle);
551}
552
Geoff Lang48dcae72014-02-05 16:28:24 -0500553Program *Context::getProgram(GLuint handle) const
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000554{
555 return mResourceManager->getProgram(handle);
556}
557
Jamie Madill570f7c82014-07-03 10:38:54 -0400558Texture *Context::getTexture(GLuint handle) const
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000559{
560 return mResourceManager->getTexture(handle);
561}
562
Geoff Lang70d0f492015-12-10 17:45:46 -0500563Renderbuffer *Context::getRenderbuffer(GLuint handle) const
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000564{
565 return mResourceManager->getRenderbuffer(handle);
566}
567
Jamie Madillcd055f82013-07-26 11:55:15 -0400568FenceSync *Context::getFenceSync(GLsync handle) const
569{
Minmin Gong794e0002015-04-07 18:31:54 -0700570 return mResourceManager->getFenceSync(static_cast<GLuint>(reinterpret_cast<uintptr_t>(handle)));
Jamie Madillcd055f82013-07-26 11:55:15 -0400571}
572
Jamie Madill57a89722013-07-02 11:57:03 -0400573VertexArray *Context::getVertexArray(GLuint handle) const
574{
575 auto vertexArray = mVertexArrayMap.find(handle);
Geoff Lang36167ab2015-12-07 10:27:14 -0500576 return (vertexArray != mVertexArrayMap.end()) ? vertexArray->second : nullptr;
Jamie Madill57a89722013-07-02 11:57:03 -0400577}
578
Jamie Madilldc356042013-07-19 16:36:57 -0400579Sampler *Context::getSampler(GLuint handle) const
580{
581 return mResourceManager->getSampler(handle);
582}
583
Geoff Langc8058452014-02-03 12:04:11 -0500584TransformFeedback *Context::getTransformFeedback(GLuint handle) const
585{
Geoff Lang36167ab2015-12-07 10:27:14 -0500586 auto iter = mTransformFeedbackMap.find(handle);
587 return (iter != mTransformFeedbackMap.end()) ? iter->second : nullptr;
Geoff Langc8058452014-02-03 12:04:11 -0500588}
589
Geoff Lang70d0f492015-12-10 17:45:46 -0500590LabeledObject *Context::getLabeledObject(GLenum identifier, GLuint name) const
591{
592 switch (identifier)
593 {
594 case GL_BUFFER:
595 return getBuffer(name);
596 case GL_SHADER:
597 return getShader(name);
598 case GL_PROGRAM:
599 return getProgram(name);
600 case GL_VERTEX_ARRAY:
601 return getVertexArray(name);
602 case GL_QUERY:
603 return getQuery(name);
604 case GL_TRANSFORM_FEEDBACK:
605 return getTransformFeedback(name);
606 case GL_SAMPLER:
607 return getSampler(name);
608 case GL_TEXTURE:
609 return getTexture(name);
610 case GL_RENDERBUFFER:
611 return getRenderbuffer(name);
612 case GL_FRAMEBUFFER:
613 return getFramebuffer(name);
614 default:
615 UNREACHABLE();
616 return nullptr;
617 }
618}
619
620LabeledObject *Context::getLabeledObjectFromPtr(const void *ptr) const
621{
622 return getFenceSync(reinterpret_cast<GLsync>(const_cast<void *>(ptr)));
623}
624
Jamie Madilldc356042013-07-19 16:36:57 -0400625bool Context::isSampler(GLuint samplerName) const
626{
627 return mResourceManager->isSampler(samplerName);
628}
629
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000630void Context::bindArrayBuffer(unsigned int buffer)
631{
632 mResourceManager->checkBufferAllocation(buffer);
633
Shannon Woods53a94a82014-06-24 15:20:36 -0400634 mState.setArrayBufferBinding(getBuffer(buffer));
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000635}
636
637void Context::bindElementArrayBuffer(unsigned int buffer)
638{
639 mResourceManager->checkBufferAllocation(buffer);
640
Shannon Woods53a94a82014-06-24 15:20:36 -0400641 mState.getVertexArray()->setElementArrayBuffer(getBuffer(buffer));
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000642}
643
Jamie Madilldedd7b92014-11-05 16:30:36 -0500644void Context::bindTexture(GLenum target, GLuint handle)
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000645{
Jamie Madilldedd7b92014-11-05 16:30:36 -0500646 Texture *texture = NULL;
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000647
Jamie Madilldedd7b92014-11-05 16:30:36 -0500648 if (handle == 0)
649 {
650 texture = mZeroTextures[target].get();
651 }
652 else
653 {
654 mResourceManager->checkTextureAllocation(handle, target);
655 texture = getTexture(handle);
656 }
657
658 ASSERT(texture);
659
660 mState.setSamplerTexture(target, texture);
shannon.woods%transgaming.com@gtempaccount.com90dbc442013-04-13 03:46:14 +0000661}
662
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000663void Context::bindReadFramebuffer(GLuint framebuffer)
664{
665 if (!getFramebuffer(framebuffer))
666 {
Jamie Madilld1405e52015-03-05 15:41:39 -0500667 mFramebufferMap[framebuffer] = new Framebuffer(mCaps, mRenderer, framebuffer);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000668 }
669
Shannon Woods53a94a82014-06-24 15:20:36 -0400670 mState.setReadFramebufferBinding(getFramebuffer(framebuffer));
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000671}
672
673void Context::bindDrawFramebuffer(GLuint framebuffer)
674{
675 if (!getFramebuffer(framebuffer))
676 {
Jamie Madilld1405e52015-03-05 15:41:39 -0500677 mFramebufferMap[framebuffer] = new Framebuffer(mCaps, mRenderer, framebuffer);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000678 }
679
Shannon Woods53a94a82014-06-24 15:20:36 -0400680 mState.setDrawFramebufferBinding(getFramebuffer(framebuffer));
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000681}
682
683void Context::bindRenderbuffer(GLuint renderbuffer)
684{
685 mResourceManager->checkRenderbufferAllocation(renderbuffer);
686
Shannon Woods53a94a82014-06-24 15:20:36 -0400687 mState.setRenderbufferBinding(getRenderbuffer(renderbuffer));
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000688}
689
Jamie Madill57a89722013-07-02 11:57:03 -0400690void Context::bindVertexArray(GLuint vertexArray)
691{
Geoff Lang36167ab2015-12-07 10:27:14 -0500692 checkVertexArrayAllocation(vertexArray);
Jamie Madill57a89722013-07-02 11:57:03 -0400693
Shannon Woods53a94a82014-06-24 15:20:36 -0400694 mState.setVertexArrayBinding(getVertexArray(vertexArray));
Jamie Madill57a89722013-07-02 11:57:03 -0400695}
696
Jamie Madilldc356042013-07-19 16:36:57 -0400697void Context::bindSampler(GLuint textureUnit, GLuint sampler)
698{
Geoff Lang76b10c92014-09-05 16:28:14 -0400699 ASSERT(textureUnit < mCaps.maxCombinedTextureImageUnits);
Jamie Madilldc356042013-07-19 16:36:57 -0400700 mResourceManager->checkSamplerAllocation(sampler);
701
Shannon Woods53a94a82014-06-24 15:20:36 -0400702 mState.setSamplerBinding(textureUnit, getSampler(sampler));
Jamie Madilldc356042013-07-19 16:36:57 -0400703}
704
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +0000705void Context::bindGenericUniformBuffer(GLuint buffer)
706{
707 mResourceManager->checkBufferAllocation(buffer);
708
Shannon Woods53a94a82014-06-24 15:20:36 -0400709 mState.setGenericUniformBufferBinding(getBuffer(buffer));
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +0000710}
711
712void Context::bindIndexedUniformBuffer(GLuint buffer, GLuint index, GLintptr offset, GLsizeiptr size)
shannon.woods%transgaming.com@gtempaccount.com34089352013-04-13 03:36:57 +0000713{
714 mResourceManager->checkBufferAllocation(buffer);
715
Shannon Woods53a94a82014-06-24 15:20:36 -0400716 mState.setIndexedUniformBufferBinding(index, getBuffer(buffer), offset, size);
shannon.woods%transgaming.com@gtempaccount.com34089352013-04-13 03:36:57 +0000717}
718
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +0000719void Context::bindGenericTransformFeedbackBuffer(GLuint buffer)
720{
721 mResourceManager->checkBufferAllocation(buffer);
722
Geoff Lang045536b2015-03-27 15:17:18 -0400723 mState.getCurrentTransformFeedback()->bindGenericBuffer(getBuffer(buffer));
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +0000724}
725
726void Context::bindIndexedTransformFeedbackBuffer(GLuint buffer, GLuint index, GLintptr offset, GLsizeiptr size)
shannon.woods%transgaming.com@gtempaccount.com34089352013-04-13 03:36:57 +0000727{
728 mResourceManager->checkBufferAllocation(buffer);
729
Geoff Lang045536b2015-03-27 15:17:18 -0400730 mState.getCurrentTransformFeedback()->bindIndexedBuffer(index, getBuffer(buffer), offset, size);
shannon.woods%transgaming.com@gtempaccount.com34089352013-04-13 03:36:57 +0000731}
732
shannon.woods%transgaming.com@gtempaccount.com51171882013-04-13 03:39:10 +0000733void Context::bindCopyReadBuffer(GLuint buffer)
734{
735 mResourceManager->checkBufferAllocation(buffer);
736
Shannon Woods53a94a82014-06-24 15:20:36 -0400737 mState.setCopyReadBufferBinding(getBuffer(buffer));
shannon.woods%transgaming.com@gtempaccount.com51171882013-04-13 03:39:10 +0000738}
739
740void Context::bindCopyWriteBuffer(GLuint buffer)
741{
742 mResourceManager->checkBufferAllocation(buffer);
743
Shannon Woods53a94a82014-06-24 15:20:36 -0400744 mState.setCopyWriteBufferBinding(getBuffer(buffer));
shannon.woods%transgaming.com@gtempaccount.com51171882013-04-13 03:39:10 +0000745}
746
shannon.woods%transgaming.com@gtempaccount.comc926e5f2013-04-13 03:39:18 +0000747void Context::bindPixelPackBuffer(GLuint buffer)
748{
749 mResourceManager->checkBufferAllocation(buffer);
750
Shannon Woods53a94a82014-06-24 15:20:36 -0400751 mState.setPixelPackBufferBinding(getBuffer(buffer));
shannon.woods%transgaming.com@gtempaccount.comc926e5f2013-04-13 03:39:18 +0000752}
753
754void Context::bindPixelUnpackBuffer(GLuint buffer)
755{
756 mResourceManager->checkBufferAllocation(buffer);
757
Shannon Woods53a94a82014-06-24 15:20:36 -0400758 mState.setPixelUnpackBufferBinding(getBuffer(buffer));
shannon.woods%transgaming.com@gtempaccount.comc926e5f2013-04-13 03:39:18 +0000759}
760
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000761void Context::useProgram(GLuint program)
762{
Geoff Lang7dd2e102014-11-10 15:19:26 -0500763 mState.setProgram(getProgram(program));
daniel@transgaming.com95d29422012-07-24 18:36:10 +0000764}
765
Geoff Langc8058452014-02-03 12:04:11 -0500766void Context::bindTransformFeedback(GLuint transformFeedback)
767{
Geoff Lang36167ab2015-12-07 10:27:14 -0500768 checkTransformFeedbackAllocation(transformFeedback);
769
Shannon Woods53a94a82014-06-24 15:20:36 -0400770 mState.setTransformFeedbackBinding(getTransformFeedback(transformFeedback));
Geoff Langc8058452014-02-03 12:04:11 -0500771}
772
Geoff Lang5aad9672014-09-08 11:10:42 -0400773Error Context::beginQuery(GLenum target, GLuint query)
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000774{
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000775 Query *queryObject = getQuery(query, true, target);
Jamie Madilldb2f14c2014-05-13 13:56:30 -0400776 ASSERT(queryObject);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000777
Geoff Lang5aad9672014-09-08 11:10:42 -0400778 // begin query
779 Error error = queryObject->begin();
780 if (error.isError())
781 {
782 return error;
783 }
784
785 // set query as active for specified target only if begin succeeded
Shannon Woods53a94a82014-06-24 15:20:36 -0400786 mState.setActiveQuery(target, queryObject);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000787
Geoff Lang5aad9672014-09-08 11:10:42 -0400788 return Error(GL_NO_ERROR);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000789}
790
Geoff Lang5aad9672014-09-08 11:10:42 -0400791Error Context::endQuery(GLenum target)
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000792{
Shannon Woods53a94a82014-06-24 15:20:36 -0400793 Query *queryObject = mState.getActiveQuery(target);
Jamie Madill45c785d2014-05-13 14:09:34 -0400794 ASSERT(queryObject);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000795
Geoff Lang5aad9672014-09-08 11:10:42 -0400796 gl::Error error = queryObject->end();
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000797
Geoff Lang5aad9672014-09-08 11:10:42 -0400798 // Always unbind the query, even if there was an error. This may delete the query object.
Shannon Woods53a94a82014-06-24 15:20:36 -0400799 mState.setActiveQuery(target, NULL);
Geoff Lang5aad9672014-09-08 11:10:42 -0400800
801 return error;
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000802}
803
Jamie Madill1fc7e2c2014-01-21 16:47:10 -0500804Framebuffer *Context::getFramebuffer(unsigned int handle) const
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000805{
Jamie Madill1fc7e2c2014-01-21 16:47:10 -0500806 FramebufferMap::const_iterator framebuffer = mFramebufferMap.find(handle);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000807
808 if (framebuffer == mFramebufferMap.end())
809 {
810 return NULL;
811 }
812 else
813 {
814 return framebuffer->second;
815 }
816}
817
Jamie Madill33dc8432013-07-26 11:55:05 -0400818FenceNV *Context::getFenceNV(unsigned int handle)
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000819{
Jamie Madill33dc8432013-07-26 11:55:05 -0400820 FenceNVMap::iterator fence = mFenceNVMap.find(handle);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000821
Jamie Madill33dc8432013-07-26 11:55:05 -0400822 if (fence == mFenceNVMap.end())
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000823 {
824 return NULL;
825 }
826 else
827 {
828 return fence->second;
829 }
830}
831
832Query *Context::getQuery(unsigned int handle, bool create, GLenum type)
833{
834 QueryMap::iterator query = mQueryMap.find(handle);
835
836 if (query == mQueryMap.end())
837 {
838 return NULL;
839 }
840 else
841 {
842 if (!query->second && create)
843 {
Brandon Jones3b579e32014-08-08 10:54:25 -0700844 query->second = new Query(mRenderer->createQuery(type), handle);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000845 query->second->addRef();
846 }
847 return query->second;
848 }
849}
850
Geoff Lang70d0f492015-12-10 17:45:46 -0500851Query *Context::getQuery(GLuint handle) const
852{
853 auto iter = mQueryMap.find(handle);
854 return (iter != mQueryMap.end()) ? iter->second : nullptr;
855}
856
Jamie Madill1fc7e2c2014-01-21 16:47:10 -0500857Texture *Context::getTargetTexture(GLenum target) const
858{
Geoff Lang691e58c2014-12-19 17:03:25 -0500859 ASSERT(ValidTextureTarget(this, target));
Jamie Madillc29968b2016-01-20 11:17:23 -0500860 return mState.getTargetTexture(target);
shannon.woods%transgaming.com@gtempaccount.comc926e5f2013-04-13 03:39:18 +0000861}
862
Geoff Lang76b10c92014-09-05 16:28:14 -0400863Texture *Context::getSamplerTexture(unsigned int sampler, GLenum type) const
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000864{
Jamie Madilldedd7b92014-11-05 16:30:36 -0500865 return mState.getSamplerTexture(sampler, type);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000866}
867
Geoff Lang492a7e42014-11-05 13:27:06 -0500868Compiler *Context::getCompiler() const
869{
870 return mCompiler;
871}
872
Jamie Madill893ab082014-05-16 16:56:10 -0400873void Context::getBooleanv(GLenum pname, GLboolean *params)
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000874{
875 switch (pname)
876 {
daniel@transgaming.comf39967e2012-11-28 19:35:56 +0000877 case GL_SHADER_COMPILER: *params = GL_TRUE; break;
daniel@transgaming.comf39967e2012-11-28 19:35:56 +0000878 case GL_CONTEXT_ROBUST_ACCESS_EXT: *params = mRobustAccess ? GL_TRUE : GL_FALSE; break;
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000879 default:
Shannon Woods53a94a82014-06-24 15:20:36 -0400880 mState.getBooleanv(pname, params);
Jamie Madill893ab082014-05-16 16:56:10 -0400881 break;
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000882 }
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000883}
884
Jamie Madill893ab082014-05-16 16:56:10 -0400885void Context::getFloatv(GLenum pname, GLfloat *params)
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000886{
Shannon Woods53a94a82014-06-24 15:20:36 -0400887 // Queries about context capabilities and maximums are answered by Context.
888 // Queries about current GL state values are answered by State.
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000889 switch (pname)
890 {
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000891 case GL_ALIASED_LINE_WIDTH_RANGE:
Geoff Langc0b9ef42014-07-02 10:02:37 -0400892 params[0] = mCaps.minAliasedLineWidth;
893 params[1] = mCaps.maxAliasedLineWidth;
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000894 break;
895 case GL_ALIASED_POINT_SIZE_RANGE:
Geoff Langc0b9ef42014-07-02 10:02:37 -0400896 params[0] = mCaps.minAliasedPointSize;
897 params[1] = mCaps.maxAliasedPointSize;
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000898 break;
daniel@transgaming.com07ab8412012-07-12 15:17:09 +0000899 case GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT:
Geoff Langc0b9ef42014-07-02 10:02:37 -0400900 ASSERT(mExtensions.textureFilterAnisotropic);
901 *params = mExtensions.maxTextureAnisotropy;
daniel@transgaming.com07ab8412012-07-12 15:17:09 +0000902 break;
Geoff Lange6d4e122015-06-29 13:33:55 -0400903 case GL_MAX_TEXTURE_LOD_BIAS:
904 *params = mCaps.maxLODBias;
905 break;
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000906 default:
Shannon Woods53a94a82014-06-24 15:20:36 -0400907 mState.getFloatv(pname, params);
Jamie Madill893ab082014-05-16 16:56:10 -0400908 break;
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000909 }
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000910}
911
Jamie Madill893ab082014-05-16 16:56:10 -0400912void Context::getIntegerv(GLenum pname, GLint *params)
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000913{
Shannon Woods53a94a82014-06-24 15:20:36 -0400914 // Queries about context capabilities and maximums are answered by Context.
915 // Queries about current GL state values are answered by State.
shannon.woods%transgaming.com@gtempaccount.combc373e52013-04-13 03:31:23 +0000916
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000917 switch (pname)
918 {
Geoff Lang301d1612014-07-09 10:34:37 -0400919 case GL_MAX_VERTEX_ATTRIBS: *params = mCaps.maxVertexAttributes; break;
920 case GL_MAX_VERTEX_UNIFORM_VECTORS: *params = mCaps.maxVertexUniformVectors; break;
921 case GL_MAX_VERTEX_UNIFORM_COMPONENTS: *params = mCaps.maxVertexUniformComponents; break;
Geoff Lang3a61c322014-07-10 13:01:54 -0400922 case GL_MAX_VARYING_VECTORS: *params = mCaps.maxVaryingVectors; break;
923 case GL_MAX_VARYING_COMPONENTS: *params = mCaps.maxVertexOutputComponents; break;
924 case GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS: *params = mCaps.maxCombinedTextureImageUnits; break;
Geoff Lang301d1612014-07-09 10:34:37 -0400925 case GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS: *params = mCaps.maxVertexTextureImageUnits; break;
926 case GL_MAX_TEXTURE_IMAGE_UNITS: *params = mCaps.maxTextureImageUnits; break;
927 case GL_MAX_FRAGMENT_UNIFORM_VECTORS: *params = mCaps.maxFragmentUniformVectors; break;
Geoff Lange7468902015-10-02 10:46:24 -0400928 case GL_MAX_FRAGMENT_UNIFORM_COMPONENTS: *params = mCaps.maxFragmentUniformComponents; break;
Geoff Langc0b9ef42014-07-02 10:02:37 -0400929 case GL_MAX_RENDERBUFFER_SIZE: *params = mCaps.maxRenderbufferSize; break;
930 case GL_MAX_COLOR_ATTACHMENTS_EXT: *params = mCaps.maxColorAttachments; break;
931 case GL_MAX_DRAW_BUFFERS_EXT: *params = mCaps.maxDrawBuffers; break;
Jamie Madill1caff072013-07-19 16:36:56 -0400932 //case GL_FRAMEBUFFER_BINDING: // now equivalent to GL_DRAW_FRAMEBUFFER_BINDING_ANGLE
Jamie Madill1caff072013-07-19 16:36:56 -0400933 case GL_SUBPIXEL_BITS: *params = 4; break;
Geoff Langc0b9ef42014-07-02 10:02:37 -0400934 case GL_MAX_TEXTURE_SIZE: *params = mCaps.max2DTextureSize; break;
935 case GL_MAX_CUBE_MAP_TEXTURE_SIZE: *params = mCaps.maxCubeMapTextureSize; break;
936 case GL_MAX_3D_TEXTURE_SIZE: *params = mCaps.max3DTextureSize; break;
937 case GL_MAX_ARRAY_TEXTURE_LAYERS: *params = mCaps.maxArrayTextureLayers; break;
Geoff Lang3a61c322014-07-10 13:01:54 -0400938 case GL_UNIFORM_BUFFER_OFFSET_ALIGNMENT: *params = mCaps.uniformBufferOffsetAlignment; break;
939 case GL_MAX_UNIFORM_BUFFER_BINDINGS: *params = mCaps.maxUniformBufferBindings; break;
Geoff Lang301d1612014-07-09 10:34:37 -0400940 case GL_MAX_VERTEX_UNIFORM_BLOCKS: *params = mCaps.maxVertexUniformBlocks; break;
941 case GL_MAX_FRAGMENT_UNIFORM_BLOCKS: *params = mCaps.maxFragmentUniformBlocks; break;
Geoff Lang3a61c322014-07-10 13:01:54 -0400942 case GL_MAX_COMBINED_UNIFORM_BLOCKS: *params = mCaps.maxCombinedTextureImageUnits; break;
Geoff Lange6d4e122015-06-29 13:33:55 -0400943 case GL_MAX_VERTEX_OUTPUT_COMPONENTS: *params = mCaps.maxVertexOutputComponents; break;
944 case GL_MAX_FRAGMENT_INPUT_COMPONENTS: *params = mCaps.maxFragmentInputComponents; break;
945 case GL_MIN_PROGRAM_TEXEL_OFFSET: *params = mCaps.minProgramTexelOffset; break;
946 case GL_MAX_PROGRAM_TEXEL_OFFSET: *params = mCaps.maxProgramTexelOffset; break;
Jamie Madillee7010d2013-10-17 10:45:47 -0400947 case GL_MAJOR_VERSION: *params = mClientVersion; break;
948 case GL_MINOR_VERSION: *params = 0; break;
Geoff Lang900013c2014-07-07 11:32:19 -0400949 case GL_MAX_ELEMENTS_INDICES: *params = mCaps.maxElementsIndices; break;
950 case GL_MAX_ELEMENTS_VERTICES: *params = mCaps.maxElementsVertices; break;
Geoff Lang05881a02014-07-10 14:05:30 -0400951 case GL_MAX_TRANSFORM_FEEDBACK_INTERLEAVED_COMPONENTS: *params = mCaps.maxTransformFeedbackInterleavedComponents; break;
952 case GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_ATTRIBS: *params = mCaps.maxTransformFeedbackSeparateAttributes; break;
953 case GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_COMPONENTS: *params = mCaps.maxTransformFeedbackSeparateComponents; break;
Cooper Partin4d61f7e2015-08-12 10:56:50 -0700954 case GL_NUM_COMPRESSED_TEXTURE_FORMATS:
955 *params = static_cast<GLint>(mCaps.compressedTextureFormats.size());
956 break;
Geoff Langdef624b2015-04-13 10:46:56 -0400957 case GL_MAX_SAMPLES_ANGLE: *params = mCaps.maxSamples; break;
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000958 case GL_MAX_VIEWPORT_DIMS:
959 {
Geoff Langc0b9ef42014-07-02 10:02:37 -0400960 params[0] = mCaps.maxViewportWidth;
961 params[1] = mCaps.maxViewportHeight;
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000962 }
963 break;
964 case GL_COMPRESSED_TEXTURE_FORMATS:
Geoff Lang900013c2014-07-07 11:32:19 -0400965 std::copy(mCaps.compressedTextureFormats.begin(), mCaps.compressedTextureFormats.end(), params);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000966 break;
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000967 case GL_RESET_NOTIFICATION_STRATEGY_EXT:
968 *params = mResetStrategy;
969 break;
Geoff Lang900013c2014-07-07 11:32:19 -0400970 case GL_NUM_SHADER_BINARY_FORMATS:
Cooper Partin4d61f7e2015-08-12 10:56:50 -0700971 *params = static_cast<GLint>(mCaps.shaderBinaryFormats.size());
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000972 break;
Geoff Lang900013c2014-07-07 11:32:19 -0400973 case GL_SHADER_BINARY_FORMATS:
974 std::copy(mCaps.shaderBinaryFormats.begin(), mCaps.shaderBinaryFormats.end(), params);
975 break;
976 case GL_NUM_PROGRAM_BINARY_FORMATS:
Cooper Partin4d61f7e2015-08-12 10:56:50 -0700977 *params = static_cast<GLint>(mCaps.programBinaryFormats.size());
Geoff Lang900013c2014-07-07 11:32:19 -0400978 break;
979 case GL_PROGRAM_BINARY_FORMATS:
980 std::copy(mCaps.programBinaryFormats.begin(), mCaps.programBinaryFormats.end(), params);
apatrick@chromium.org144f2802012-07-12 01:42:34 +0000981 break;
Geoff Lang23c81692013-08-12 10:46:58 -0400982 case GL_NUM_EXTENSIONS:
Geoff Langcec35902014-04-16 10:52:36 -0400983 *params = static_cast<GLint>(mExtensionStrings.size());
Geoff Lang23c81692013-08-12 10:46:58 -0400984 break;
Geoff Lang70d0f492015-12-10 17:45:46 -0500985
986 // GL_KHR_debug
987 case GL_MAX_DEBUG_MESSAGE_LENGTH:
988 *params = mExtensions.maxDebugMessageLength;
989 break;
990 case GL_MAX_DEBUG_LOGGED_MESSAGES:
991 *params = mExtensions.maxDebugLoggedMessages;
992 break;
993 case GL_MAX_DEBUG_GROUP_STACK_DEPTH:
994 *params = mExtensions.maxDebugGroupStackDepth;
995 break;
996 case GL_MAX_LABEL_LENGTH:
997 *params = mExtensions.maxLabelLength;
998 break;
999
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001000 default:
Jamie Madill48faf802014-11-06 15:27:22 -05001001 mState.getIntegerv(getData(), pname, params);
Jamie Madill893ab082014-05-16 16:56:10 -04001002 break;
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001003 }
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001004}
1005
Jamie Madill893ab082014-05-16 16:56:10 -04001006void Context::getInteger64v(GLenum pname, GLint64 *params)
Jamie Madill0fda9862013-07-19 16:36:55 -04001007{
Shannon Woods53a94a82014-06-24 15:20:36 -04001008 // Queries about context capabilities and maximums are answered by Context.
1009 // Queries about current GL state values are answered by State.
Jamie Madill0fda9862013-07-19 16:36:55 -04001010 switch (pname)
1011 {
1012 case GL_MAX_ELEMENT_INDEX:
Geoff Langc0b9ef42014-07-02 10:02:37 -04001013 *params = mCaps.maxElementIndex;
Jamie Madill0fda9862013-07-19 16:36:55 -04001014 break;
1015 case GL_MAX_UNIFORM_BLOCK_SIZE:
Geoff Lang3a61c322014-07-10 13:01:54 -04001016 *params = mCaps.maxUniformBlockSize;
Jamie Madill0fda9862013-07-19 16:36:55 -04001017 break;
1018 case GL_MAX_COMBINED_VERTEX_UNIFORM_COMPONENTS:
Geoff Lang3a61c322014-07-10 13:01:54 -04001019 *params = mCaps.maxCombinedVertexUniformComponents;
Jamie Madill0fda9862013-07-19 16:36:55 -04001020 break;
1021 case GL_MAX_COMBINED_FRAGMENT_UNIFORM_COMPONENTS:
Geoff Lang3a61c322014-07-10 13:01:54 -04001022 *params = mCaps.maxCombinedFragmentUniformComponents;
Jamie Madill0fda9862013-07-19 16:36:55 -04001023 break;
1024 case GL_MAX_SERVER_WAIT_TIMEOUT:
Geoff Lang900013c2014-07-07 11:32:19 -04001025 *params = mCaps.maxServerWaitTimeout;
Jamie Madill0fda9862013-07-19 16:36:55 -04001026 break;
1027 default:
Jamie Madill893ab082014-05-16 16:56:10 -04001028 UNREACHABLE();
1029 break;
Jamie Madill0fda9862013-07-19 16:36:55 -04001030 }
Jamie Madill0fda9862013-07-19 16:36:55 -04001031}
1032
Geoff Lang70d0f492015-12-10 17:45:46 -05001033void Context::getPointerv(GLenum pname, void **params) const
1034{
1035 mState.getPointerv(pname, params);
1036}
1037
Shannon Woods1b2fb852013-08-19 14:28:48 -04001038bool Context::getIndexedIntegerv(GLenum target, GLuint index, GLint *data)
1039{
Shannon Woods53a94a82014-06-24 15:20:36 -04001040 // Queries about context capabilities and maximums are answered by Context.
1041 // Queries about current GL state values are answered by State.
Jamie Madill77a72f62015-04-14 11:18:32 -04001042 // Indexed integer queries all refer to current state, so this function is a
Shannon Woods53a94a82014-06-24 15:20:36 -04001043 // mere passthrough.
1044 return mState.getIndexedIntegerv(target, index, data);
Shannon Woods1b2fb852013-08-19 14:28:48 -04001045}
1046
1047bool Context::getIndexedInteger64v(GLenum target, GLuint index, GLint64 *data)
1048{
Shannon Woods53a94a82014-06-24 15:20:36 -04001049 // Queries about context capabilities and maximums are answered by Context.
1050 // Queries about current GL state values are answered by State.
Jamie Madill77a72f62015-04-14 11:18:32 -04001051 // Indexed integer queries all refer to current state, so this function is a
Shannon Woods53a94a82014-06-24 15:20:36 -04001052 // mere passthrough.
1053 return mState.getIndexedInteger64v(target, index, data);
Shannon Woods1b2fb852013-08-19 14:28:48 -04001054}
1055
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001056bool Context::getQueryParameterInfo(GLenum pname, GLenum *type, unsigned int *numParams)
1057{
shannon.woods%transgaming.com@gtempaccount.combc373e52013-04-13 03:31:23 +00001058 if (pname >= GL_DRAW_BUFFER0_EXT && pname <= GL_DRAW_BUFFER15_EXT)
1059 {
1060 *type = GL_INT;
1061 *numParams = 1;
1062 return true;
1063 }
1064
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001065 // Please note: the query type returned for DEPTH_CLEAR_VALUE in this implementation
1066 // is FLOAT rather than INT, as would be suggested by the GL ES 2.0 spec. This is due
1067 // to the fact that it is stored internally as a float, and so would require conversion
Jamie Madill893ab082014-05-16 16:56:10 -04001068 // if returned from Context::getIntegerv. Since this conversion is already implemented
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001069 // in the case that one calls glGetIntegerv to retrieve a float-typed state variable, we
1070 // place DEPTH_CLEAR_VALUE with the floats. This should make no difference to the calling
1071 // application.
1072 switch (pname)
1073 {
1074 case GL_COMPRESSED_TEXTURE_FORMATS:
1075 {
1076 *type = GL_INT;
Cooper Partin4d61f7e2015-08-12 10:56:50 -07001077 *numParams = static_cast<unsigned int>(mCaps.compressedTextureFormats.size());
Geoff Lang900013c2014-07-07 11:32:19 -04001078 }
1079 return true;
1080 case GL_PROGRAM_BINARY_FORMATS_OES:
1081 {
1082 *type = GL_INT;
Cooper Partin4d61f7e2015-08-12 10:56:50 -07001083 *numParams = static_cast<unsigned int>(mCaps.programBinaryFormats.size());
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001084 }
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001085 return true;
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001086 case GL_SHADER_BINARY_FORMATS:
1087 {
1088 *type = GL_INT;
Cooper Partin4d61f7e2015-08-12 10:56:50 -07001089 *numParams = static_cast<unsigned int>(mCaps.shaderBinaryFormats.size());
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001090 }
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001091 return true;
Jamie Madillb9293972015-02-19 11:07:54 -05001092
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001093 case GL_MAX_VERTEX_ATTRIBS:
1094 case GL_MAX_VERTEX_UNIFORM_VECTORS:
1095 case GL_MAX_VARYING_VECTORS:
1096 case GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS:
1097 case GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS:
1098 case GL_MAX_TEXTURE_IMAGE_UNITS:
1099 case GL_MAX_FRAGMENT_UNIFORM_VECTORS:
1100 case GL_MAX_RENDERBUFFER_SIZE:
shannon.woods%transgaming.com@gtempaccount.com9790c472013-04-13 03:28:23 +00001101 case GL_MAX_COLOR_ATTACHMENTS_EXT:
shannon.woods%transgaming.com@gtempaccount.combc373e52013-04-13 03:31:23 +00001102 case GL_MAX_DRAW_BUFFERS_EXT:
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001103 case GL_NUM_SHADER_BINARY_FORMATS:
1104 case GL_NUM_COMPRESSED_TEXTURE_FORMATS:
1105 case GL_ARRAY_BUFFER_BINDING:
Vladimir Vukicevic1e514352014-05-13 15:53:06 -07001106 //case GL_FRAMEBUFFER_BINDING: // equivalent to DRAW_FRAMEBUFFER_BINDING_ANGLE
1107 case GL_DRAW_FRAMEBUFFER_BINDING_ANGLE:
1108 case GL_READ_FRAMEBUFFER_BINDING_ANGLE:
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001109 case GL_RENDERBUFFER_BINDING:
1110 case GL_CURRENT_PROGRAM:
1111 case GL_PACK_ALIGNMENT:
1112 case GL_PACK_REVERSE_ROW_ORDER_ANGLE:
1113 case GL_UNPACK_ALIGNMENT:
1114 case GL_GENERATE_MIPMAP_HINT:
1115 case GL_FRAGMENT_SHADER_DERIVATIVE_HINT_OES:
1116 case GL_RED_BITS:
1117 case GL_GREEN_BITS:
1118 case GL_BLUE_BITS:
1119 case GL_ALPHA_BITS:
1120 case GL_DEPTH_BITS:
1121 case GL_STENCIL_BITS:
1122 case GL_ELEMENT_ARRAY_BUFFER_BINDING:
1123 case GL_CULL_FACE_MODE:
1124 case GL_FRONT_FACE:
1125 case GL_ACTIVE_TEXTURE:
1126 case GL_STENCIL_FUNC:
1127 case GL_STENCIL_VALUE_MASK:
1128 case GL_STENCIL_REF:
1129 case GL_STENCIL_FAIL:
1130 case GL_STENCIL_PASS_DEPTH_FAIL:
1131 case GL_STENCIL_PASS_DEPTH_PASS:
1132 case GL_STENCIL_BACK_FUNC:
1133 case GL_STENCIL_BACK_VALUE_MASK:
1134 case GL_STENCIL_BACK_REF:
1135 case GL_STENCIL_BACK_FAIL:
1136 case GL_STENCIL_BACK_PASS_DEPTH_FAIL:
1137 case GL_STENCIL_BACK_PASS_DEPTH_PASS:
1138 case GL_DEPTH_FUNC:
1139 case GL_BLEND_SRC_RGB:
1140 case GL_BLEND_SRC_ALPHA:
1141 case GL_BLEND_DST_RGB:
1142 case GL_BLEND_DST_ALPHA:
1143 case GL_BLEND_EQUATION_RGB:
1144 case GL_BLEND_EQUATION_ALPHA:
1145 case GL_STENCIL_WRITEMASK:
1146 case GL_STENCIL_BACK_WRITEMASK:
1147 case GL_STENCIL_CLEAR_VALUE:
1148 case GL_SUBPIXEL_BITS:
1149 case GL_MAX_TEXTURE_SIZE:
1150 case GL_MAX_CUBE_MAP_TEXTURE_SIZE:
1151 case GL_SAMPLE_BUFFERS:
1152 case GL_SAMPLES:
1153 case GL_IMPLEMENTATION_COLOR_READ_TYPE:
1154 case GL_IMPLEMENTATION_COLOR_READ_FORMAT:
1155 case GL_TEXTURE_BINDING_2D:
1156 case GL_TEXTURE_BINDING_CUBE_MAP:
1157 case GL_RESET_NOTIFICATION_STRATEGY_EXT:
1158 case GL_NUM_PROGRAM_BINARY_FORMATS_OES:
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001159 {
1160 *type = GL_INT;
1161 *numParams = 1;
1162 }
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001163 return true;
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001164 case GL_MAX_SAMPLES_ANGLE:
1165 {
Geoff Langc0b9ef42014-07-02 10:02:37 -04001166 if (mExtensions.framebufferMultisample)
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001167 {
1168 *type = GL_INT;
1169 *numParams = 1;
1170 }
1171 else
1172 {
1173 return false;
1174 }
1175 }
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001176 return true;
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001177 case GL_MAX_VIEWPORT_DIMS:
1178 {
1179 *type = GL_INT;
1180 *numParams = 2;
1181 }
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001182 return true;
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001183 case GL_VIEWPORT:
1184 case GL_SCISSOR_BOX:
1185 {
1186 *type = GL_INT;
1187 *numParams = 4;
1188 }
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001189 return true;
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001190 case GL_SHADER_COMPILER:
1191 case GL_SAMPLE_COVERAGE_INVERT:
1192 case GL_DEPTH_WRITEMASK:
1193 case GL_CULL_FACE: // CULL_FACE through DITHER are natural to IsEnabled,
1194 case GL_POLYGON_OFFSET_FILL: // but can be retrieved through the Get{Type}v queries.
1195 case GL_SAMPLE_ALPHA_TO_COVERAGE: // For this purpose, they are treated here as bool-natural
1196 case GL_SAMPLE_COVERAGE:
1197 case GL_SCISSOR_TEST:
1198 case GL_STENCIL_TEST:
1199 case GL_DEPTH_TEST:
1200 case GL_BLEND:
1201 case GL_DITHER:
1202 case GL_CONTEXT_ROBUST_ACCESS_EXT:
1203 {
1204 *type = GL_BOOL;
1205 *numParams = 1;
1206 }
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001207 return true;
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001208 case GL_COLOR_WRITEMASK:
1209 {
1210 *type = GL_BOOL;
1211 *numParams = 4;
1212 }
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001213 return true;
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001214 case GL_POLYGON_OFFSET_FACTOR:
1215 case GL_POLYGON_OFFSET_UNITS:
1216 case GL_SAMPLE_COVERAGE_VALUE:
1217 case GL_DEPTH_CLEAR_VALUE:
1218 case GL_LINE_WIDTH:
1219 {
1220 *type = GL_FLOAT;
1221 *numParams = 1;
1222 }
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001223 return true;
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001224 case GL_ALIASED_LINE_WIDTH_RANGE:
1225 case GL_ALIASED_POINT_SIZE_RANGE:
1226 case GL_DEPTH_RANGE:
1227 {
1228 *type = GL_FLOAT;
1229 *numParams = 2;
1230 }
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001231 return true;
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001232 case GL_COLOR_CLEAR_VALUE:
1233 case GL_BLEND_COLOR:
1234 {
1235 *type = GL_FLOAT;
1236 *numParams = 4;
1237 }
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001238 return true;
daniel@transgaming.com07ab8412012-07-12 15:17:09 +00001239 case GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT:
Geoff Langc0b9ef42014-07-02 10:02:37 -04001240 if (!mExtensions.maxTextureAnisotropy)
daniel@transgaming.com07ab8412012-07-12 15:17:09 +00001241 {
1242 return false;
1243 }
1244 *type = GL_FLOAT;
1245 *numParams = 1;
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001246 return true;
1247 }
1248
Geoff Lang70d0f492015-12-10 17:45:46 -05001249 if (mExtensions.debug)
1250 {
1251 switch (pname)
1252 {
1253 case GL_DEBUG_LOGGED_MESSAGES:
1254 case GL_DEBUG_NEXT_LOGGED_MESSAGE_LENGTH:
1255 case GL_DEBUG_GROUP_STACK_DEPTH:
1256 case GL_MAX_DEBUG_MESSAGE_LENGTH:
1257 case GL_MAX_DEBUG_LOGGED_MESSAGES:
1258 case GL_MAX_DEBUG_GROUP_STACK_DEPTH:
1259 case GL_MAX_LABEL_LENGTH:
1260 *type = GL_INT;
1261 *numParams = 1;
1262 return true;
1263
1264 case GL_DEBUG_OUTPUT_SYNCHRONOUS:
1265 case GL_DEBUG_OUTPUT:
1266 *type = GL_BOOL;
1267 *numParams = 1;
1268 return true;
1269 }
1270 }
1271
Austin Kinrossbc781f32015-10-26 09:27:38 -07001272 // Check for ES3.0+ parameter names which are also exposed as ES2 extensions
1273 switch (pname)
1274 {
1275 case GL_PACK_ROW_LENGTH:
1276 case GL_PACK_SKIP_ROWS:
1277 case GL_PACK_SKIP_PIXELS:
1278 if ((mClientVersion < 3) && !mExtensions.packSubimage)
1279 {
1280 return false;
1281 }
1282 *type = GL_INT;
1283 *numParams = 1;
1284 return true;
1285 case GL_UNPACK_ROW_LENGTH:
1286 case GL_UNPACK_SKIP_ROWS:
1287 case GL_UNPACK_SKIP_PIXELS:
1288 if ((mClientVersion < 3) && !mExtensions.unpackSubimage)
1289 {
1290 return false;
1291 }
1292 *type = GL_INT;
1293 *numParams = 1;
1294 return true;
1295 case GL_VERTEX_ARRAY_BINDING:
1296 if ((mClientVersion < 3) && !mExtensions.vertexArrayObject)
1297 {
1298 return false;
1299 }
1300 *type = GL_INT;
1301 *numParams = 1;
1302 return true;
Yuly Novikov5807a532015-12-03 13:01:22 -05001303 case GL_PIXEL_PACK_BUFFER_BINDING:
1304 case GL_PIXEL_UNPACK_BUFFER_BINDING:
1305 if ((mClientVersion < 3) && !mExtensions.pixelBufferObject)
1306 {
1307 return false;
1308 }
1309 *type = GL_INT;
1310 *numParams = 1;
1311 return true;
Austin Kinrossbc781f32015-10-26 09:27:38 -07001312 }
1313
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001314 if (mClientVersion < 3)
1315 {
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001316 return false;
1317 }
1318
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001319 // Check for ES3.0+ parameter names
1320 switch (pname)
1321 {
shannonwoods@chromium.org97c3d502013-05-30 00:04:34 +00001322 case GL_MAX_UNIFORM_BUFFER_BINDINGS:
1323 case GL_UNIFORM_BUFFER_OFFSET_ALIGNMENT:
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001324 case GL_UNIFORM_BUFFER_BINDING:
1325 case GL_TRANSFORM_FEEDBACK_BINDING:
Geoff Lang045536b2015-03-27 15:17:18 -04001326 case GL_TRANSFORM_FEEDBACK_BUFFER_BINDING:
shannon.woods%transgaming.com@gtempaccount.com51171882013-04-13 03:39:10 +00001327 case GL_COPY_READ_BUFFER_BINDING:
1328 case GL_COPY_WRITE_BUFFER_BINDING:
shannon.woods%transgaming.com@gtempaccount.comc416e1c2013-04-13 03:45:05 +00001329 case GL_TEXTURE_BINDING_3D:
shannon.woods%transgaming.com@gtempaccount.com90dbc442013-04-13 03:46:14 +00001330 case GL_TEXTURE_BINDING_2D_ARRAY:
shannon.woods%transgaming.com@gtempaccount.comc1fdf6b2013-04-13 03:44:41 +00001331 case GL_MAX_3D_TEXTURE_SIZE:
shannon.woods%transgaming.com@gtempaccount.coma98a8112013-04-13 03:45:57 +00001332 case GL_MAX_ARRAY_TEXTURE_LAYERS:
shannonwoods@chromium.orgf2d76f82013-05-30 00:06:32 +00001333 case GL_MAX_VERTEX_UNIFORM_BLOCKS:
1334 case GL_MAX_FRAGMENT_UNIFORM_BLOCKS:
1335 case GL_MAX_COMBINED_UNIFORM_BLOCKS:
Geoff Lange6d4e122015-06-29 13:33:55 -04001336 case GL_MAX_VERTEX_OUTPUT_COMPONENTS:
1337 case GL_MAX_FRAGMENT_INPUT_COMPONENTS:
Geoff Langd3ff9002014-05-08 11:19:27 -04001338 case GL_MAX_VARYING_COMPONENTS:
Jamie Madill38850df2013-07-19 16:36:55 -04001339 case GL_MAX_VERTEX_UNIFORM_COMPONENTS:
1340 case GL_MAX_FRAGMENT_UNIFORM_COMPONENTS:
Geoff Lange6d4e122015-06-29 13:33:55 -04001341 case GL_MIN_PROGRAM_TEXEL_OFFSET:
1342 case GL_MAX_PROGRAM_TEXEL_OFFSET:
Geoff Lang23c81692013-08-12 10:46:58 -04001343 case GL_NUM_EXTENSIONS:
Jamie Madillee7010d2013-10-17 10:45:47 -04001344 case GL_MAJOR_VERSION:
1345 case GL_MINOR_VERSION:
Jamie Madill13a2f852013-12-11 16:35:08 -05001346 case GL_MAX_ELEMENTS_INDICES:
1347 case GL_MAX_ELEMENTS_VERTICES:
Geoff Lang1b6edcb2014-02-03 14:27:56 -05001348 case GL_MAX_TRANSFORM_FEEDBACK_INTERLEAVED_COMPONENTS:
Jamie Madill2e503552013-12-19 13:48:34 -05001349 case GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_ATTRIBS:
Geoff Lang1b6edcb2014-02-03 14:27:56 -05001350 case GL_MAX_TRANSFORM_FEEDBACK_SEPARATE_COMPONENTS:
Minmin Gongadff67b2015-10-14 10:34:45 -04001351 case GL_UNPACK_IMAGE_HEIGHT:
Jamie Madill023a2902015-10-23 16:43:24 +00001352 case GL_UNPACK_SKIP_IMAGES:
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001353 {
1354 *type = GL_INT;
1355 *numParams = 1;
1356 }
1357 return true;
Jamie Madill0fda9862013-07-19 16:36:55 -04001358
1359 case GL_MAX_ELEMENT_INDEX:
1360 case GL_MAX_UNIFORM_BLOCK_SIZE:
1361 case GL_MAX_COMBINED_VERTEX_UNIFORM_COMPONENTS:
1362 case GL_MAX_COMBINED_FRAGMENT_UNIFORM_COMPONENTS:
1363 case GL_MAX_SERVER_WAIT_TIMEOUT:
1364 {
1365 *type = GL_INT_64_ANGLEX;
1366 *numParams = 1;
1367 }
1368 return true;
Jamie Madill2e503552013-12-19 13:48:34 -05001369
1370 case GL_TRANSFORM_FEEDBACK_ACTIVE:
Geoff Lang1b6edcb2014-02-03 14:27:56 -05001371 case GL_TRANSFORM_FEEDBACK_PAUSED:
Jamie Madille2cd53d2015-10-27 11:15:46 -04001372 case GL_PRIMITIVE_RESTART_FIXED_INDEX:
Geoff Langab831f02015-12-01 09:39:10 -05001373 case GL_RASTERIZER_DISCARD:
Jamie Madill2e503552013-12-19 13:48:34 -05001374 {
1375 *type = GL_BOOL;
1376 *numParams = 1;
1377 }
1378 return true;
Geoff Lange6d4e122015-06-29 13:33:55 -04001379
1380 case GL_MAX_TEXTURE_LOD_BIAS:
1381 {
1382 *type = GL_FLOAT;
1383 *numParams = 1;
1384 }
1385 return true;
shannon.woods%transgaming.com@gtempaccount.com667a29c2013-04-13 03:39:04 +00001386 }
1387
1388 return false;
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001389}
1390
Shannon Woods1b2fb852013-08-19 14:28:48 -04001391bool Context::getIndexedQueryParameterInfo(GLenum target, GLenum *type, unsigned int *numParams)
1392{
1393 if (mClientVersion < 3)
1394 {
1395 return false;
1396 }
1397
1398 switch (target)
1399 {
1400 case GL_TRANSFORM_FEEDBACK_BUFFER_BINDING:
1401 case GL_UNIFORM_BUFFER_BINDING:
1402 {
1403 *type = GL_INT;
1404 *numParams = 1;
1405 }
1406 return true;
1407 case GL_TRANSFORM_FEEDBACK_BUFFER_START:
1408 case GL_TRANSFORM_FEEDBACK_BUFFER_SIZE:
1409 case GL_UNIFORM_BUFFER_START:
1410 case GL_UNIFORM_BUFFER_SIZE:
1411 {
1412 *type = GL_INT_64_ANGLEX;
1413 *numParams = 1;
1414 }
1415 }
1416
1417 return false;
1418}
1419
Geoff Langf6db0982015-08-25 13:04:00 -04001420Error Context::drawArrays(GLenum mode, GLint first, GLsizei count)
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001421{
Jamie Madill1b94d432015-08-07 13:23:23 -04001422 syncRendererState();
Geoff Langf6db0982015-08-25 13:04:00 -04001423 Error error = mRenderer->drawArrays(getData(), mode, first, count);
Geoff Lang520c4ae2015-05-05 13:12:36 -04001424 if (error.isError())
1425 {
1426 return error;
1427 }
1428
Geoff Langf6db0982015-08-25 13:04:00 -04001429 MarkTransformFeedbackBufferUsage(mState.getCurrentTransformFeedback());
Geoff Lang520c4ae2015-05-05 13:12:36 -04001430
1431 return Error(GL_NO_ERROR);
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001432}
1433
Geoff Langf6db0982015-08-25 13:04:00 -04001434Error Context::drawArraysInstanced(GLenum mode, GLint first, GLsizei count, GLsizei instanceCount)
1435{
1436 syncRendererState();
1437 Error error = mRenderer->drawArraysInstanced(getData(), mode, first, count, instanceCount);
1438 if (error.isError())
1439 {
1440 return error;
1441 }
1442
1443 MarkTransformFeedbackBufferUsage(mState.getCurrentTransformFeedback());
1444
1445 return Error(GL_NO_ERROR);
1446}
1447
1448Error Context::drawElements(GLenum mode,
1449 GLsizei count,
1450 GLenum type,
1451 const GLvoid *indices,
Geoff Lang3edfe032015-09-04 16:38:24 -04001452 const IndexRange &indexRange)
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001453{
Jamie Madill1b94d432015-08-07 13:23:23 -04001454 syncRendererState();
Geoff Langf6db0982015-08-25 13:04:00 -04001455 return mRenderer->drawElements(getData(), mode, count, type, indices, indexRange);
1456}
1457
1458Error Context::drawElementsInstanced(GLenum mode,
1459 GLsizei count,
1460 GLenum type,
1461 const GLvoid *indices,
1462 GLsizei instances,
Geoff Lang3edfe032015-09-04 16:38:24 -04001463 const IndexRange &indexRange)
Geoff Langf6db0982015-08-25 13:04:00 -04001464{
1465 syncRendererState();
1466 return mRenderer->drawElementsInstanced(getData(), mode, count, type, indices, instances,
1467 indexRange);
1468}
1469
1470Error Context::drawRangeElements(GLenum mode,
1471 GLuint start,
1472 GLuint end,
1473 GLsizei count,
1474 GLenum type,
1475 const GLvoid *indices,
Geoff Lang3edfe032015-09-04 16:38:24 -04001476 const IndexRange &indexRange)
Geoff Langf6db0982015-08-25 13:04:00 -04001477{
1478 syncRendererState();
1479 return mRenderer->drawRangeElements(getData(), mode, start, end, count, type, indices,
1480 indexRange);
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001481}
1482
Geoff Lang129753a2015-01-09 16:52:09 -05001483Error Context::flush()
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001484{
Geoff Lang129753a2015-01-09 16:52:09 -05001485 return mRenderer->flush();
1486}
1487
1488Error Context::finish()
1489{
1490 return mRenderer->finish();
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001491}
1492
Austin Kinross6ee1e782015-05-29 17:05:37 -07001493void Context::insertEventMarker(GLsizei length, const char *marker)
1494{
1495 ASSERT(mRenderer);
1496 mRenderer->insertEventMarker(length, marker);
1497}
1498
1499void Context::pushGroupMarker(GLsizei length, const char *marker)
1500{
1501 ASSERT(mRenderer);
1502 mRenderer->pushGroupMarker(length, marker);
1503}
1504
1505void Context::popGroupMarker()
1506{
1507 ASSERT(mRenderer);
1508 mRenderer->popGroupMarker();
1509}
1510
Geoff Langda5777c2014-07-11 09:52:58 -04001511void Context::recordError(const Error &error)
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001512{
Geoff Langda5777c2014-07-11 09:52:58 -04001513 if (error.isError())
1514 {
1515 mErrors.insert(error.getCode());
Geoff Lang70d0f492015-12-10 17:45:46 -05001516
1517 if (!error.getMessage().empty())
1518 {
1519 auto &debug = mState.getDebug();
1520 debug.insertMessage(GL_DEBUG_SOURCE_API, GL_DEBUG_TYPE_ERROR, error.getID(),
1521 GL_DEBUG_SEVERITY_HIGH, error.getMessage());
1522 }
Geoff Langda5777c2014-07-11 09:52:58 -04001523 }
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001524}
1525
1526// Get one of the recorded errors and clear its flag, if any.
1527// [OpenGL ES 2.0.24] section 2.5 page 13.
1528GLenum Context::getError()
1529{
Geoff Langda5777c2014-07-11 09:52:58 -04001530 if (mErrors.empty())
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001531 {
Geoff Langda5777c2014-07-11 09:52:58 -04001532 return GL_NO_ERROR;
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001533 }
Geoff Langda5777c2014-07-11 09:52:58 -04001534 else
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001535 {
Geoff Langda5777c2014-07-11 09:52:58 -04001536 GLenum error = *mErrors.begin();
1537 mErrors.erase(mErrors.begin());
1538 return error;
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001539 }
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001540}
1541
1542GLenum Context::getResetStatus()
1543{
Jamie Madill93e13fb2014-11-06 15:27:25 -05001544 //TODO(jmadill): needs MANGLE reworking
shannon.woods@transgaming.comddd6c802013-02-28 23:05:14 +00001545 if (mResetStatus == GL_NO_ERROR && !mContextLost)
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001546 {
daniel@transgaming.comf688c0d2012-10-31 17:52:57 +00001547 // mResetStatus will be set by the markContextLost callback
1548 // in the case a notification is sent
Jamie Madill4c76fea2014-11-24 11:38:52 -05001549 if (mRenderer->testDeviceLost())
Jamie Madill9dd0cf02014-11-24 11:38:51 -05001550 {
1551 mRenderer->notifyDeviceLost();
1552 }
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001553 }
1554
1555 GLenum status = mResetStatus;
1556
1557 if (mResetStatus != GL_NO_ERROR)
1558 {
shannon.woods@transgaming.comddd6c802013-02-28 23:05:14 +00001559 ASSERT(mContextLost);
1560
daniel@transgaming.com621ce052012-10-31 17:52:29 +00001561 if (mRenderer->testDeviceResettable())
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001562 {
1563 mResetStatus = GL_NO_ERROR;
1564 }
1565 }
Jamie Madill893ab082014-05-16 16:56:10 -04001566
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001567 return status;
1568}
1569
1570bool Context::isResetNotificationEnabled()
1571{
1572 return (mResetStrategy == GL_LOSE_CONTEXT_ON_RESET_EXT);
1573}
1574
Corentin Walleze3b10e82015-05-20 11:06:25 -04001575const egl::Config *Context::getConfig() const
Régis Fénéon83107972015-02-05 12:57:44 +01001576{
Corentin Walleze3b10e82015-05-20 11:06:25 -04001577 return mConfig;
Régis Fénéon83107972015-02-05 12:57:44 +01001578}
1579
1580EGLenum Context::getClientType() const
1581{
1582 return mClientType;
1583}
1584
1585EGLenum Context::getRenderBuffer() const
1586{
Corentin Wallez37c39792015-08-20 14:19:46 -04001587 auto framebufferIt = mFramebufferMap.find(0);
1588 if (framebufferIt != mFramebufferMap.end())
1589 {
1590 const Framebuffer *framebuffer = framebufferIt->second;
1591 const FramebufferAttachment *backAttachment = framebuffer->getAttachment(GL_BACK);
1592
1593 ASSERT(backAttachment != nullptr);
1594 return backAttachment->getSurface()->getRenderBuffer();
1595 }
1596 else
1597 {
1598 return EGL_NONE;
1599 }
Régis Fénéon83107972015-02-05 12:57:44 +01001600}
1601
Geoff Lang36167ab2015-12-07 10:27:14 -05001602void Context::checkVertexArrayAllocation(GLuint vertexArray)
1603{
1604 if (!getVertexArray(vertexArray))
1605 {
1606 VertexArray *vertexArrayObject =
1607 new VertexArray(mRenderer, vertexArray, MAX_VERTEX_ATTRIBS);
1608 mVertexArrayMap[vertexArray] = vertexArrayObject;
1609 }
1610}
1611
1612void Context::checkTransformFeedbackAllocation(GLuint transformFeedback)
1613{
1614 if (!getTransformFeedback(transformFeedback))
1615 {
1616 TransformFeedback *transformFeedbackObject =
1617 new TransformFeedback(mRenderer->createTransformFeedback(), transformFeedback, mCaps);
1618 transformFeedbackObject->addRef();
1619 mTransformFeedbackMap[transformFeedback] = transformFeedbackObject;
1620 }
1621}
1622
1623bool Context::isVertexArrayGenerated(GLuint vertexArray)
1624{
1625 return mVertexArrayMap.find(vertexArray) != mVertexArrayMap.end();
1626}
1627
1628bool Context::isTransformFeedbackGenerated(GLuint transformFeedback)
1629{
1630 return mTransformFeedbackMap.find(transformFeedback) != mTransformFeedbackMap.end();
1631}
1632
Shannon Woods53a94a82014-06-24 15:20:36 -04001633void Context::detachTexture(GLuint texture)
1634{
1635 // Simple pass-through to State's detachTexture method, as textures do not require
1636 // allocation map management either here or in the resource manager at detach time.
1637 // Zero textures are held by the Context, and we don't attempt to request them from
1638 // the State.
Jamie Madille6382c32014-11-07 15:05:26 -05001639 mState.detachTexture(mZeroTextures, texture);
Shannon Woods53a94a82014-06-24 15:20:36 -04001640}
1641
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001642void Context::detachBuffer(GLuint buffer)
1643{
Yuly Novikov5807a532015-12-03 13:01:22 -05001644 // Simple pass-through to State's detachBuffer method, since
1645 // only buffer attachments to container objects that are bound to the current context
1646 // should be detached. And all those are available in State.
Shannon Woods53a94a82014-06-24 15:20:36 -04001647
Yuly Novikov5807a532015-12-03 13:01:22 -05001648 // [OpenGL ES 3.2] section 5.1.2 page 45:
1649 // Attachments to unbound container objects, such as
1650 // deletion of a buffer attached to a vertex array object which is not bound to the context,
1651 // are not affected and continue to act as references on the deleted object
1652 mState.detachBuffer(buffer);
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001653}
1654
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001655void Context::detachFramebuffer(GLuint framebuffer)
1656{
Shannon Woods53a94a82014-06-24 15:20:36 -04001657 // Framebuffer detachment is handled by Context, because 0 is a valid
1658 // Framebuffer object, and a pointer to it must be passed from Context
1659 // to State at binding time.
1660
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001661 // [OpenGL ES 2.0.24] section 4.4 page 107:
1662 // If a framebuffer that is currently bound to the target FRAMEBUFFER is deleted, it is as though
1663 // BindFramebuffer had been executed with the target of FRAMEBUFFER and framebuffer of zero.
1664
Gregoire Payen de La Garanderieed54e5d2015-03-17 16:51:24 +00001665 if (mState.removeReadFramebufferBinding(framebuffer) && framebuffer != 0)
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001666 {
1667 bindReadFramebuffer(0);
1668 }
1669
Gregoire Payen de La Garanderieed54e5d2015-03-17 16:51:24 +00001670 if (mState.removeDrawFramebufferBinding(framebuffer) && framebuffer != 0)
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001671 {
1672 bindDrawFramebuffer(0);
1673 }
1674}
1675
1676void Context::detachRenderbuffer(GLuint renderbuffer)
1677{
Shannon Woods53a94a82014-06-24 15:20:36 -04001678 mState.detachRenderbuffer(renderbuffer);
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001679}
1680
Jamie Madill57a89722013-07-02 11:57:03 -04001681void Context::detachVertexArray(GLuint vertexArray)
1682{
Jamie Madill77a72f62015-04-14 11:18:32 -04001683 // Vertex array detachment is handled by Context, because 0 is a valid
1684 // VAO, and a pointer to it must be passed from Context to State at
Shannon Woods53a94a82014-06-24 15:20:36 -04001685 // binding time.
1686
Jamie Madill57a89722013-07-02 11:57:03 -04001687 // [OpenGL ES 3.0.2] section 2.10 page 43:
1688 // If a vertex array object that is currently bound is deleted, the binding
1689 // for that object reverts to zero and the default vertex array becomes current.
Shannon Woods53a94a82014-06-24 15:20:36 -04001690 if (mState.removeVertexArrayBinding(vertexArray))
Jamie Madill57a89722013-07-02 11:57:03 -04001691 {
1692 bindVertexArray(0);
1693 }
1694}
1695
Geoff Langc8058452014-02-03 12:04:11 -05001696void Context::detachTransformFeedback(GLuint transformFeedback)
1697{
Shannon Woods53a94a82014-06-24 15:20:36 -04001698 mState.detachTransformFeedback(transformFeedback);
Geoff Langc8058452014-02-03 12:04:11 -05001699}
1700
Jamie Madilldc356042013-07-19 16:36:57 -04001701void Context::detachSampler(GLuint sampler)
1702{
Shannon Woods53a94a82014-06-24 15:20:36 -04001703 mState.detachSampler(sampler);
Jamie Madilldc356042013-07-19 16:36:57 -04001704}
1705
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001706void Context::setVertexAttribDivisor(GLuint index, GLuint divisor)
1707{
Jamie Madill0b9e9032015-08-17 11:51:52 +00001708 mState.setVertexAttribDivisor(index, divisor);
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001709}
1710
Jamie Madille29d1672013-07-19 16:36:57 -04001711void Context::samplerParameteri(GLuint sampler, GLenum pname, GLint param)
1712{
1713 mResourceManager->checkSamplerAllocation(sampler);
1714
1715 Sampler *samplerObject = getSampler(sampler);
1716 ASSERT(samplerObject);
1717
Geoff Lang69cce582015-09-17 13:20:36 -04001718 // clang-format off
Jamie Madille29d1672013-07-19 16:36:57 -04001719 switch (pname)
1720 {
Geoff Lang69cce582015-09-17 13:20:36 -04001721 case GL_TEXTURE_MIN_FILTER: samplerObject->setMinFilter(static_cast<GLenum>(param)); break;
1722 case GL_TEXTURE_MAG_FILTER: samplerObject->setMagFilter(static_cast<GLenum>(param)); break;
1723 case GL_TEXTURE_WRAP_S: samplerObject->setWrapS(static_cast<GLenum>(param)); break;
1724 case GL_TEXTURE_WRAP_T: samplerObject->setWrapT(static_cast<GLenum>(param)); break;
1725 case GL_TEXTURE_WRAP_R: samplerObject->setWrapR(static_cast<GLenum>(param)); break;
1726 case GL_TEXTURE_MAX_ANISOTROPY_EXT: samplerObject->setMaxAnisotropy(std::min(static_cast<GLfloat>(param), getExtensions().maxTextureAnisotropy)); break;
1727 case GL_TEXTURE_MIN_LOD: samplerObject->setMinLod(static_cast<GLfloat>(param)); break;
1728 case GL_TEXTURE_MAX_LOD: samplerObject->setMaxLod(static_cast<GLfloat>(param)); break;
1729 case GL_TEXTURE_COMPARE_MODE: samplerObject->setCompareMode(static_cast<GLenum>(param)); break;
1730 case GL_TEXTURE_COMPARE_FUNC: samplerObject->setCompareFunc(static_cast<GLenum>(param)); break;
1731 default: UNREACHABLE(); break;
Jamie Madille29d1672013-07-19 16:36:57 -04001732 }
Geoff Lang69cce582015-09-17 13:20:36 -04001733 // clang-format on
Jamie Madille29d1672013-07-19 16:36:57 -04001734}
1735
1736void Context::samplerParameterf(GLuint sampler, GLenum pname, GLfloat param)
1737{
1738 mResourceManager->checkSamplerAllocation(sampler);
1739
1740 Sampler *samplerObject = getSampler(sampler);
1741 ASSERT(samplerObject);
1742
Geoff Lang69cce582015-09-17 13:20:36 -04001743 // clang-format off
Jamie Madille29d1672013-07-19 16:36:57 -04001744 switch (pname)
1745 {
Geoff Lang69cce582015-09-17 13:20:36 -04001746 case GL_TEXTURE_MIN_FILTER: samplerObject->setMinFilter(uiround<GLenum>(param)); break;
1747 case GL_TEXTURE_MAG_FILTER: samplerObject->setMagFilter(uiround<GLenum>(param)); break;
1748 case GL_TEXTURE_WRAP_S: samplerObject->setWrapS(uiround<GLenum>(param)); break;
1749 case GL_TEXTURE_WRAP_T: samplerObject->setWrapT(uiround<GLenum>(param)); break;
1750 case GL_TEXTURE_WRAP_R: samplerObject->setWrapR(uiround<GLenum>(param)); break;
1751 case GL_TEXTURE_MAX_ANISOTROPY_EXT: samplerObject->setMaxAnisotropy(std::min(param, getExtensions().maxTextureAnisotropy)); break;
1752 case GL_TEXTURE_MIN_LOD: samplerObject->setMinLod(param); break;
1753 case GL_TEXTURE_MAX_LOD: samplerObject->setMaxLod(param); break;
1754 case GL_TEXTURE_COMPARE_MODE: samplerObject->setCompareMode(uiround<GLenum>(param)); break;
1755 case GL_TEXTURE_COMPARE_FUNC: samplerObject->setCompareFunc(uiround<GLenum>(param)); break;
1756 default: UNREACHABLE(); break;
Jamie Madille29d1672013-07-19 16:36:57 -04001757 }
Geoff Lang69cce582015-09-17 13:20:36 -04001758 // clang-format on
Jamie Madille29d1672013-07-19 16:36:57 -04001759}
1760
Jamie Madill9675b802013-07-19 16:36:59 -04001761GLint Context::getSamplerParameteri(GLuint sampler, GLenum pname)
1762{
1763 mResourceManager->checkSamplerAllocation(sampler);
1764
1765 Sampler *samplerObject = getSampler(sampler);
1766 ASSERT(samplerObject);
1767
Geoff Lang69cce582015-09-17 13:20:36 -04001768 // clang-format off
Jamie Madill9675b802013-07-19 16:36:59 -04001769 switch (pname)
1770 {
Geoff Lang69cce582015-09-17 13:20:36 -04001771 case GL_TEXTURE_MIN_FILTER: return static_cast<GLint>(samplerObject->getMinFilter());
1772 case GL_TEXTURE_MAG_FILTER: return static_cast<GLint>(samplerObject->getMagFilter());
1773 case GL_TEXTURE_WRAP_S: return static_cast<GLint>(samplerObject->getWrapS());
1774 case GL_TEXTURE_WRAP_T: return static_cast<GLint>(samplerObject->getWrapT());
1775 case GL_TEXTURE_WRAP_R: return static_cast<GLint>(samplerObject->getWrapR());
1776 case GL_TEXTURE_MAX_ANISOTROPY_EXT: return static_cast<GLint>(samplerObject->getMaxAnisotropy());
1777 case GL_TEXTURE_MIN_LOD: return uiround<GLint>(samplerObject->getMinLod());
1778 case GL_TEXTURE_MAX_LOD: return uiround<GLint>(samplerObject->getMaxLod());
1779 case GL_TEXTURE_COMPARE_MODE: return static_cast<GLint>(samplerObject->getCompareMode());
1780 case GL_TEXTURE_COMPARE_FUNC: return static_cast<GLint>(samplerObject->getCompareFunc());
1781 default: UNREACHABLE(); return 0;
Jamie Madill9675b802013-07-19 16:36:59 -04001782 }
Geoff Lang69cce582015-09-17 13:20:36 -04001783 // clang-format on
Jamie Madill9675b802013-07-19 16:36:59 -04001784}
1785
1786GLfloat Context::getSamplerParameterf(GLuint sampler, GLenum pname)
1787{
1788 mResourceManager->checkSamplerAllocation(sampler);
1789
1790 Sampler *samplerObject = getSampler(sampler);
1791 ASSERT(samplerObject);
1792
Geoff Lang69cce582015-09-17 13:20:36 -04001793 // clang-format off
Jamie Madill9675b802013-07-19 16:36:59 -04001794 switch (pname)
1795 {
Geoff Lang69cce582015-09-17 13:20:36 -04001796 case GL_TEXTURE_MIN_FILTER: return static_cast<GLfloat>(samplerObject->getMinFilter());
1797 case GL_TEXTURE_MAG_FILTER: return static_cast<GLfloat>(samplerObject->getMagFilter());
1798 case GL_TEXTURE_WRAP_S: return static_cast<GLfloat>(samplerObject->getWrapS());
1799 case GL_TEXTURE_WRAP_T: return static_cast<GLfloat>(samplerObject->getWrapT());
1800 case GL_TEXTURE_WRAP_R: return static_cast<GLfloat>(samplerObject->getWrapR());
1801 case GL_TEXTURE_MAX_ANISOTROPY_EXT: return samplerObject->getMaxAnisotropy();
1802 case GL_TEXTURE_MIN_LOD: return samplerObject->getMinLod();
1803 case GL_TEXTURE_MAX_LOD: return samplerObject->getMaxLod();
1804 case GL_TEXTURE_COMPARE_MODE: return static_cast<GLfloat>(samplerObject->getCompareMode());
1805 case GL_TEXTURE_COMPARE_FUNC: return static_cast<GLfloat>(samplerObject->getCompareFunc());
1806 default: UNREACHABLE(); return 0;
Jamie Madill9675b802013-07-19 16:36:59 -04001807 }
Geoff Lang69cce582015-09-17 13:20:36 -04001808 // clang-format on
Jamie Madill9675b802013-07-19 16:36:59 -04001809}
1810
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001811void Context::initRendererString()
1812{
daniel@transgaming.comca1ac1f2013-01-11 04:13:05 +00001813 std::ostringstream rendererString;
1814 rendererString << "ANGLE (";
1815 rendererString << mRenderer->getRendererDescription();
1816 rendererString << ")";
1817
Geoff Langcec35902014-04-16 10:52:36 -04001818 mRendererString = MakeStaticString(rendererString.str());
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001819}
1820
Geoff Langc0b9ef42014-07-02 10:02:37 -04001821const std::string &Context::getRendererString() const
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001822{
daniel@transgaming.comca1ac1f2013-01-11 04:13:05 +00001823 return mRendererString;
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001824}
1825
Geoff Langcec35902014-04-16 10:52:36 -04001826void Context::initExtensionStrings()
1827{
Geoff Lang493daf52014-07-03 13:38:44 -04001828 mExtensionStrings = mExtensions.getStrings();
Geoff Langcec35902014-04-16 10:52:36 -04001829
Geoff Langc0b9ef42014-07-02 10:02:37 -04001830 std::ostringstream combinedStringStream;
1831 std::copy(mExtensionStrings.begin(), mExtensionStrings.end(), std::ostream_iterator<std::string>(combinedStringStream, " "));
1832 mExtensionString = combinedStringStream.str();
Geoff Langcec35902014-04-16 10:52:36 -04001833}
1834
Geoff Langc0b9ef42014-07-02 10:02:37 -04001835const std::string &Context::getExtensionString() const
Geoff Langcec35902014-04-16 10:52:36 -04001836{
1837 return mExtensionString;
1838}
1839
Geoff Langc0b9ef42014-07-02 10:02:37 -04001840const std::string &Context::getExtensionString(size_t idx) const
Geoff Langcec35902014-04-16 10:52:36 -04001841{
1842 return mExtensionStrings[idx];
1843}
1844
1845size_t Context::getExtensionStringCount() const
1846{
1847 return mExtensionStrings.size();
1848}
1849
Geoff Lang493daf52014-07-03 13:38:44 -04001850void Context::initCaps(GLuint clientVersion)
1851{
1852 mCaps = mRenderer->getRendererCaps();
1853
1854 mExtensions = mRenderer->getRendererExtensions();
1855
Austin Kinross02df7962015-07-01 10:03:42 -07001856 mLimitations = mRenderer->getRendererLimitations();
1857
Geoff Lang493daf52014-07-03 13:38:44 -04001858 if (clientVersion < 3)
1859 {
1860 // Disable ES3+ extensions
1861 mExtensions.colorBufferFloat = false;
1862 }
1863
1864 if (clientVersion > 2)
1865 {
1866 // FIXME(geofflang): Don't support EXT_sRGB in non-ES2 contexts
1867 //mExtensions.sRGB = false;
1868 }
1869
Geoff Lang70d0f492015-12-10 17:45:46 -05001870 // Explicitly enable GL_KHR_debug
1871 mExtensions.debug = true;
1872 mExtensions.maxDebugMessageLength = 1024;
1873 mExtensions.maxDebugLoggedMessages = 1024;
1874 mExtensions.maxDebugGroupStackDepth = 1024;
1875 mExtensions.maxLabelLength = 1024;
1876
Geoff Lang301d1612014-07-09 10:34:37 -04001877 // Apply implementation limits
1878 mCaps.maxVertexAttributes = std::min<GLuint>(mCaps.maxVertexAttributes, MAX_VERTEX_ATTRIBS);
Geoff Lang301d1612014-07-09 10:34:37 -04001879 mCaps.maxVertexUniformBlocks = std::min<GLuint>(mCaps.maxVertexUniformBlocks, IMPLEMENTATION_MAX_VERTEX_SHADER_UNIFORM_BUFFERS);
1880 mCaps.maxVertexOutputComponents = std::min<GLuint>(mCaps.maxVertexOutputComponents, IMPLEMENTATION_MAX_VARYING_VECTORS * 4);
1881
1882 mCaps.maxFragmentInputComponents = std::min<GLuint>(mCaps.maxFragmentInputComponents, IMPLEMENTATION_MAX_VARYING_VECTORS * 4);
Geoff Lang3a61c322014-07-10 13:01:54 -04001883
Geoff Lang900013c2014-07-07 11:32:19 -04001884 mCaps.compressedTextureFormats.clear();
1885
Geoff Lang493daf52014-07-03 13:38:44 -04001886 const TextureCapsMap &rendererFormats = mRenderer->getRendererTextureCaps();
1887 for (TextureCapsMap::const_iterator i = rendererFormats.begin(); i != rendererFormats.end(); i++)
1888 {
1889 GLenum format = i->first;
1890 TextureCaps formatCaps = i->second;
1891
Geoff Lang5d601382014-07-22 15:14:06 -04001892 const InternalFormat &formatInfo = GetInternalFormatInfo(format);
Geoff Langd87878e2014-09-19 15:42:59 -04001893
Geoff Lang0d8b7242015-09-09 14:56:53 -04001894 // Update the format caps based on the client version and extensions.
1895 // Caps are AND'd with the renderer caps because some core formats are still unsupported in
1896 // ES3.
1897 formatCaps.texturable =
1898 formatCaps.texturable && formatInfo.textureSupport(clientVersion, mExtensions);
1899 formatCaps.renderable =
1900 formatCaps.renderable && formatInfo.renderSupport(clientVersion, mExtensions);
1901 formatCaps.filterable =
1902 formatCaps.filterable && formatInfo.filterSupport(clientVersion, mExtensions);
Geoff Langd87878e2014-09-19 15:42:59 -04001903
1904 // OpenGL ES does not support multisampling with integer formats
1905 if (!formatInfo.renderSupport || formatInfo.componentType == GL_INT || formatInfo.componentType == GL_UNSIGNED_INT)
Geoff Lang493daf52014-07-03 13:38:44 -04001906 {
Geoff Langd87878e2014-09-19 15:42:59 -04001907 formatCaps.sampleCounts.clear();
Geoff Lang493daf52014-07-03 13:38:44 -04001908 }
Geoff Langd87878e2014-09-19 15:42:59 -04001909
1910 if (formatCaps.texturable && formatInfo.compressed)
1911 {
1912 mCaps.compressedTextureFormats.push_back(format);
1913 }
1914
1915 mTextureCaps.insert(format, formatCaps);
Geoff Lang493daf52014-07-03 13:38:44 -04001916 }
1917}
1918
Jamie Madill1b94d432015-08-07 13:23:23 -04001919void Context::syncRendererState()
1920{
1921 const State::DirtyBits &dirtyBits = mState.getDirtyBits();
1922 if (dirtyBits.any())
1923 {
Jamie Madill63e4e722015-11-06 19:15:11 +00001924 mRenderer->syncState(mState, dirtyBits);
Jamie Madill1b94d432015-08-07 13:23:23 -04001925 mState.clearDirtyBits();
1926 }
1927}
1928
1929void Context::syncRendererState(const State::DirtyBits &bitMask)
1930{
1931 const State::DirtyBits &dirtyBits = (mState.getDirtyBits() & bitMask);
1932 if (dirtyBits.any())
1933 {
Jamie Madill63e4e722015-11-06 19:15:11 +00001934 mRenderer->syncState(mState, dirtyBits);
Jamie Madill1b94d432015-08-07 13:23:23 -04001935 mState.clearDirtyBits(dirtyBits);
1936 }
1937}
Jamie Madillc29968b2016-01-20 11:17:23 -05001938
1939void Context::blitFramebuffer(GLint srcX0,
1940 GLint srcY0,
1941 GLint srcX1,
1942 GLint srcY1,
1943 GLint dstX0,
1944 GLint dstY0,
1945 GLint dstX1,
1946 GLint dstY1,
1947 GLbitfield mask,
1948 GLenum filter)
1949{
1950 Framebuffer *readFramebuffer = mState.getReadFramebuffer();
1951 ASSERT(readFramebuffer);
1952
1953 Framebuffer *drawFramebuffer = mState.getDrawFramebuffer();
1954 ASSERT(drawFramebuffer);
1955
1956 Rectangle srcArea(srcX0, srcY0, srcX1 - srcX0, srcY1 - srcY0);
1957 Rectangle dstArea(dstX0, dstY0, dstX1 - dstX0, dstY1 - dstY0);
1958
1959 syncRendererState(mState.blitStateBitMask());
1960
1961 Error error = drawFramebuffer->blit(mState, srcArea, dstArea, mask, filter, readFramebuffer);
1962 if (error.isError())
1963 {
1964 recordError(error);
1965 return;
1966 }
apatrick@chromium.org144f2802012-07-12 01:42:34 +00001967}
Jamie Madillc29968b2016-01-20 11:17:23 -05001968
1969void Context::clear(GLbitfield mask)
1970{
1971 // Sync the clear state
1972 syncRendererState(mState.clearStateBitMask());
1973
1974 Error error = mState.getDrawFramebuffer()->clear(mData, mask);
1975 if (error.isError())
1976 {
1977 recordError(error);
1978 }
1979}
1980
1981void Context::clearBufferfv(GLenum buffer, GLint drawbuffer, const GLfloat *values)
1982{
1983 // Sync the clear state
1984 syncRendererState(mState.clearStateBitMask());
1985
1986 Error error = mState.getDrawFramebuffer()->clearBufferfv(mData, buffer, drawbuffer, values);
1987 if (error.isError())
1988 {
1989 recordError(error);
1990 }
1991}
1992
1993void Context::clearBufferuiv(GLenum buffer, GLint drawbuffer, const GLuint *values)
1994{
1995 // Sync the clear state
1996 syncRendererState(mState.clearStateBitMask());
1997
1998 Error error = mState.getDrawFramebuffer()->clearBufferuiv(mData, buffer, drawbuffer, values);
1999 if (error.isError())
2000 {
2001 recordError(error);
2002 }
2003}
2004
2005void Context::clearBufferiv(GLenum buffer, GLint drawbuffer, const GLint *values)
2006{
2007 // Sync the clear state
2008 syncRendererState(mState.clearStateBitMask());
2009
2010 Error error = mState.getDrawFramebuffer()->clearBufferiv(mData, buffer, drawbuffer, values);
2011 if (error.isError())
2012 {
2013 recordError(error);
2014 }
2015}
2016
2017void Context::clearBufferfi(GLenum buffer, GLint drawbuffer, GLfloat depth, GLint stencil)
2018{
2019 Framebuffer *framebufferObject = mState.getDrawFramebuffer();
2020 ASSERT(framebufferObject);
2021
2022 // If a buffer is not present, the clear has no effect
2023 if (framebufferObject->getDepthbuffer() == nullptr &&
2024 framebufferObject->getStencilbuffer() == nullptr)
2025 {
2026 return;
2027 }
2028
2029 // Sync the clear state
2030 syncRendererState(mState.clearStateBitMask());
2031
2032 Error error = framebufferObject->clearBufferfi(mData, buffer, drawbuffer, depth, stencil);
2033 if (error.isError())
2034 {
2035 recordError(error);
2036 }
2037}
2038
2039void Context::readPixels(GLint x,
2040 GLint y,
2041 GLsizei width,
2042 GLsizei height,
2043 GLenum format,
2044 GLenum type,
2045 GLvoid *pixels)
2046{
2047 // Sync pack state
2048 syncRendererState(mState.packStateBitMask());
2049
2050 Framebuffer *framebufferObject = mState.getReadFramebuffer();
2051 ASSERT(framebufferObject);
2052
2053 Rectangle area(x, y, width, height);
2054 Error error = framebufferObject->readPixels(mState, area, format, type, pixels);
2055 if (error.isError())
2056 {
2057 recordError(error);
2058 }
2059}
2060
2061void Context::copyTexImage2D(GLenum target,
2062 GLint level,
2063 GLenum internalformat,
2064 GLint x,
2065 GLint y,
2066 GLsizei width,
2067 GLsizei height,
2068 GLint border)
2069{
2070 Rectangle sourceArea(x, y, width, height);
2071
2072 const Framebuffer *framebuffer = mState.getReadFramebuffer();
2073 Texture *texture =
2074 getTargetTexture(IsCubeMapTextureTarget(target) ? GL_TEXTURE_CUBE_MAP : target);
2075 Error error = texture->copyImage(target, level, sourceArea, internalformat, framebuffer);
2076 if (error.isError())
2077 {
2078 recordError(error);
2079 }
2080}
2081
2082void Context::copyTexSubImage2D(GLenum target,
2083 GLint level,
2084 GLint xoffset,
2085 GLint yoffset,
2086 GLint x,
2087 GLint y,
2088 GLsizei width,
2089 GLsizei height)
2090{
2091 Offset destOffset(xoffset, yoffset, 0);
2092 Rectangle sourceArea(x, y, width, height);
2093
2094 const Framebuffer *framebuffer = mState.getReadFramebuffer();
2095 Texture *texture =
2096 getTargetTexture(IsCubeMapTextureTarget(target) ? GL_TEXTURE_CUBE_MAP : target);
2097 Error error = texture->copySubImage(target, level, destOffset, sourceArea, framebuffer);
2098 if (error.isError())
2099 {
2100 recordError(error);
2101 }
2102}
2103
2104void Context::copyTexSubImage3D(GLenum target,
2105 GLint level,
2106 GLint xoffset,
2107 GLint yoffset,
2108 GLint zoffset,
2109 GLint x,
2110 GLint y,
2111 GLsizei width,
2112 GLsizei height)
2113{
2114 Offset destOffset(xoffset, yoffset, zoffset);
2115 Rectangle sourceArea(x, y, width, height);
2116
2117 const Framebuffer *framebuffer = mState.getReadFramebuffer();
2118 Texture *texture = getTargetTexture(target);
2119 Error error = texture->copySubImage(target, level, destOffset, sourceArea, framebuffer);
2120 if (error.isError())
2121 {
2122 recordError(error);
2123 }
2124}
2125
2126void Context::framebufferTexture2D(GLenum target,
2127 GLenum attachment,
2128 GLenum textarget,
2129 GLuint texture,
2130 GLint level)
2131{
2132 Framebuffer *framebuffer = mState.getTargetFramebuffer(target);
2133 ASSERT(framebuffer);
2134
2135 if (texture != 0)
2136 {
2137 Texture *textureObj = getTexture(texture);
2138
2139 ImageIndex index = ImageIndex::MakeInvalid();
2140
2141 if (textarget == GL_TEXTURE_2D)
2142 {
2143 index = ImageIndex::Make2D(level);
2144 }
2145 else
2146 {
2147 ASSERT(IsCubeMapTextureTarget(textarget));
2148 index = ImageIndex::MakeCube(textarget, level);
2149 }
2150
2151 framebuffer->setAttachment(GL_TEXTURE, attachment, index, textureObj);
2152 }
2153 else
2154 {
2155 framebuffer->resetAttachment(attachment);
2156 }
2157}
2158
2159void Context::framebufferRenderbuffer(GLenum target,
2160 GLenum attachment,
2161 GLenum renderbuffertarget,
2162 GLuint renderbuffer)
2163{
2164 Framebuffer *framebuffer = mState.getTargetFramebuffer(target);
2165 ASSERT(framebuffer);
2166
2167 if (renderbuffer != 0)
2168 {
2169 Renderbuffer *renderbufferObject = getRenderbuffer(renderbuffer);
2170 framebuffer->setAttachment(GL_RENDERBUFFER, attachment, gl::ImageIndex::MakeInvalid(),
2171 renderbufferObject);
2172 }
2173 else
2174 {
2175 framebuffer->resetAttachment(attachment);
2176 }
2177}
2178
2179void Context::framebufferTextureLayer(GLenum target,
2180 GLenum attachment,
2181 GLuint texture,
2182 GLint level,
2183 GLint layer)
2184{
2185 Framebuffer *framebuffer = mState.getTargetFramebuffer(target);
2186 ASSERT(framebuffer);
2187
2188 if (texture != 0)
2189 {
2190 Texture *textureObject = getTexture(texture);
2191
2192 ImageIndex index = ImageIndex::MakeInvalid();
2193
2194 if (textureObject->getTarget() == GL_TEXTURE_3D)
2195 {
2196 index = ImageIndex::Make3D(level, layer);
2197 }
2198 else
2199 {
2200 ASSERT(textureObject->getTarget() == GL_TEXTURE_2D_ARRAY);
2201 index = ImageIndex::Make2DArray(level, layer);
2202 }
2203
2204 framebuffer->setAttachment(GL_TEXTURE, attachment, index, textureObject);
2205 }
2206 else
2207 {
2208 framebuffer->resetAttachment(attachment);
2209 }
2210}
2211
2212void Context::drawBuffers(GLsizei n, const GLenum *bufs)
2213{
2214 Framebuffer *framebuffer = mState.getDrawFramebuffer();
2215 ASSERT(framebuffer);
2216 framebuffer->setDrawBuffers(n, bufs);
2217}
2218
2219void Context::readBuffer(GLenum mode)
2220{
2221 Framebuffer *readFBO = mState.getReadFramebuffer();
2222 readFBO->setReadBuffer(mode);
2223}
2224
2225void Context::discardFramebuffer(GLenum target, GLsizei numAttachments, const GLenum *attachments)
2226{
2227 Framebuffer *framebuffer = mState.getTargetFramebuffer(target);
2228 ASSERT(framebuffer);
2229
2230 // The specification isn't clear what should be done when the framebuffer isn't complete.
2231 // We leave it up to the framebuffer implementation to decide what to do.
2232 Error error = framebuffer->discard(numAttachments, attachments);
2233 if (error.isError())
2234 {
2235 recordError(error);
2236 }
2237}
2238
2239void Context::invalidateFramebuffer(GLenum target,
2240 GLsizei numAttachments,
2241 const GLenum *attachments)
2242{
2243 Framebuffer *framebuffer = mState.getTargetFramebuffer(target);
2244 ASSERT(framebuffer);
2245
2246 if (framebuffer->checkStatus(mData) == GL_FRAMEBUFFER_COMPLETE)
2247 {
2248 Error error = framebuffer->invalidate(numAttachments, attachments);
2249 if (error.isError())
2250 {
2251 recordError(error);
2252 return;
2253 }
2254 }
2255}
2256
2257void Context::invalidateSubFramebuffer(GLenum target,
2258 GLsizei numAttachments,
2259 const GLenum *attachments,
2260 GLint x,
2261 GLint y,
2262 GLsizei width,
2263 GLsizei height)
2264{
2265 Framebuffer *framebuffer = mState.getTargetFramebuffer(target);
2266 ASSERT(framebuffer);
2267
2268 if (framebuffer->checkStatus(mData) == GL_FRAMEBUFFER_COMPLETE)
2269 {
2270 Rectangle area(x, y, width, height);
2271 Error error = framebuffer->invalidateSub(numAttachments, attachments, area);
2272 if (error.isError())
2273 {
2274 recordError(error);
2275 return;
2276 }
2277 }
2278}
2279
2280} // namespace gl