// // Copyright(c) 2014 The ANGLE Project Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // // global_state.cpp : Implements functions for querying the thread-local GL and EGL state. #include "libGLESv2/global_state.h" #include "libANGLE/Context.h" #include "libANGLE/Error.h" #include "common/debug.h" #include "common/platform.h" #include "common/tls.h" namespace { static TLSIndex currentTLS = TLS_INVALID_INDEX; struct Current { EGLint error; EGLenum API; egl::Display *display; egl::Surface *drawSurface; egl::Surface *readSurface; gl::Context *context; }; Current *AllocateCurrent() { ASSERT(currentTLS != TLS_INVALID_INDEX); if (currentTLS == TLS_INVALID_INDEX) { return NULL; } Current *current = new Current(); current->error = EGL_SUCCESS; current->API = EGL_OPENGL_ES_API; current->display = reinterpret_cast<egl::Display*>(EGL_NO_DISPLAY); current->drawSurface = reinterpret_cast<egl::Surface*>(EGL_NO_SURFACE); current->readSurface = reinterpret_cast<egl::Surface*>(EGL_NO_SURFACE); current->context = reinterpret_cast<gl::Context*>(EGL_NO_CONTEXT); if (!SetTLSValue(currentTLS, current)) { ERR("Could not set thread local storage."); return NULL; } return current; } Current *GetCurrentData() { // Create a TLS index if one has not been created for this DLL if (currentTLS == TLS_INVALID_INDEX) { currentTLS = CreateTLSIndex(); } Current *current = reinterpret_cast<Current*>(GetTLSValue(currentTLS)); // ANGLE issue 488: when the dll is loaded after thread initialization, // thread local storage (current) might not exist yet. return (current ? current : AllocateCurrent()); } #ifdef ANGLE_PLATFORM_WINDOWS void DeallocateCurrent() { Current *current = reinterpret_cast<Current*>(GetTLSValue(currentTLS)); SafeDelete(current); SetTLSValue(currentTLS, NULL); } extern "C" BOOL WINAPI DllMain(HINSTANCE, DWORD reason, LPVOID) { switch (reason) { case DLL_PROCESS_ATTACH: currentTLS = CreateTLSIndex(); if (currentTLS == TLS_INVALID_INDEX) { return FALSE; } AllocateCurrent(); break; case DLL_THREAD_ATTACH: AllocateCurrent(); break; case DLL_THREAD_DETACH: DeallocateCurrent(); break; case DLL_PROCESS_DETACH: DeallocateCurrent(); if (currentTLS != TLS_INVALID_INDEX) { DestroyTLSIndex(currentTLS); currentTLS = TLS_INVALID_INDEX; } break; } return TRUE; } #endif } namespace gl { Context *GetGlobalContext() { Current *current = GetCurrentData(); return current->context; } Context *GetValidGlobalContext() { gl::Context *context = GetGlobalContext(); if (context) { if (context->isContextLost()) { context->handleError(gl::Error(GL_OUT_OF_MEMORY, "Context has been lost.")); return nullptr; } else { return context; } } return nullptr; } } namespace egl { void SetGlobalError(const Error &error) { Current *current = GetCurrentData(); current->error = error.getCode(); } EGLint GetGlobalError() { Current *current = GetCurrentData(); return current->error; } EGLenum GetGlobalAPI() { Current *current = GetCurrentData(); return current->API; } void SetGlobalAPI(EGLenum API) { Current *current = GetCurrentData(); current->API = API; } void SetGlobalDisplay(Display *dpy) { Current *current = GetCurrentData(); current->display = dpy; } Display *GetGlobalDisplay() { Current *current = GetCurrentData(); return current->display; } void SetGlobalDrawSurface(Surface *surface) { Current *current = GetCurrentData(); current->drawSurface = surface; } Surface *GetGlobalDrawSurface() { Current *current = GetCurrentData(); return current->drawSurface; } void SetGlobalReadSurface(Surface *surface) { Current *current = GetCurrentData(); current->readSurface = surface; } Surface *GetGlobalReadSurface() { Current *current = GetCurrentData(); return current->readSurface; } void SetGlobalContext(gl::Context *context) { Current *current = GetCurrentData(); current->context = context; } gl::Context *GetGlobalContext() { Current *current = GetCurrentData(); return current->context; } }