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