eglglobals.c revision 7e995a2e
13464ebd5Sriastradh/************************************************************************** 23464ebd5Sriastradh * 3af69d88dSmrg * Copyright 2008 VMware, Inc. 43464ebd5Sriastradh * Copyright 2009-2010 Chia-I Wu <olvaffe@gmail.com> 53464ebd5Sriastradh * Copyright 2010-2011 LunarG, Inc. 63464ebd5Sriastradh * All Rights Reserved. 73464ebd5Sriastradh * 83464ebd5Sriastradh * Permission is hereby granted, free of charge, to any person obtaining a 93464ebd5Sriastradh * copy of this software and associated documentation files (the 103464ebd5Sriastradh * "Software"), to deal in the Software without restriction, including 113464ebd5Sriastradh * without limitation the rights to use, copy, modify, merge, publish, 123464ebd5Sriastradh * distribute, sub license, and/or sell copies of the Software, and to 133464ebd5Sriastradh * permit persons to whom the Software is furnished to do so, subject to 143464ebd5Sriastradh * the following conditions: 153464ebd5Sriastradh * 163464ebd5Sriastradh * The above copyright notice and this permission notice (including the 173464ebd5Sriastradh * next paragraph) shall be included in all copies or substantial portions 183464ebd5Sriastradh * of the Software. 193464ebd5Sriastradh * 203464ebd5Sriastradh * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 213464ebd5Sriastradh * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 223464ebd5Sriastradh * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 233464ebd5Sriastradh * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 243464ebd5Sriastradh * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 253464ebd5Sriastradh * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER 263464ebd5Sriastradh * DEALINGS IN THE SOFTWARE. 273464ebd5Sriastradh * 283464ebd5Sriastradh **************************************************************************/ 293464ebd5Sriastradh 303464ebd5Sriastradh 314a49301eSmrg#include <stdlib.h> 327e995a2eSmrg#include <stdio.h> 337e995a2eSmrg#include <string.h> 344a49301eSmrg#include <assert.h> 357e995a2eSmrg#include "c11/threads.h" 367e995a2eSmrg 374a49301eSmrg#include "eglglobals.h" 387e995a2eSmrg#include "egldevice.h" 39cdc920a0Smrg#include "egldisplay.h" 404a49301eSmrg#include "egldriver.h" 417e995a2eSmrg#include "egllog.h" 427e995a2eSmrg 437e995a2eSmrg#ifdef HAVE_MINCORE 447e995a2eSmrg#include <unistd.h> 457e995a2eSmrg#include <sys/mman.h> 467e995a2eSmrg#endif 474a49301eSmrg 484a49301eSmrg 497e995a2eSmrgstatic mtx_t _eglGlobalMutex = _MTX_INITIALIZER_NP; 50af69d88dSmrg 514a49301eSmrgstruct _egl_global _eglGlobal = 524a49301eSmrg{ 537e995a2eSmrg .Mutex = &_eglGlobalMutex, 547e995a2eSmrg .DisplayList = NULL, 557e995a2eSmrg .DeviceList = &_eglSoftwareDevice, 567e995a2eSmrg .NumAtExitCalls = 3, 577e995a2eSmrg .AtExitCalls = { 584a49301eSmrg /* default AtExitCalls, called in reverse order */ 597e995a2eSmrg _eglFiniDevice, /* always called last */ 607e995a2eSmrg _eglUnloadDrivers, 617e995a2eSmrg _eglFiniDisplay, 62af69d88dSmrg }, 63af69d88dSmrg 647e995a2eSmrg .ClientOnlyExtensionString = 65af69d88dSmrg "EGL_EXT_client_extensions" 667e995a2eSmrg " EGL_EXT_device_base" 677e995a2eSmrg " EGL_EXT_device_enumeration" 687e995a2eSmrg " EGL_EXT_device_query" 69af69d88dSmrg " EGL_EXT_platform_base" 707e995a2eSmrg " EGL_KHR_client_get_all_proc_addresses" 717e995a2eSmrg " EGL_KHR_debug", 727e995a2eSmrg 737e995a2eSmrg .PlatformExtensionString = 747e995a2eSmrg#ifdef HAVE_WAYLAND_PLATFORM 75af69d88dSmrg " EGL_EXT_platform_wayland" 767e995a2eSmrg#endif 777e995a2eSmrg#ifdef HAVE_X11_PLATFORM 787e995a2eSmrg " EGL_EXT_platform_x11" 797e995a2eSmrg#endif 807e995a2eSmrg#ifdef HAVE_DRM_PLATFORM 81af69d88dSmrg " EGL_MESA_platform_gbm" 827e995a2eSmrg#endif 837e995a2eSmrg#ifdef HAVE_SURFACELESS_PLATFORM 847e995a2eSmrg " EGL_MESA_platform_surfaceless" 857e995a2eSmrg#endif 867e995a2eSmrg "", 877e995a2eSmrg 887e995a2eSmrg .ClientExtensionString = NULL, 897e995a2eSmrg 907e995a2eSmrg .debugCallback = NULL, 917e995a2eSmrg .debugTypesEnabled = _EGL_DEBUG_BIT_CRITICAL | _EGL_DEBUG_BIT_ERROR, 924a49301eSmrg}; 934a49301eSmrg 94d63b28c1Smartinstatic EGLBoolean registered = EGL_FALSE; 954a49301eSmrg 96d63b28c1Smartinstatic void __attribute__((__destructor__)) 974a49301eSmrg_eglAtExit(void) 984a49301eSmrg{ 994a49301eSmrg EGLint i; 100d63b28c1Smartin 101d63b28c1Smartin if (!registered) 1027e995a2eSmrg return; 103d63b28c1Smartin 1044a49301eSmrg for (i = _eglGlobal.NumAtExitCalls - 1; i >= 0; i--) 1054a49301eSmrg _eglGlobal.AtExitCalls[i](); 1064a49301eSmrg} 1074a49301eSmrg 1084a49301eSmrg 1094a49301eSmrgvoid 1104a49301eSmrg_eglAddAtExitCall(void (*func)(void)) 1114a49301eSmrg{ 1124a49301eSmrg if (func) { 1137e995a2eSmrg static EGLBoolean registered = EGL_FALSE; 1144a49301eSmrg 1157e995a2eSmrg mtx_lock(_eglGlobal.Mutex); 1164a49301eSmrg 117d63b28c1Smartin registered = EGL_TRUE; 1184a49301eSmrg 1194a49301eSmrg assert(_eglGlobal.NumAtExitCalls < ARRAY_SIZE(_eglGlobal.AtExitCalls)); 1204a49301eSmrg _eglGlobal.AtExitCalls[_eglGlobal.NumAtExitCalls++] = func; 1214a49301eSmrg 1227e995a2eSmrg mtx_unlock(_eglGlobal.Mutex); 1237e995a2eSmrg } 1247e995a2eSmrg} 1257e995a2eSmrg 1267e995a2eSmrgconst char * 1277e995a2eSmrg_eglGetClientExtensionString(void) 1287e995a2eSmrg{ 1297e995a2eSmrg const char *ret; 1307e995a2eSmrg 1317e995a2eSmrg mtx_lock(_eglGlobal.Mutex); 1327e995a2eSmrg 1337e995a2eSmrg if (_eglGlobal.ClientExtensionString == NULL) { 1347e995a2eSmrg size_t clientLen = strlen(_eglGlobal.ClientOnlyExtensionString); 1357e995a2eSmrg size_t platformLen = strlen(_eglGlobal.PlatformExtensionString); 1367e995a2eSmrg 1377e995a2eSmrg _eglGlobal.ClientExtensionString = (char *) malloc(clientLen + platformLen + 1); 1387e995a2eSmrg if (_eglGlobal.ClientExtensionString != NULL) { 1397e995a2eSmrg char *ptr = _eglGlobal.ClientExtensionString; 1407e995a2eSmrg 1417e995a2eSmrg memcpy(ptr, _eglGlobal.ClientOnlyExtensionString, clientLen); 1427e995a2eSmrg ptr += clientLen; 1437e995a2eSmrg 1447e995a2eSmrg if (platformLen > 0) { 1457e995a2eSmrg // Note that if PlatformExtensionString is not empty, then it will 1467e995a2eSmrg // already have a leading space. 1477e995a2eSmrg assert(_eglGlobal.PlatformExtensionString[0] == ' '); 1487e995a2eSmrg memcpy(ptr, _eglGlobal.PlatformExtensionString, platformLen); 1497e995a2eSmrg ptr += platformLen; 1507e995a2eSmrg } 1517e995a2eSmrg *ptr = '\0'; 1527e995a2eSmrg } 1534a49301eSmrg } 1547e995a2eSmrg ret = _eglGlobal.ClientExtensionString; 1557e995a2eSmrg 1567e995a2eSmrg mtx_unlock(_eglGlobal.Mutex); 1577e995a2eSmrg return ret; 1587e995a2eSmrg} 1597e995a2eSmrg 1607e995a2eSmrgEGLBoolean 1617e995a2eSmrg_eglPointerIsDereferencable(void *p) 1627e995a2eSmrg{ 1637e995a2eSmrg#ifdef HAVE_MINCORE 1647e995a2eSmrg uintptr_t addr = (uintptr_t) p; 1657e995a2eSmrg unsigned char valid = 0; 1667e995a2eSmrg const long page_size = getpagesize(); 1677e995a2eSmrg 1687e995a2eSmrg if (p == NULL) 1697e995a2eSmrg return EGL_FALSE; 1707e995a2eSmrg 1717e995a2eSmrg /* align addr to page_size */ 1727e995a2eSmrg addr &= ~(page_size - 1); 1737e995a2eSmrg 1747e995a2eSmrg if (mincore((void *) addr, page_size, &valid) < 0) { 1757e995a2eSmrg _eglLog(_EGL_DEBUG, "mincore failed: %m"); 1767e995a2eSmrg return EGL_FALSE; 1777e995a2eSmrg } 1787e995a2eSmrg 1797e995a2eSmrg /* mincore() returns 0 on success, and -1 on failure. The last parameter 1807e995a2eSmrg * is a vector of bytes with one entry for each page queried. mincore 1817e995a2eSmrg * returns page residency information in the first bit of each byte in the 1827e995a2eSmrg * vector. 1837e995a2eSmrg * 1847e995a2eSmrg * Residency doesn't actually matter when determining whether a pointer is 1857e995a2eSmrg * dereferenceable, so the output vector can be ignored. What matters is 1867e995a2eSmrg * whether mincore succeeds. See: 1877e995a2eSmrg * 1887e995a2eSmrg * http://man7.org/linux/man-pages/man2/mincore.2.html 1897e995a2eSmrg */ 1907e995a2eSmrg return EGL_TRUE; 1917e995a2eSmrg#else 1927e995a2eSmrg return p != NULL; 1937e995a2eSmrg#endif 1944a49301eSmrg} 195