lwjgl/src/java/org/lwjgl/opengl/GLContext.java

264 lines
10 KiB
Java
Raw Normal View History

2004-08-15 19:03:54 -04:00
/*
2004-06-12 16:28:34 -04:00
* Copyright (c) 2002-2004 LWJGL Project
2004-02-18 12:48:26 -05:00
* All rights reserved.
2004-08-15 19:03:54 -04:00
*
2004-02-18 12:48:26 -05:00
* Redistribution and use in source and binary forms, with or without
2004-08-15 19:03:54 -04:00
* modification, are permitted provided that the following conditions are
2004-02-18 12:48:26 -05:00
* met:
2004-08-15 19:03:54 -04:00
*
* * Redistributions of source code must retain the above copyright
2004-02-18 12:48:26 -05:00
* 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.
*
2004-08-15 19:03:54 -04:00
* * Neither the name of 'LWJGL' nor the names of
* its contributors may be used to endorse or promote products derived
2004-02-23 11:30:48 -05:00
* from this software without specific prior written permission.
2004-08-15 19:03:54 -04:00
*
2004-02-18 12:48:26 -05:00
* 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
2004-08-15 19:03:54 -04:00
* 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
2004-02-18 12:48:26 -05:00
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
2004-08-15 19:03:54 -04:00
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
2004-02-18 12:48:26 -05:00
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package org.lwjgl.opengl;
2005-05-04 16:59:44 -04:00
import java.lang.reflect.Method;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import java.util.StringTokenizer;
import java.util.WeakHashMap;
import java.security.AccessController;
import java.security.PrivilegedAction;
import java.security.PrivilegedActionException;
import java.security.PrivilegedExceptionAction;
2004-09-09 19:51:16 -04:00
import org.lwjgl.LWJGLException;
import org.lwjgl.LWJGLUtil;
2004-09-09 19:51:16 -04:00
import org.lwjgl.Sys;
2004-02-18 12:48:26 -05:00
/**
2004-02-23 11:30:48 -05:00
* $Id$
* <p/>
2004-09-09 19:51:16 -04:00
* Manages GL contexts. Before any rendering is done by a LWJGL system, a call should be made to GLContext.useContext() with a
* context. This will ensure that GLContext has an accurate reflection of the current context's capabilities and function
* pointers.
2004-02-23 11:30:48 -05:00
*
* This class is thread-safe in the sense that multiple threads can safely call all public methods. The class is also
* thread-aware in the sense that it tracks a per-thread current context (including capabilities and function pointers).
* That way, multiple threads can have multiple contexts current and render to them concurrently.
*
2004-02-23 11:30:48 -05:00
* @author elias_naur <elias_naur@users.sourceforge.net>
* @version $Revision$
2004-02-18 12:48:26 -05:00
*/
2004-02-23 11:30:48 -05:00
public final class GLContext {
private final static ThreadLocal current_capabilities = new ThreadLocal();
private final static Map capability_cache = new WeakHashMap();
2004-02-23 11:30:48 -05:00
2004-11-09 16:29:17 -05:00
/** Map of classes that have native stubs loaded */
2004-09-09 19:51:16 -04:00
private static int gl_ref_count;
private static boolean did_auto_load;
2004-02-23 11:30:48 -05:00
static {
2004-03-27 08:48:58 -05:00
Sys.initialize();
2004-02-23 11:30:48 -05:00
}
2004-02-18 12:48:26 -05:00
/**
* Get the current capabilities instance. It contains the flags used
* to test for support of a particular extension.
2004-02-23 11:30:48 -05:00
*
* @return The current capabilities instance.
2004-02-18 12:48:26 -05:00
*/
public static ContextCapabilities getCapabilities() {
return ((ContextCapabilities)current_capabilities.get());
2004-02-23 11:30:48 -05:00
}
/**
* Set the current capabilities instance. It contains the flags used
* to test for support of a particular extension.
*
* @return The current capabilities instance.
*/
static void setCapabilities(ContextCapabilities capabilities) {
current_capabilities.set(capabilities);
}
/**
* Helper method to get a pointer to a named function in the OpenGL library
* with a name dependent on the current platform
*/
static long getPlatformSpecificFunctionAddress(String function_prefix, String[] os_prefixes, String[] os_function_prefixes, String function) {
String os_name = (String)AccessController.doPrivileged(new PrivilegedAction() {
public Object run() {
return System.getProperty("os.name");
}
});
for (int i = 0; i < os_prefixes.length; i++)
if (os_name.startsWith(os_prefixes[i])) {
String platform_function_name = function.replaceFirst(function_prefix, os_function_prefixes[i]);
long address = getFunctionAddress(platform_function_name);
return address;
}
return 0;
}
/**
* Helper method to get a pointer to a named function in the OpenGL library
*/
static native long getFunctionAddress(String name);
2004-02-18 12:48:26 -05:00
/**
* Determine which extensions are available. Helper method to ContextCapabilities.
2004-09-09 19:51:16 -04:00
*
* @return A Set containing all available extension strings.
2004-02-18 12:48:26 -05:00
*/
static Set getSupportedExtensions() {
Set supported_extensions = new HashSet();
String extensions_string = GL11.glGetString(GL11.GL_EXTENSIONS);
2005-02-21 09:46:47 -05:00
if (extensions_string == null)
throw new IllegalStateException("glGetString(GL_EXTENSIONS) returned null - is there a context current?");
StringTokenizer tokenizer = new StringTokenizer(extensions_string);
while ( tokenizer.hasMoreTokens() ) {
String extension_string = tokenizer.nextToken();
supported_extensions.add(extension_string);
}
2004-09-09 19:51:16 -04:00
String version = GL11.glGetString(GL11.GL_VERSION);
2005-01-12 03:58:41 -05:00
if (version == null)
throw new IllegalStateException("glGetString(GL_VERSION) returned null - possibly caused by missing current context.");
2004-09-13 04:04:33 -04:00
StringTokenizer version_tokenizer = new StringTokenizer(version, ". ");
String major_string = version_tokenizer.nextToken();
String minor_string = version_tokenizer.nextToken();
2004-09-09 19:51:16 -04:00
2004-09-13 04:04:33 -04:00
int majorVersion = Integer.parseInt(major_string);
int minorVersion = Integer.parseInt(minor_string);
2004-09-09 19:51:16 -04:00
2005-01-12 03:58:41 -05:00
if (majorVersion == 2) {
2004-09-09 19:51:16 -04:00
// ----------------------[ 2.X ]----------------------
supported_extensions.add("OpenGL20");
2004-09-09 19:51:16 -04:00
// ----------------------[ 1.X ]----------------------
supported_extensions.add("OpenGL11");
supported_extensions.add("OpenGL12");
supported_extensions.add("OpenGL13");
supported_extensions.add("OpenGL14");
supported_extensions.add("OpenGL15");
2004-09-09 19:51:16 -04:00
} else {
2005-01-12 03:58:41 -05:00
switch (minorVersion) {
2004-09-09 19:51:16 -04:00
case 5:
supported_extensions.add("OpenGL15");
2004-09-11 08:05:25 -04:00
// Intentional fall through
2004-09-09 19:51:16 -04:00
case 4:
supported_extensions.add("OpenGL14");
2004-09-11 08:05:25 -04:00
// Intentional fall through
2004-09-09 19:51:16 -04:00
case 3:
supported_extensions.add("OpenGL13");
2004-09-11 08:05:25 -04:00
// Intentional fall through
2004-09-09 19:51:16 -04:00
case 2:
supported_extensions.add("OpenGL12");
// Intentional fall through
case 1:
supported_extensions.add("OpenGL11");
}
}
return supported_extensions;
}
/**
* Helper method to ContextCapabilities. It will try to initialize the native stubs,
* and remove the given extension name from the extension set if the initialization fails.
*/
static void initNativeStubs(final Class extension_class, Set supported_extensions, String ext_name) {
resetNativeStubs(extension_class);
if (supported_extensions.contains(ext_name)) {
try {
AccessController.doPrivileged(new PrivilegedExceptionAction() {
public Object run() throws Exception {
Method init_stubs_method = extension_class.getDeclaredMethod("initNativeStubs", null);
init_stubs_method.invoke(null, null);
return null;
}
});
} catch (Exception e) {
LWJGLUtil.log("Failed to initialize extension " + extension_class + " - exception: " + e);
supported_extensions.remove(ext_name);
}
}
}
/**
* Makes a GL context the current LWJGL context by loading GL function pointers. The context must be current before a call to
* this method! Instead it simply ensures that the current context is reflected accurately by GLContext's extension caps and
* function pointers. Use useContext(null) when no context is active. <p>If the context is the same as last time, then this is
* a no-op. <p>If the context has not been encountered before it will be fully initialized from scratch. Otherwise a cached set
* of caps and function pointers will be used. <p>The reference to the context is held in a weak reference; therefore if no
* strong reference exists to the GL context it will automatically be forgotten by the VM at an indeterminate point in the
* future, freeing up a little RAM.
*
* @param context The context object, which uniquely identifies a GL context. If context is null, the native stubs are
* unloaded.
*
* @throws LWJGLException if context non-null, and the gl library can't be loaded or the basic GL11 functions can't be loaded
*/
public static synchronized void useContext(Object context) throws LWJGLException {
if (context == null) {
ContextCapabilities.unloadAllStubs();
setCapabilities(null);
if (did_auto_load)
unloadOpenGLLibrary();
return;
}
if (gl_ref_count == 0) {
loadOpenGLLibrary();
did_auto_load = true;
}
try {
ContextCapabilities capabilities = (ContextCapabilities)capability_cache.get(context);
if (capabilities == null) {
/*
* The capabilities object registers itself as current. This behaviour is caused
* by a chicken-and-egg situation where the constructor needs to call GL functions
* as part of its capability discovery, but GL functions cannot be called before
* a capabilities object has been set.
*/
new ContextCapabilities();
capability_cache.put(context, getCapabilities());
} else
setCapabilities(capabilities);
} catch (LWJGLException e) {
if (did_auto_load)
unloadOpenGLLibrary();
throw e;
}
}
2004-11-09 16:29:17 -05:00
/** If the OpenGL reference count is 0, the library is loaded. The reference count is then incremented. */
public static synchronized void loadOpenGLLibrary() throws LWJGLException {
if (gl_ref_count == 0)
nLoadOpenGLLibrary();
gl_ref_count++;
}
private static native void nLoadOpenGLLibrary() throws LWJGLException;
2004-11-09 16:29:17 -05:00
/** The OpenGL library reference count is decremented, and if it reaches 0, the library is unloaded. */
public static synchronized void unloadOpenGLLibrary() {
gl_ref_count--;
2004-09-09 19:51:16 -04:00
if ( gl_ref_count == 0 )
nUnloadOpenGLLibrary();
}
private static native void nUnloadOpenGLLibrary();
2004-11-09 16:29:17 -05:00
/** Native method to clear native stub bindings */
static native void resetNativeStubs(Class clazz);
2004-02-18 12:48:26 -05:00
}