2021-04-16 23:21:06 +08:00
|
|
|
# ----------------------------------------------------------------------------
|
|
|
|
# pyglet
|
|
|
|
# Copyright (c) 2006-2008 Alex Holkner
|
2022-04-30 13:56:57 +08:00
|
|
|
# Copyright (c) 2008-2022 pyglet contributors
|
2021-04-16 23:21:06 +08:00
|
|
|
# All rights reserved.
|
|
|
|
#
|
|
|
|
# Redistribution and use in source and binary forms, with or without
|
|
|
|
# modification, are permitted provided that the following conditions
|
|
|
|
# are met:
|
|
|
|
#
|
|
|
|
# * Redistributions of source code must retain the above copyright
|
|
|
|
# notice, this list of conditions and the following disclaimer.
|
|
|
|
# * Redistributions in binary form must reproduce the above copyright
|
|
|
|
# notice, this list of conditions and the following disclaimer in
|
|
|
|
# the documentation and/or other materials provided with the
|
|
|
|
# distribution.
|
|
|
|
# * Neither the name of pyglet nor the names of its
|
|
|
|
# contributors may be used to endorse or promote products
|
|
|
|
# derived from this software without specific prior written
|
|
|
|
# permission.
|
|
|
|
#
|
|
|
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
|
|
# FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
|
|
# COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
|
|
# BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
|
|
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
|
|
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
|
|
# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
# POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
# ----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
"""Information about version and extensions of current GL implementation.
|
|
|
|
|
|
|
|
Usage::
|
|
|
|
|
|
|
|
from pyglet.gl import gl_info
|
|
|
|
|
|
|
|
if gl_info.have_extension('GL_NV_register_combiners'):
|
|
|
|
# ...
|
|
|
|
|
|
|
|
If you are using more than one context, you can set up a separate GLInfo
|
|
|
|
object for each context. Call `set_active_context` after switching to the
|
|
|
|
context::
|
|
|
|
|
|
|
|
from pyglet.gl.gl_info import GLInfo
|
|
|
|
|
|
|
|
info = GLInfo()
|
|
|
|
info.set_active_context()
|
|
|
|
|
2021-09-23 06:34:23 +08:00
|
|
|
if info.have_version(4, 5):
|
2021-04-16 23:21:06 +08:00
|
|
|
# ...
|
|
|
|
|
|
|
|
"""
|
2021-09-23 06:34:23 +08:00
|
|
|
import warnings
|
2021-04-16 23:21:06 +08:00
|
|
|
|
2022-06-04 11:08:30 +08:00
|
|
|
from ctypes import c_char_p, cast
|
|
|
|
|
|
|
|
from pyglet.gl.gl import GL_EXTENSIONS, GL_RENDERER, GL_VENDOR, GL_VERSION
|
|
|
|
from pyglet.gl.gl import GL_MAJOR_VERSION, GL_MINOR_VERSION, GLint
|
|
|
|
from pyglet.gl.lib import GLException
|
2021-04-16 23:21:06 +08:00
|
|
|
from pyglet.util import asstr
|
|
|
|
|
|
|
|
|
2022-06-04 11:08:30 +08:00
|
|
|
def _get_number(parameter):
|
|
|
|
from pyglet.gl.gl import glGetIntegerv
|
|
|
|
number = GLint()
|
|
|
|
glGetIntegerv(parameter, number)
|
|
|
|
return number.value
|
|
|
|
|
|
|
|
|
2021-04-16 23:21:06 +08:00
|
|
|
class GLInfo:
|
|
|
|
"""Information interface for a single GL context.
|
|
|
|
|
|
|
|
A default instance is created automatically when the first OpenGL context
|
|
|
|
is created. You can use the module functions as a convenience for
|
|
|
|
this default instance's methods.
|
|
|
|
|
|
|
|
If you are using more than one context, you must call `set_active_context`
|
|
|
|
when the context is active for this `GLInfo` instance.
|
|
|
|
"""
|
2022-06-04 11:08:30 +08:00
|
|
|
_have_context = False
|
2021-04-16 23:21:06 +08:00
|
|
|
vendor = ''
|
|
|
|
renderer = ''
|
2022-06-04 11:08:30 +08:00
|
|
|
version = '0.0'
|
|
|
|
major_version = 0
|
|
|
|
minor_version = 0
|
|
|
|
opengl_api = 'gl'
|
2021-04-16 23:21:06 +08:00
|
|
|
extensions = set()
|
|
|
|
|
|
|
|
_have_info = False
|
|
|
|
|
|
|
|
def set_active_context(self):
|
|
|
|
"""Store information for the currently active context.
|
|
|
|
|
|
|
|
This method is called automatically for the default context.
|
|
|
|
"""
|
2022-06-04 11:08:30 +08:00
|
|
|
from pyglet.gl.gl import glGetString, glGetStringi, GL_NUM_EXTENSIONS
|
2021-09-23 06:34:23 +08:00
|
|
|
|
2022-06-04 11:08:30 +08:00
|
|
|
self._have_context = True
|
2021-04-16 23:21:06 +08:00
|
|
|
if not self._have_info:
|
|
|
|
self.vendor = asstr(cast(glGetString(GL_VENDOR), c_char_p).value)
|
|
|
|
self.renderer = asstr(cast(glGetString(GL_RENDERER), c_char_p).value)
|
|
|
|
self.version = asstr(cast(glGetString(GL_VERSION), c_char_p).value)
|
2022-05-25 09:16:38 +08:00
|
|
|
# NOTE: The version string requirements for gles is a lot stricter
|
|
|
|
# so using this to rely on detecting the API is not too unreasonable
|
|
|
|
self.opengl_api = "gles" if "opengl es" in self.version.lower() else "gl"
|
2022-06-04 11:08:30 +08:00
|
|
|
|
|
|
|
try:
|
|
|
|
self.major_version = _get_number(GL_MAJOR_VERSION)
|
|
|
|
self.minor_version = _get_number(GL_MINOR_VERSION)
|
|
|
|
num_ext = _get_number(GL_NUM_EXTENSIONS)
|
|
|
|
self.extensions = (asstr(cast(glGetStringi(GL_EXTENSIONS, i), c_char_p).value) for i in range(num_ext))
|
|
|
|
self.extensions = set(self.extensions)
|
|
|
|
except GLException:
|
|
|
|
pass # GL3 is likely not available
|
|
|
|
|
2021-04-16 23:21:06 +08:00
|
|
|
self._have_info = True
|
|
|
|
|
|
|
|
def remove_active_context(self):
|
2022-06-04 11:08:30 +08:00
|
|
|
self._have_context = False
|
2021-04-16 23:21:06 +08:00
|
|
|
self._have_info = False
|
|
|
|
|
2022-06-04 11:08:30 +08:00
|
|
|
def have_context(self):
|
|
|
|
return self._have_context
|
|
|
|
|
2021-04-16 23:21:06 +08:00
|
|
|
def have_extension(self, extension):
|
|
|
|
"""Determine if an OpenGL extension is available.
|
|
|
|
|
|
|
|
:Parameters:
|
|
|
|
`extension` : str
|
|
|
|
The name of the extension to test for, including its
|
|
|
|
``GL_`` prefix.
|
|
|
|
|
|
|
|
:return: True if the extension is provided by the driver.
|
|
|
|
:rtype: bool
|
|
|
|
"""
|
2022-06-04 11:08:30 +08:00
|
|
|
if not self._have_context:
|
2021-04-16 23:21:06 +08:00
|
|
|
warnings.warn('No GL context created yet.')
|
|
|
|
return extension in self.extensions
|
|
|
|
|
|
|
|
def get_extensions(self):
|
|
|
|
"""Get a list of available OpenGL extensions.
|
|
|
|
|
|
|
|
:return: a list of the available extensions.
|
|
|
|
:rtype: list of str
|
|
|
|
"""
|
2022-06-04 11:08:30 +08:00
|
|
|
if not self._have_context:
|
2021-04-16 23:21:06 +08:00
|
|
|
warnings.warn('No GL context created yet.')
|
|
|
|
return self.extensions
|
|
|
|
|
|
|
|
def get_version(self):
|
|
|
|
"""Get the current OpenGL version.
|
|
|
|
|
2022-05-25 09:16:38 +08:00
|
|
|
:return: The major and minor version as a tuple
|
|
|
|
:rtype: tuple
|
|
|
|
"""
|
2022-06-04 11:08:30 +08:00
|
|
|
if not self._have_context:
|
2022-05-25 09:16:38 +08:00
|
|
|
warnings.warn('No GL context created yet.')
|
|
|
|
return self.major_version, self.minor_version
|
|
|
|
|
|
|
|
def get_version_string(self):
|
|
|
|
"""Get the current OpenGL version string.
|
|
|
|
|
|
|
|
:return: The OpenGL version string
|
2021-04-16 23:21:06 +08:00
|
|
|
:rtype: str
|
|
|
|
"""
|
2022-06-04 11:08:30 +08:00
|
|
|
if not self._have_context:
|
2021-04-16 23:21:06 +08:00
|
|
|
warnings.warn('No GL context created yet.')
|
|
|
|
return self.version
|
|
|
|
|
2021-09-23 06:34:23 +08:00
|
|
|
def have_version(self, major, minor=0):
|
2021-04-16 23:21:06 +08:00
|
|
|
"""Determine if a version of OpenGL is supported.
|
|
|
|
|
|
|
|
:Parameters:
|
|
|
|
`major` : int
|
|
|
|
The major revision number (typically 1 or 2).
|
|
|
|
`minor` : int
|
|
|
|
The minor revision number.
|
|
|
|
|
|
|
|
:rtype: bool
|
|
|
|
:return: True if the requested or a later version is supported.
|
|
|
|
"""
|
|
|
|
|
2022-06-04 11:08:30 +08:00
|
|
|
if not self._have_context:
|
2021-04-16 23:21:06 +08:00
|
|
|
warnings.warn('No GL context created yet.')
|
2022-05-25 09:16:38 +08:00
|
|
|
if not self.major_version and not self.minor_version:
|
2021-04-16 23:21:06 +08:00
|
|
|
return False
|
2022-05-25 09:16:38 +08:00
|
|
|
|
|
|
|
return (self.major_version > major or
|
|
|
|
(self.major_version == major and self.minor_version >= minor) or
|
|
|
|
(self.major_version == major and self.minor_version == minor))
|
2021-04-16 23:21:06 +08:00
|
|
|
|
|
|
|
def get_renderer(self):
|
|
|
|
"""Determine the renderer string of the OpenGL context.
|
|
|
|
|
|
|
|
:rtype: str
|
|
|
|
"""
|
2022-06-04 11:08:30 +08:00
|
|
|
if not self._have_context:
|
2021-04-16 23:21:06 +08:00
|
|
|
warnings.warn('No GL context created yet.')
|
|
|
|
return self.renderer
|
|
|
|
|
|
|
|
def get_vendor(self):
|
|
|
|
"""Determine the vendor string of the OpenGL context.
|
|
|
|
|
|
|
|
:rtype: str
|
|
|
|
"""
|
2022-06-04 11:08:30 +08:00
|
|
|
if not self._have_context:
|
2021-04-16 23:21:06 +08:00
|
|
|
warnings.warn('No GL context created yet.')
|
|
|
|
return self.vendor
|
|
|
|
|
2022-05-25 09:16:38 +08:00
|
|
|
def get_opengl_api(self):
|
|
|
|
"""Determine the OpenGL API version.
|
|
|
|
Usually ``gl`` or ``gles``.
|
|
|
|
|
|
|
|
:rtype: str
|
|
|
|
"""
|
2022-06-04 11:08:30 +08:00
|
|
|
if not self._have_context:
|
2022-05-25 09:16:38 +08:00
|
|
|
warnings.warn('No GL context created yet.')
|
|
|
|
return self.opengl_api
|
|
|
|
|
2021-04-16 23:21:06 +08:00
|
|
|
|
2021-09-23 06:34:23 +08:00
|
|
|
# Single instance useful for apps with only a single context
|
|
|
|
# (or all contexts have the same GL driver, a common case).
|
2021-04-16 23:21:06 +08:00
|
|
|
_gl_info = GLInfo()
|
|
|
|
|
|
|
|
get_extensions = _gl_info.get_extensions
|
|
|
|
get_version = _gl_info.get_version
|
2022-05-25 09:16:38 +08:00
|
|
|
get_version_string = _gl_info.get_version_string
|
2021-04-16 23:21:06 +08:00
|
|
|
have_version = _gl_info.have_version
|
|
|
|
get_renderer = _gl_info.get_renderer
|
|
|
|
get_vendor = _gl_info.get_vendor
|
2022-05-25 09:16:38 +08:00
|
|
|
get_opengl_api = _gl_info.get_opengl_api
|
2022-06-04 11:08:30 +08:00
|
|
|
have_extension = _gl_info.have_extension
|
|
|
|
have_context = _gl_info.have_context
|
|
|
|
remove_active_context = _gl_info.remove_active_context
|
|
|
|
set_active_context = _gl_info.set_active_context
|