org.lwjgl.opengles.GLES Maven / Gradle / Ivy
Show all versions of lwjgl-opengles Show documentation
/*
* Copyright LWJGL. All rights reserved.
* License terms: https://www.lwjgl.org/license
*/
package org.lwjgl.opengles;
import org.jspecify.annotations.*;
import org.lwjgl.*;
import org.lwjgl.system.*;
import java.nio.*;
import java.util.*;
import java.util.function.*;
import static java.lang.Math.*;
import static org.lwjgl.opengles.GLES30.*;
import static org.lwjgl.system.APIUtil.*;
import static org.lwjgl.system.Checks.*;
import static org.lwjgl.system.JNI.*;
import static org.lwjgl.system.MemoryStack.*;
import static org.lwjgl.system.MemoryUtil.*;
/**
* This class must be used before any OpenGL ES function is called. It has the following responsibilities:
*
* - Loads the OpenGL ES native library into the JVM process.
* - Creates instances of {@link GLESCapabilities} classes. A {@code GLESCapabilities} instance contains flags for functionality that is available in an
* OpenGL ES context. Internally, it also contains function pointers that are only valid in that specific OpenGL ES context.
* - Maintains thread-local state for {@code GLESCapabilities} instances, corresponding to OpenGL ES contexts that are current in those threads.
*
*
* Library lifecycle
* The OpenGL ES library is loaded automatically when this class is initialized. Set the {@link Configuration#OPENGLES_EXPLICIT_INIT} option to override
* this behavior. Manual loading/unloading can be achieved with the {@link #create} and {@link #destroy} functions. The name of the library loaded can
* be overridden with the {@link Configuration#OPENGLES_LIBRARY_NAME} option. The maximum OpenGL ES version loaded can be set with the
* {@link Configuration#OPENGLES_MAXVERSION} option. This can be useful to ensure that no functionality above a specific version is used during development.
*
* GLESCapabilities creation
* Instances of {@code GLESCapabilities} can be created with the {@link #createCapabilities} method. An OpenGL ES context must be current in the current
* thread before it is called. Calling this method is expensive, so the {@code GLESCapabilities} instance should be associated with the OpenGL ES context and
* reused as necessary.
*
* Thread-local state
* Before a function for a given OpenGL ES context can be called, the corresponding {@code GLESCapabilities} instance must be passed to the {@link
* #setCapabilities} method. The user is also responsible for clearing the current {@code GLESCapabilities} instance when the context is destroyed or made
* current in another thread.
*
* Note that the {@link #createCapabilities} method implicitly calls {@link #setCapabilities} with the newly created instance.
*/
public final class GLES {
private static final @Nullable APIVersion MAX_VERSION;
private static @Nullable FunctionProvider functionProvider;
private static final ThreadLocal<@Nullable GLESCapabilities> capabilitiesTLS = new ThreadLocal<>();
private static ICD icd = new ICDStatic();
static {
Library.loadSystem(System::load, System::loadLibrary, GLES.class, "org.lwjgl.opengles", Platform.mapLibraryNameBundled("lwjgl_opengles"));
MAX_VERSION = apiParseVersion(Configuration.OPENGLES_MAXVERSION);
if (!Configuration.OPENGLES_EXPLICIT_INIT.get(false)) {
create();
}
}
private GLES() {}
/** Ensures that the lwjgl_opengles shared library has been loaded. */
static void initialize() {
// intentionally empty to trigger static initializer
}
/** Loads the OpenGL ES native library, using the default library name. */
public static void create() {
create(Library.loadNative(GLES.class, "org.lwjgl.opengles", Configuration.OPENGLES_LIBRARY_NAME, Configuration.OPENGLES_LIBRARY_NAME_DEFAULTS()));
}
/**
* Loads the OpenGL ES native library, using the specified library name.
*
* @param libName the native library name
*/
public static void create(String libName) {
create(Library.loadNative(GLES.class, "org.lwjgl.opengles", libName));
}
private static @Nullable SharedLibrary getContextProvider() {
SharedLibrary GL = null;
String contextAPI = Configuration.OPENGLES_CONTEXT_API.get();
if ("native".equals(contextAPI)) {
GL = loadNative();
} else if ("OSMesa".equals(contextAPI)) {
GL = loadOSMesa();
}
if (GL == null) {
GL = loadEGL();
if (GL == null && !"EGL".equals(contextAPI)) {
if (!"native".equals(contextAPI)) {
GL = loadNative();
}
if (GL == null && !"OSMesa".equals(contextAPI)) {
GL = loadOSMesa();
}
}
}
return GL == null ? null : new SharedLibrary.Delegate(GL) {
private final long GetProcAddress;
{
long GetProcAddress = NULL;
switch (Platform.get()) {
case FREEBSD:
case LINUX:
GetProcAddress = library.getFunctionAddress("glXGetProcAddress");
if (GetProcAddress == NULL) {
GetProcAddress = library.getFunctionAddress("glXGetProcAddressARB");
}
break;
case WINDOWS:
GetProcAddress = library.getFunctionAddress("wglGetProcAddress");
break;
}
if (GetProcAddress == NULL) {
GetProcAddress = library.getFunctionAddress("eglGetProcAddress");
}
if (GetProcAddress == NULL) {
GetProcAddress = library.getFunctionAddress("OSMesaGetProcAddress");
}
this.GetProcAddress = GetProcAddress;
}
@Override
public long getFunctionAddress(ByteBuffer functionName) {
long address = GetProcAddress == NULL ? NULL : callPP(memAddress(functionName), GetProcAddress);
if (address == NULL) {
address = library.getFunctionAddress(functionName);
}
return address;
}
};
}
private static @Nullable SharedLibrary loadEGL() {
try {
return Library.loadNative(GLES.class, "org.lwjgl.opengles", Configuration.EGL_LIBRARY_NAME, Configuration.EGL_LIBRARY_NAME_DEFAULTS());
} catch (Throwable ignored) {
apiLog("[GLES] Failed to initialize context management based on EGL");
return null;
}
}
private static @Nullable SharedLibrary loadNative() {
try {
return Library.loadNative(GLES.class, "org.lwjgl.opengles", Configuration.OPENGL_LIBRARY_NAME, Configuration.OPENGL_LIBRARY_NAME_DEFAULTS());
} catch (Throwable ignored) {
apiLog("[GLES] Failed to initialize context management based on native OpenGL platform API");
return null;
}
}
private static @Nullable SharedLibrary loadOSMesa() {
try {
return Library.loadNative(GLES.class, "org.lwjgl.opengles", Configuration.OPENGL_OSMESA_LIBRARY_NAME, Configuration.OPENGL_OSMESA_LIBRARY_NAME_DEFAULTS());
} catch (Throwable ignored) {
apiLog("[GLES] Failed to initialize context management based on OSMesa");
return null;
}
}
private static void create(SharedLibrary GLES) {
SharedLibrary contextProvider = getContextProvider();
if (contextProvider == null) {
GLES.free();
throw new IllegalStateException("There is no OpenGL ES context management API available.");
}
SharedLibrary provider = new SharedLibrary.Delegate(GLES) {
@Override
public long getFunctionAddress(ByteBuffer functionName) {
long address = contextProvider.getFunctionAddress(functionName);
if (address == NULL) {
address = library.getFunctionAddress(functionName);
if (address == NULL && DEBUG_FUNCTIONS) {
apiLogMissing("GLES", functionName);
}
}
return address;
}
@Override
public void free() {
super.free();
contextProvider.free();
}
};
try {
create((FunctionProvider)provider);
} catch (RuntimeException e) {
provider.free();
throw e;
}
}
/**
* Initializes OpenGL ES with the specified {@link FunctionProvider}. This method can be used to implement custom OpenGL ES library loading.
*
* @param functionProvider the provider of OpenGL ES function addresses
*/
public static void create(FunctionProvider functionProvider) {
if (GLES.functionProvider != null) {
throw new IllegalStateException("OpenGL ES has already been created.");
}
GLES.functionProvider = functionProvider;
ThreadLocalUtil.setFunctionMissingAddresses(GLESCapabilities.ADDRESS_BUFFER_SIZE);
}
/** Unloads the OpenGL ES native library. */
public static void destroy() {
if (functionProvider == null) {
return;
}
ThreadLocalUtil.setFunctionMissingAddresses(0);
if (functionProvider instanceof NativeResource) {
((NativeResource)functionProvider).free();
}
functionProvider = null;
}
/** Returns the {@link FunctionProvider} for the OpenGL ES native library. */
public static @Nullable FunctionProvider getFunctionProvider() {
return functionProvider;
}
/**
* Sets the {@link GLESCapabilities} of the OpenGL ES context that is current in the current thread.
*
* This {@code GLESCapabilities} instance will be used by any OpenGL ES call in the current thread, until {@code setCapabilities} is called again with
* a different value.
*/
public static void setCapabilities(@Nullable GLESCapabilities caps) {
capabilitiesTLS.set(caps);
ThreadLocalUtil.setCapabilities(caps == null ? NULL : memAddress(caps.addresses));
icd.set(caps);
}
/**
* Returns the {@link GLESCapabilities} of the OpenGL ES context that is current in the current thread.
*
* @throws IllegalStateException if {@link #setCapabilities} has never been called in the current thread or was last called with a {@code null} value
*/
public static GLESCapabilities getCapabilities() {
return checkCapabilities(capabilitiesTLS.get());
}
private static GLESCapabilities checkCapabilities(@Nullable GLESCapabilities caps) {
if (caps == null) {
throw new IllegalStateException(
"No GLESCapabilities instance set for the current thread. Possible solutions:\n" +
"\ta) Call GLES.createCapabilities() after making a context current in the current thread.\n" +
"\tb) Call GLES.setCapabilities() if a GLESCapabilities instance already exists for the current context."
);
}
return caps;
}
/**
* Creates a new {@link GLESCapabilities} instance for the OpenGL ES context that is current in the current thread.
*
* This method calls {@link #setCapabilities(GLESCapabilities)} with the new instance before returning.
*
* @return the {@code GLESCapabilities} instance
*
* @throws IllegalStateException if no OpenGL ES context is current in the current thread
*/
public static GLESCapabilities createCapabilities() {
return createCapabilities(null);
}
/**
* Creates a new {@link GLESCapabilities} instance for the OpenGL ES context that is current in the current thread.
*
* This method calls {@link #setCapabilities(GLESCapabilities)} with the new instance before returning.
*
* @param bufferFactory a function that allocates a {@link PointerBuffer} given a size. The buffer must be filled with zeroes. If {@code null}, LWJGL will
* allocate a GC-managed buffer internally.
*
* @return the {@code GLESCapabilities} instance
*
* @throws IllegalStateException if no OpenGL ES context is current in the current thread
*/
public static GLESCapabilities createCapabilities(@Nullable IntFunction bufferFactory) {
FunctionProvider functionProvider = GLES.functionProvider;
if (functionProvider == null) {
throw new IllegalStateException("OpenGL ES library not been loaded.");
}
// We don't have a current GLESCapabilities when this method is called
// so we have to use the native bindings directly.
long GetError = functionProvider.getFunctionAddress("glGetError");
long GetString = functionProvider.getFunctionAddress("glGetString");
long GetIntegerv = functionProvider.getFunctionAddress("glGetIntegerv");
if (GetError == NULL || GetString == NULL || GetIntegerv == NULL) {
throw new IllegalStateException(
"Core OpenGL ES functions could not be found. Make sure that the OpenGL ES library has been loaded correctly."
);
}
int errorCode = callI(GetError);
if (errorCode != GL_NO_ERROR) {
apiLog(String.format("An OpenGL ES context was in an error state before the creation of its capabilities instance. Error: [0x%X]", errorCode));
}
int majorVersion;
int minorVersion;
try (MemoryStack stack = stackPush()) {
IntBuffer pi = stack.ints(0);
// Try the 3.0+ version query first
callPV(GL_MAJOR_VERSION, memAddress(pi), GetIntegerv);
if (callI(GetError) == GL_NO_ERROR && 3 <= (majorVersion = pi.get(0))) {
// We're on an 3.0+ context.
callPV(GL_MINOR_VERSION, memAddress(pi), GetIntegerv);
minorVersion = pi.get(0);
} else {
// Fallback to the string query.
String versionString = memUTF8Safe(callP(GL_VERSION, GetString));
if (versionString == null || callI(GetError) != GL_NO_ERROR) {
throw new IllegalStateException("There is no OpenGL ES context current in the current thread.");
}
APIVersion version = apiParseVersion(versionString);
majorVersion = version.major;
minorVersion = version.minor;
}
}
if (majorVersion < 2) {
throw new IllegalStateException("OpenGL ES 2.0 is required.");
}
int[] GL_VERSIONS = {
-1, // OpenGL ES 1.0 not supported
0, // OpenGL ES 2.0
2 // OpenGL ES 3.0 to 3.2
};
Set supportedExtensions = new HashSet<>(128);
int maxMajor = min(majorVersion, GL_VERSIONS.length);
if (MAX_VERSION != null) {
maxMajor = min(MAX_VERSION.major, maxMajor);
}
for (int M = 2; M <= maxMajor; M++) {
int maxMinor = GL_VERSIONS[M - 1];
if (M == majorVersion) {
maxMinor = min(minorVersion, maxMinor);
}
if (MAX_VERSION != null && M == MAX_VERSION.major) {
maxMinor = min(MAX_VERSION.minor, maxMinor);
}
for (int m = 0; m <= maxMinor; m++) {
supportedExtensions.add("GLES" + M + m);
}
}
if (majorVersion < 3) {
// Parse EXTENSIONS string
String extensionsString = memASCIISafe(callP(GL_EXTENSIONS, GetString));
if (extensionsString != null) {
StringTokenizer tokenizer = new StringTokenizer(extensionsString);
while (tokenizer.hasMoreTokens()) {
supportedExtensions.add(tokenizer.nextToken());
}
}
} else {
// Use indexed EXTENSIONS
int extensionCount;
try (MemoryStack stack = stackPush()) {
IntBuffer pi = stack.ints(0);
callPV(GL_NUM_EXTENSIONS, memAddress(pi), GetIntegerv);
extensionCount = pi.get(0);
}
long GetStringi = apiGetFunctionAddress(functionProvider, "glGetStringi");
for (int i = 0; i < extensionCount; i++) {
supportedExtensions.add(memASCII(callP(GL_EXTENSIONS, i, GetStringi)));
}
}
apiFilterExtensions(supportedExtensions, Configuration.OPENGLES_EXTENSION_FILTER);
GLESCapabilities caps = new GLESCapabilities(functionProvider, supportedExtensions, bufferFactory == null ? BufferUtils::createPointerBuffer : bufferFactory);
setCapabilities(caps);
return caps;
}
// Only used by array overloads
static GLESCapabilities getICD() {
return checkCapabilities(icd.get());
}
/** Function pointer provider. */
private interface ICD {
default void set(@Nullable GLESCapabilities caps) {}
@Nullable GLESCapabilities get();
}
/**
* Write-once {@link ICD}.
*
* This is the default implementation that skips the thread-local lookup. When a new GLESCapabilities is set, we compare it to the write-once
* capabilities. If different function pointers are found, we fall back to the expensive lookup.
*/
private static class ICDStatic implements ICD {
private static @Nullable GLESCapabilities tempCaps;
@SuppressWarnings("AssignmentToStaticFieldFromInstanceMethod")
@Override
public void set(@Nullable GLESCapabilities caps) {
if (tempCaps == null) {
tempCaps = caps;
} else if (caps != null && caps != tempCaps && ThreadLocalUtil.areCapabilitiesDifferent(tempCaps.addresses, caps.addresses)) {
apiLog("[WARNING] Incompatible context detected. Falling back to thread-local lookup for GLES contexts.");
icd = GLES::getCapabilities; // fall back to thread/process lookup
}
}
@Override
public GLESCapabilities get() {
return WriteOnce.caps;
}
private static final class WriteOnce {
// This will be initialized the first time get() above is called
static final GLESCapabilities caps;
static {
GLESCapabilities tempCaps = ICDStatic.tempCaps;
if (tempCaps == null) {
throw new IllegalStateException("No GLESCapabilities instance has been set");
}
caps = tempCaps;
}
}
}
}