2018-11-30 22:52:05 +03:00
|
|
|
/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
2012-05-21 15:12:37 +04:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
2010-06-10 02:27:29 +04:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
#if defined(MOZ_WIDGET_GTK)
|
2018-05-10 17:14:03 +03:00
|
|
|
#define GET_NATIVE_WINDOW_FROM_REAL_WIDGET(aWidget) \
|
|
|
|
((EGLNativeWindowType)aWidget->GetNativeData(NS_NATIVE_EGL_WINDOW))
|
|
|
|
#define GET_NATIVE_WINDOW_FROM_COMPOSITOR_WIDGET(aWidget) \
|
|
|
|
((EGLNativeWindowType)aWidget->RealWidget()->GetNativeData( \
|
|
|
|
NS_NATIVE_EGL_WINDOW))
|
2017-01-21 02:50:44 +03:00
|
|
|
#elif defined(MOZ_WIDGET_ANDROID)
|
2017-01-21 03:20:59 +03:00
|
|
|
#define GET_NATIVE_WINDOW_FROM_REAL_WIDGET(aWidget) \
|
|
|
|
((EGLNativeWindowType)aWidget->GetNativeData(NS_JAVA_SURFACE))
|
|
|
|
#define GET_NATIVE_WINDOW_FROM_COMPOSITOR_WIDGET(aWidget) \
|
|
|
|
(aWidget->AsAndroid()->GetEGLNativeWindow())
|
2017-03-29 17:14:19 +03:00
|
|
|
#elif defined(XP_WIN)
|
|
|
|
#define GET_NATIVE_WINDOW_FROM_REAL_WIDGET(aWidget) \
|
|
|
|
((EGLNativeWindowType)aWidget->GetNativeData(NS_NATIVE_WINDOW))
|
|
|
|
#define GET_NATIVE_WINDOW_FROM_COMPOSITOR_WIDGET(aWidget) \
|
|
|
|
((EGLNativeWindowType)aWidget->AsWindows()->GetHwnd())
|
2015-09-24 22:21:05 +03:00
|
|
|
#else
|
2017-01-21 03:20:59 +03:00
|
|
|
#define GET_NATIVE_WINDOW_FROM_REAL_WIDGET(aWidget) \
|
|
|
|
((EGLNativeWindowType)aWidget->GetNativeData(NS_NATIVE_WINDOW))
|
|
|
|
#define GET_NATIVE_WINDOW_FROM_COMPOSITOR_WIDGET(aWidget) \
|
|
|
|
((EGLNativeWindowType)aWidget->RealWidget()->GetNativeData(NS_NATIVE_WINDOW))
|
2016-06-08 16:38:45 +03:00
|
|
|
#endif
|
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
#if defined(XP_UNIX)
|
2016-06-08 16:38:45 +03:00
|
|
|
#ifdef MOZ_WIDGET_ANDROID
|
2016-08-20 00:17:24 +03:00
|
|
|
#include <android/native_window.h>
|
|
|
|
#include <android/native_window_jni.h>
|
2017-01-21 03:20:59 +03:00
|
|
|
#include "mozilla/widget/AndroidCompositorWidget.h"
|
2015-09-24 22:21:05 +03:00
|
|
|
#endif
|
2012-01-20 18:18:56 +04:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
#define GLES2_LIB "libGLESv2.so"
|
|
|
|
#define GLES2_LIB2 "libGLESv2.so.2"
|
2010-06-10 02:27:29 +04:00
|
|
|
|
2010-08-07 09:09:18 +04:00
|
|
|
#elif defined(XP_WIN)
|
2017-03-29 17:14:19 +03:00
|
|
|
#include "mozilla/widget/WinCompositorWidget.h"
|
2015-09-24 22:21:05 +03:00
|
|
|
#include "nsIFile.h"
|
2010-08-07 09:09:18 +04:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
#define GLES2_LIB "libGLESv2.dll"
|
2010-08-07 09:09:18 +04:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
#ifndef WIN32_LEAN_AND_MEAN
|
|
|
|
#define WIN32_LEAN_AND_MEAN 1
|
|
|
|
#endif
|
2010-08-07 09:09:18 +04:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
#include <windows.h>
|
2010-06-10 02:27:29 +04:00
|
|
|
#else
|
2015-09-24 22:21:05 +03:00
|
|
|
#error "Platform not recognized"
|
2010-06-10 02:27:29 +04:00
|
|
|
#endif
|
|
|
|
|
2010-06-23 19:50:51 +04:00
|
|
|
#include "gfxASurface.h"
|
2015-09-24 22:21:05 +03:00
|
|
|
#include "gfxCrashReporterUtils.h"
|
|
|
|
#include "gfxFailure.h"
|
2010-07-01 20:30:38 +04:00
|
|
|
#include "gfxPlatform.h"
|
2015-09-24 22:21:05 +03:00
|
|
|
#include "gfxUtils.h"
|
|
|
|
#include "GLBlitHelper.h"
|
|
|
|
#include "GLContextEGL.h"
|
2010-06-10 02:27:29 +04:00
|
|
|
#include "GLContextProvider.h"
|
2012-03-17 02:24:12 +04:00
|
|
|
#include "GLLibraryEGL.h"
|
2017-09-19 05:26:31 +03:00
|
|
|
#include "LayersLogging.h"
|
2015-09-24 22:21:05 +03:00
|
|
|
#include "mozilla/ArrayUtils.h"
|
|
|
|
#include "mozilla/Preferences.h"
|
2017-03-29 17:14:19 +03:00
|
|
|
#include "mozilla/gfx/gfxVars.h"
|
2017-01-13 23:16:52 +03:00
|
|
|
#include "mozilla/layers/CompositorOptions.h"
|
2016-07-25 21:41:00 +03:00
|
|
|
#include "mozilla/widget/CompositorWidget.h"
|
2010-06-10 02:27:29 +04:00
|
|
|
#include "nsDebug.h"
|
|
|
|
#include "nsIWidget.h"
|
2015-09-24 22:21:05 +03:00
|
|
|
#include "nsThreadUtils.h"
|
2013-12-01 02:20:57 +04:00
|
|
|
#include "ScopedGLHelpers.h"
|
2015-09-24 22:21:05 +03:00
|
|
|
#include "TextureImageEGL.h"
|
2013-12-01 02:20:57 +04:00
|
|
|
|
2018-05-22 11:34:32 +03:00
|
|
|
#if defined(MOZ_WAYLAND)
|
|
|
|
#include "nsAutoPtr.h"
|
|
|
|
#include "nsDataHashtable.h"
|
|
|
|
|
|
|
|
#include <gtk/gtk.h>
|
|
|
|
#include <gdk/gdkx.h>
|
|
|
|
#include <gdk/gdkwayland.h>
|
|
|
|
#include <wayland-egl.h>
|
|
|
|
#include <dlfcn.h>
|
|
|
|
#endif
|
|
|
|
|
2013-07-04 21:25:50 +04:00
|
|
|
using namespace mozilla::gfx;
|
2012-07-20 23:20:51 +04:00
|
|
|
|
2010-06-10 02:27:29 +04:00
|
|
|
namespace mozilla {
|
|
|
|
namespace gl {
|
|
|
|
|
2016-07-25 21:41:00 +03:00
|
|
|
using namespace mozilla::widget;
|
|
|
|
|
2018-05-22 11:34:32 +03:00
|
|
|
#if defined(MOZ_WAYLAND)
|
|
|
|
class WaylandGLSurface {
|
|
|
|
public:
|
|
|
|
WaylandGLSurface(struct wl_surface* aWaylandSurface,
|
|
|
|
struct wl_egl_window* aEGLWindow);
|
|
|
|
~WaylandGLSurface();
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-05-22 11:34:32 +03:00
|
|
|
private:
|
|
|
|
struct wl_surface* mWaylandSurface;
|
|
|
|
struct wl_egl_window* mEGLWindow;
|
|
|
|
};
|
|
|
|
|
|
|
|
static nsDataHashtable<nsPtrHashKey<void>, WaylandGLSurface*> sWaylandGLSurface;
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-05-22 11:34:32 +03:00
|
|
|
void DeleteWaylandGLSurface(EGLSurface surface) {
|
|
|
|
// We're running on Wayland which means our EGLSurface may
|
|
|
|
// have attached Wayland backend data which must be released.
|
2018-11-12 18:25:38 +03:00
|
|
|
if (!GDK_IS_X11_DISPLAY(gdk_display_get_default())) {
|
2018-05-22 11:34:32 +03:00
|
|
|
auto entry = sWaylandGLSurface.Lookup(surface);
|
|
|
|
if (entry) {
|
|
|
|
delete entry.Data();
|
|
|
|
entry.Remove();
|
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
2018-05-22 11:34:32 +03:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-01-07 01:07:12 +03:00
|
|
|
#define ADD_ATTR_2(_array, _k, _v) \
|
|
|
|
do { \
|
|
|
|
(_array).AppendElement(_k); \
|
|
|
|
(_array).AppendElement(_v); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define ADD_ATTR_1(_array, _k) \
|
|
|
|
do { \
|
|
|
|
(_array).AppendElement(_k); \
|
|
|
|
} while (0)
|
|
|
|
|
2017-06-14 13:32:04 +03:00
|
|
|
static bool CreateConfig(EGLConfig* aConfig, bool aEnableDepthBuffer);
|
2010-12-14 01:36:35 +03:00
|
|
|
|
2013-12-14 00:15:07 +04:00
|
|
|
// append three zeros at the end of attribs list to work around
|
|
|
|
// EGL implementation bugs that iterate until they find 0, instead of
|
|
|
|
// EGL_NONE. See bug 948406.
|
|
|
|
#define EGL_ATTRIBS_LIST_SAFE_TERMINATION_WORKING_AROUND_BUGS \
|
|
|
|
LOCAL_EGL_NONE, 0, 0, 0
|
|
|
|
|
2017-02-10 07:31:36 +03:00
|
|
|
static EGLint kTerminationAttribs[] = {
|
2013-12-14 00:15:07 +04:00
|
|
|
EGL_ATTRIBS_LIST_SAFE_TERMINATION_WORKING_AROUND_BUGS};
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2010-07-19 09:01:14 +04:00
|
|
|
static int next_power_of_two(int v) {
|
|
|
|
v--;
|
|
|
|
v |= v >> 1;
|
|
|
|
v |= v >> 2;
|
|
|
|
v |= v >> 4;
|
|
|
|
v |= v >> 8;
|
|
|
|
v |= v >> 16;
|
|
|
|
v++;
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2010-07-19 09:01:14 +04:00
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool is_power_of_two(int v) {
|
|
|
|
NS_ASSERTION(v >= 0, "bad value");
|
|
|
|
|
|
|
|
if (v == 0) return true;
|
|
|
|
|
|
|
|
return (v & (v - 1)) == 0;
|
|
|
|
}
|
|
|
|
|
2018-10-26 19:30:48 +03:00
|
|
|
static void DestroySurface(EGLSurface oldSurface) {
|
|
|
|
auto* egl = gl::GLLibraryEGL::Get();
|
2018-06-04 05:09:29 +03:00
|
|
|
|
2018-10-26 19:30:48 +03:00
|
|
|
if (oldSurface != EGL_NO_SURFACE) {
|
|
|
|
// TODO: This breaks TLS MakeCurrent caching.
|
|
|
|
egl->fMakeCurrent(EGL_DISPLAY(), EGL_NO_SURFACE, EGL_NO_SURFACE,
|
|
|
|
EGL_NO_CONTEXT);
|
|
|
|
egl->fDestroySurface(EGL_DISPLAY(), oldSurface);
|
2018-05-22 11:34:32 +03:00
|
|
|
#if defined(MOZ_WAYLAND)
|
2018-10-26 19:30:48 +03:00
|
|
|
DeleteWaylandGLSurface(oldSurface);
|
2018-05-22 11:34:32 +03:00
|
|
|
#endif
|
2018-10-26 19:30:48 +03:00
|
|
|
}
|
2013-11-15 20:28:54 +04:00
|
|
|
}
|
|
|
|
|
2018-04-25 22:30:27 +03:00
|
|
|
static EGLSurface CreateFallbackSurface(const EGLConfig& config) {
|
2018-06-29 00:01:03 +03:00
|
|
|
nsCString discardFailureId;
|
|
|
|
if (!GLLibraryEGL::EnsureInitialized(false, &discardFailureId)) {
|
|
|
|
gfxCriticalNote << "Failed to load EGL library 3!";
|
|
|
|
return EGL_NO_SURFACE;
|
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-06-04 05:09:29 +03:00
|
|
|
auto* egl = gl::GLLibraryEGL::Get();
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-06-04 05:09:29 +03:00
|
|
|
if (egl->IsExtensionSupported(GLLibraryEGL::KHR_surfaceless_context)) {
|
2018-04-25 22:30:27 +03:00
|
|
|
// We don't need a PBuffer surface in this case
|
|
|
|
return EGL_NO_SURFACE;
|
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-04-25 22:30:27 +03:00
|
|
|
std::vector<EGLint> pbattrs;
|
|
|
|
pbattrs.push_back(LOCAL_EGL_WIDTH);
|
|
|
|
pbattrs.push_back(1);
|
|
|
|
pbattrs.push_back(LOCAL_EGL_HEIGHT);
|
|
|
|
pbattrs.push_back(1);
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-04-25 22:30:27 +03:00
|
|
|
for (const auto& cur : kTerminationAttribs) {
|
|
|
|
pbattrs.push_back(cur);
|
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-06-04 05:09:29 +03:00
|
|
|
EGLSurface surface =
|
|
|
|
egl->fCreatePbufferSurface(EGL_DISPLAY(), config, pbattrs.data());
|
2018-04-25 22:30:27 +03:00
|
|
|
if (!surface) {
|
|
|
|
MOZ_CRASH("Failed to create fallback EGLSurface");
|
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-04-25 22:30:27 +03:00
|
|
|
return surface;
|
|
|
|
}
|
2013-11-15 20:28:54 +04:00
|
|
|
|
2018-04-25 22:30:27 +03:00
|
|
|
static EGLSurface CreateSurfaceFromNativeWindow(EGLNativeWindowType window,
|
|
|
|
const EGLConfig& config) {
|
2017-01-21 03:20:59 +03:00
|
|
|
MOZ_ASSERT(window);
|
2018-06-04 05:09:29 +03:00
|
|
|
auto* egl = gl::GLLibraryEGL::Get();
|
2018-04-25 22:30:27 +03:00
|
|
|
EGLSurface newSurface = EGL_NO_SURFACE;
|
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
#ifdef MOZ_WIDGET_ANDROID
|
2016-06-08 16:38:45 +03:00
|
|
|
JNIEnv* const env = jni::GetEnvForThread();
|
2016-08-20 00:17:24 +03:00
|
|
|
ANativeWindow* const nativeWindow =
|
2017-01-21 03:20:59 +03:00
|
|
|
ANativeWindow_fromSurface(env, reinterpret_cast<jobject>(window));
|
2018-06-04 05:09:29 +03:00
|
|
|
newSurface = egl->fCreateWindowSurface(egl->fGetDisplay(EGL_DEFAULT_DISPLAY),
|
2016-08-20 00:17:24 +03:00
|
|
|
config, nativeWindow, 0);
|
|
|
|
ANativeWindow_release(nativeWindow);
|
2015-09-24 22:21:05 +03:00
|
|
|
#else
|
2018-06-04 05:09:29 +03:00
|
|
|
newSurface = egl->fCreateWindowSurface(EGL_DISPLAY(), config, window, 0);
|
2015-09-24 22:21:05 +03:00
|
|
|
#endif
|
2013-11-15 20:28:54 +04:00
|
|
|
return newSurface;
|
|
|
|
}
|
|
|
|
|
2017-01-21 03:20:59 +03:00
|
|
|
/* GLContextEGLFactory class was added as a friend of GLContextEGL
|
|
|
|
* so that it could access GLContextEGL::CreateGLContext. This was
|
|
|
|
* done so that a new function would not need to be added to the shared
|
|
|
|
* GLContextProvider interface.
|
|
|
|
*/
|
|
|
|
class GLContextEGLFactory {
|
|
|
|
public:
|
2017-01-29 16:03:30 +03:00
|
|
|
static already_AddRefed<GLContext> Create(EGLNativeWindowType aWindow,
|
|
|
|
bool aWebRender);
|
2017-01-21 03:20:59 +03:00
|
|
|
|
2017-10-14 01:33:58 +03:00
|
|
|
private:
|
|
|
|
GLContextEGLFactory() {}
|
|
|
|
~GLContextEGLFactory() {}
|
2018-04-25 22:30:27 +03:00
|
|
|
};
|
2017-01-21 03:20:59 +03:00
|
|
|
|
2017-03-14 23:59:00 +03:00
|
|
|
already_AddRefed<GLContext> GLContextEGLFactory::Create(
|
|
|
|
EGLNativeWindowType aWindow, bool aWebRender) {
|
|
|
|
nsCString discardFailureId;
|
2017-01-21 03:20:59 +03:00
|
|
|
if (!GLLibraryEGL::EnsureInitialized(false, &discardFailureId)) {
|
2017-09-19 05:26:31 +03:00
|
|
|
gfxCriticalNote << "Failed to load EGL library 3!";
|
2017-01-21 03:20:59 +03:00
|
|
|
return nullptr;
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
|
|
|
|
2017-01-21 03:20:59 +03:00
|
|
|
auto* egl = gl::GLLibraryEGL::Get();
|
|
|
|
bool doubleBuffered = true;
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2017-01-21 03:20:59 +03:00
|
|
|
EGLConfig config;
|
|
|
|
if (aWebRender && egl->IsANGLE()) {
|
2017-10-14 01:33:58 +03:00
|
|
|
// Force enable alpha channel to make sure ANGLE use correct framebuffer
|
|
|
|
// formart
|
2017-09-19 05:26:31 +03:00
|
|
|
const int bpp = 32;
|
2017-10-14 01:33:58 +03:00
|
|
|
const bool withDepth = true;
|
2017-09-19 05:26:31 +03:00
|
|
|
if (!CreateConfig(&config, bpp, withDepth)) {
|
|
|
|
gfxCriticalNote << "Failed to create EGLConfig for WebRender ANGLE!";
|
2017-01-21 03:20:59 +03:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
} else {
|
2018-06-04 05:09:29 +03:00
|
|
|
if (!CreateConfig(&config, aWebRender)) {
|
|
|
|
gfxCriticalNote << "Failed to create EGLConfig!";
|
2017-10-14 01:33:58 +03:00
|
|
|
return nullptr;
|
2017-11-24 03:09:29 +03:00
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
|
|
|
|
2018-04-25 22:30:27 +03:00
|
|
|
EGLSurface surface = EGL_NO_SURFACE;
|
|
|
|
if (aWindow) {
|
|
|
|
surface = mozilla::gl::CreateSurfaceFromNativeWindow(aWindow, config);
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
|
|
|
|
2017-01-29 16:03:30 +03:00
|
|
|
CreateContextFlags flags = CreateContextFlags::NONE;
|
|
|
|
if (aWebRender) {
|
|
|
|
flags |= CreateContextFlags::PREFER_ES3;
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
2017-01-21 03:20:59 +03:00
|
|
|
SurfaceCaps caps = SurfaceCaps::Any();
|
2017-03-14 23:59:00 +03:00
|
|
|
RefPtr<GLContextEGL> gl = GLContextEGL::CreateGLContext(
|
2015-11-27 22:02:52 +03:00
|
|
|
flags, caps, false, config, surface, &discardFailureId);
|
2015-09-24 22:21:05 +03:00
|
|
|
if (!gl) {
|
2017-10-14 01:33:58 +03:00
|
|
|
gfxCriticalNote << "Failed to create EGLContext!";
|
2018-10-26 19:30:48 +03:00
|
|
|
mozilla::gl::DestroySurface(surface);
|
2017-10-14 01:33:58 +03:00
|
|
|
return nullptr;
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
|
|
|
|
2017-01-21 03:20:59 +03:00
|
|
|
gl->MakeCurrent();
|
|
|
|
gl->SetIsDoubleBuffered(doubleBuffered);
|
2018-06-04 05:09:29 +03:00
|
|
|
if (aWebRender && egl->IsANGLE()) {
|
2017-11-24 03:09:29 +03:00
|
|
|
MOZ_ASSERT(doubleBuffered);
|
2018-06-04 05:09:29 +03:00
|
|
|
egl->fSwapInterval(EGL_DISPLAY(), 0);
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
2017-01-21 03:20:59 +03:00
|
|
|
return gl.forget();
|
|
|
|
}
|
|
|
|
|
2016-06-07 03:31:53 +03:00
|
|
|
GLContextEGL::GLContextEGL(CreateContextFlags flags, const SurfaceCaps& caps,
|
2017-03-14 23:59:00 +03:00
|
|
|
bool isOffscreen, EGLConfig config,
|
|
|
|
EGLSurface surface, EGLContext context)
|
2017-12-22 06:19:03 +03:00
|
|
|
: GLContext(flags, caps, nullptr, isOffscreen, false),
|
2014-01-08 00:02:18 +04:00
|
|
|
mConfig(config),
|
2018-06-04 05:09:52 +03:00
|
|
|
mEgl(gl::GLLibraryEGL::Get()),
|
2014-01-08 00:02:18 +04:00
|
|
|
mSurface(surface),
|
2018-04-25 22:30:27 +03:00
|
|
|
mFallbackSurface(CreateFallbackSurface(config)),
|
2015-09-23 05:26:13 +03:00
|
|
|
mContext(context) {
|
2012-02-22 01:15:39 +04:00
|
|
|
#ifdef DEBUG
|
2014-01-08 00:02:18 +04:00
|
|
|
printf_stderr("Initializing context %p surface %p on display %p\n", mContext,
|
|
|
|
mSurface, EGL_DISPLAY());
|
2012-08-24 23:42:45 +04:00
|
|
|
#endif
|
2014-01-08 00:02:18 +04:00
|
|
|
}
|
2010-06-10 02:27:29 +04:00
|
|
|
|
2019-01-16 11:50:07 +03:00
|
|
|
void GLContextEGL::OnMarkDestroyed() {
|
2019-01-11 03:04:26 +03:00
|
|
|
if (mSurfaceOverride != EGL_NO_SURFACE) {
|
|
|
|
SetEGLSurfaceOverride(EGL_NO_SURFACE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
GLContextEGL::~GLContextEGL() {
|
|
|
|
MarkDestroyed();
|
2010-08-07 09:09:18 +04:00
|
|
|
|
2014-04-15 18:57:26 +04:00
|
|
|
// Wrapped context should not destroy eglContext/Surface
|
|
|
|
if (!mOwnsContext) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-02-22 01:15:39 +04:00
|
|
|
#ifdef DEBUG
|
2014-01-08 00:02:18 +04:00
|
|
|
printf_stderr("Destroying context %p surface %p on display %p\n", mContext,
|
|
|
|
mSurface, EGL_DISPLAY());
|
2010-09-13 19:53:52 +04:00
|
|
|
#endif
|
2010-09-10 00:40:11 +04:00
|
|
|
|
2018-06-04 05:09:52 +03:00
|
|
|
mEgl->fDestroyContext(EGL_DISPLAY(), mContext);
|
2018-10-26 19:30:48 +03:00
|
|
|
|
|
|
|
mozilla::gl::DestroySurface(mSurface);
|
|
|
|
mozilla::gl::DestroySurface(mFallbackSurface);
|
2014-01-08 00:02:18 +04:00
|
|
|
}
|
2010-06-10 02:27:29 +04:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
bool GLContextEGL::Init() {
|
2012-03-01 01:55:46 +04:00
|
|
|
#if defined(ANDROID)
|
2014-01-08 00:02:18 +04:00
|
|
|
// We can't use LoadApitraceLibrary here because the GLContext
|
|
|
|
// expects its own handle to the GL library
|
|
|
|
if (!OpenLibrary(APITRACE_LIB))
|
2012-03-01 01:55:46 +04:00
|
|
|
#endif
|
2014-01-08 00:02:18 +04:00
|
|
|
if (!OpenLibrary(GLES2_LIB)) {
|
2012-01-20 18:18:56 +04:00
|
|
|
#if defined(XP_UNIX)
|
2014-01-08 00:02:18 +04:00
|
|
|
if (!OpenLibrary(GLES2_LIB2)) {
|
|
|
|
NS_WARNING("Couldn't load GLES2 LIB.");
|
|
|
|
return false;
|
2012-03-01 01:55:46 +04:00
|
|
|
}
|
2014-01-08 00:02:18 +04:00
|
|
|
#endif
|
2011-08-31 18:27:05 +04:00
|
|
|
}
|
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
SetupLookupFunction();
|
|
|
|
if (!InitWithPrefix("gl", true)) return false;
|
2018-10-26 19:30:48 +03:00
|
|
|
|
|
|
|
bool current = MakeCurrent();
|
|
|
|
if (!current) {
|
|
|
|
gfx::LogFailure(
|
|
|
|
NS_LITERAL_CSTRING("Couldn't get device attachments for device."));
|
|
|
|
return false;
|
|
|
|
}
|
2010-09-10 00:40:11 +04:00
|
|
|
|
2016-08-23 18:24:54 +03:00
|
|
|
static_assert(sizeof(GLint) >= sizeof(int32_t),
|
|
|
|
"GLint is smaller than int32_t");
|
2014-01-08 00:02:18 +04:00
|
|
|
mMaxTextureImageSize = INT32_MAX;
|
2010-09-10 00:40:11 +04:00
|
|
|
|
2018-06-04 05:09:52 +03:00
|
|
|
mShareWithEGLImage = mEgl->HasKHRImageBase() &&
|
|
|
|
mEgl->HasKHRImageTexture2D() &&
|
2018-06-04 05:09:29 +03:00
|
|
|
IsExtensionSupported(OES_EGL_image);
|
2010-06-10 02:27:29 +04:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
return true;
|
|
|
|
}
|
2011-12-15 04:57:09 +04:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
bool GLContextEGL::BindTexImage() {
|
|
|
|
if (!mSurface) return false;
|
2011-11-19 07:57:29 +04:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
if (mBound && !ReleaseTexImage()) return false;
|
2010-06-23 13:24:31 +04:00
|
|
|
|
2018-06-04 05:09:52 +03:00
|
|
|
EGLBoolean success = mEgl->fBindTexImage(EGL_DISPLAY(), (EGLSurface)mSurface,
|
2014-01-08 00:02:18 +04:00
|
|
|
LOCAL_EGL_BACK_BUFFER);
|
|
|
|
if (success == LOCAL_EGL_FALSE) return false;
|
2010-06-24 14:04:28 +04:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
mBound = true;
|
|
|
|
return true;
|
|
|
|
}
|
2010-06-23 13:24:31 +04:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
bool GLContextEGL::ReleaseTexImage() {
|
|
|
|
if (!mBound) return true;
|
2010-06-23 13:24:31 +04:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
if (!mSurface) return false;
|
2010-06-23 13:24:31 +04:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
EGLBoolean success;
|
2018-06-04 05:09:52 +03:00
|
|
|
success = mEgl->fReleaseTexImage(EGL_DISPLAY(), (EGLSurface)mSurface,
|
|
|
|
LOCAL_EGL_BACK_BUFFER);
|
2014-01-08 00:02:18 +04:00
|
|
|
if (success == LOCAL_EGL_FALSE) return false;
|
2010-06-23 13:24:31 +04:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
mBound = false;
|
|
|
|
return true;
|
|
|
|
}
|
2010-06-23 13:24:31 +04:00
|
|
|
|
2018-10-26 19:30:48 +03:00
|
|
|
void GLContextEGL::SetEGLSurfaceOverride(EGLSurface surf) {
|
2014-01-08 00:02:18 +04:00
|
|
|
if (Screen()) {
|
|
|
|
/* Blit `draw` to `read` if we need to, before we potentially juggle
|
|
|
|
* `read` around. If we don't, we might attach a different `read`,
|
|
|
|
* and *then* hit AssureBlitted, which will blit a dirty `draw` onto
|
|
|
|
* the wrong `read`!
|
|
|
|
*/
|
|
|
|
Screen()->AssureBlitted();
|
2010-06-23 13:24:31 +04:00
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2015-04-08 18:35:54 +03:00
|
|
|
mSurfaceOverride = surf;
|
2018-10-26 19:30:48 +03:00
|
|
|
DebugOnly<bool> ok = MakeCurrent(true);
|
|
|
|
MOZ_ASSERT(ok);
|
2014-01-08 00:02:18 +04:00
|
|
|
}
|
2013-02-14 03:26:24 +04:00
|
|
|
|
2017-08-15 23:21:37 +03:00
|
|
|
bool GLContextEGL::MakeCurrentImpl() const {
|
2018-10-26 19:30:48 +03:00
|
|
|
EGLSurface surface =
|
|
|
|
(mSurfaceOverride != EGL_NO_SURFACE) ? mSurfaceOverride : mSurface;
|
2018-04-25 22:30:27 +03:00
|
|
|
if (!surface) {
|
|
|
|
surface = mFallbackSurface;
|
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-06-04 05:09:52 +03:00
|
|
|
const bool succeeded =
|
|
|
|
mEgl->fMakeCurrent(EGL_DISPLAY(), surface, surface, mContext);
|
2017-08-15 23:21:37 +03:00
|
|
|
if (!succeeded) {
|
2018-06-04 05:09:52 +03:00
|
|
|
const auto eglError = mEgl->fGetError();
|
2017-08-15 23:21:37 +03:00
|
|
|
if (eglError == LOCAL_EGL_CONTEXT_LOST) {
|
2018-11-27 07:23:19 +03:00
|
|
|
OnContextLostError();
|
2017-08-15 23:21:37 +03:00
|
|
|
} else {
|
|
|
|
NS_WARNING("Failed to make GL context current!");
|
2012-08-01 22:57:54 +04:00
|
|
|
#ifdef DEBUG
|
2017-08-15 23:21:37 +03:00
|
|
|
printf_stderr("EGL Error: 0x%04x\n", eglError);
|
2012-08-01 22:57:54 +04:00
|
|
|
#endif
|
2010-06-10 02:27:29 +04:00
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
2010-06-10 02:27:29 +04:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
return succeeded;
|
|
|
|
}
|
|
|
|
|
2017-08-11 23:54:40 +03:00
|
|
|
bool GLContextEGL::IsCurrentImpl() const {
|
2018-06-04 05:09:52 +03:00
|
|
|
return mEgl->fGetCurrentContext() == mContext;
|
2014-01-08 00:02:18 +04:00
|
|
|
}
|
2012-08-22 07:30:20 +04:00
|
|
|
|
2018-10-26 19:30:48 +03:00
|
|
|
bool GLContextEGL::RenewSurface(CompositorWidget* aWidget) {
|
2014-04-15 18:57:26 +04:00
|
|
|
if (!mOwnsContext) {
|
|
|
|
return false;
|
|
|
|
}
|
2014-01-08 00:02:18 +04:00
|
|
|
// unconditionally release the surface and create a new one. Don't try to
|
|
|
|
// optimize this away. If we get here, then by definition we know that we want
|
|
|
|
// to get a new surface.
|
|
|
|
ReleaseSurface();
|
2017-01-21 02:31:02 +03:00
|
|
|
MOZ_ASSERT(aWidget);
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-04-25 22:30:27 +03:00
|
|
|
EGLNativeWindowType nativeWindow =
|
|
|
|
GET_NATIVE_WINDOW_FROM_COMPOSITOR_WIDGET(aWidget);
|
|
|
|
if (nativeWindow) {
|
|
|
|
mSurface =
|
|
|
|
mozilla::gl::CreateSurfaceFromNativeWindow(nativeWindow, mConfig);
|
|
|
|
if (!mSurface) {
|
|
|
|
NS_WARNING("Failed to create EGLSurface from native window");
|
|
|
|
return false;
|
2010-12-14 01:36:35 +03:00
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
2018-04-25 22:30:27 +03:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
return MakeCurrent(true);
|
|
|
|
}
|
2010-12-14 01:36:35 +03:00
|
|
|
|
2018-10-26 19:30:48 +03:00
|
|
|
void GLContextEGL::ReleaseSurface() {
|
|
|
|
if (mOwnsContext) {
|
|
|
|
mozilla::gl::DestroySurface(mSurface);
|
|
|
|
}
|
|
|
|
if (mSurface == mSurfaceOverride) {
|
|
|
|
mSurfaceOverride = EGL_NO_SURFACE;
|
|
|
|
}
|
|
|
|
mSurface = EGL_NO_SURFACE;
|
2014-01-08 00:02:18 +04:00
|
|
|
}
|
2011-08-24 00:48:27 +04:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
bool GLContextEGL::SetupLookupFunction() {
|
2018-06-04 05:09:52 +03:00
|
|
|
mLookupFunc = mEgl->GetLookupFunction();
|
2014-01-08 00:02:18 +04:00
|
|
|
return true;
|
|
|
|
}
|
2010-06-10 02:27:29 +04:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
bool GLContextEGL::SwapBuffers() {
|
2015-04-08 18:35:54 +03:00
|
|
|
EGLSurface surface =
|
|
|
|
mSurfaceOverride != EGL_NO_SURFACE ? mSurfaceOverride : mSurface;
|
|
|
|
if (surface) {
|
2018-06-04 05:09:52 +03:00
|
|
|
return mEgl->fSwapBuffers(EGL_DISPLAY(), surface);
|
2014-01-08 00:02:18 +04:00
|
|
|
} else {
|
|
|
|
return false;
|
2010-06-26 04:52:37 +04:00
|
|
|
}
|
2014-01-08 00:02:18 +04:00
|
|
|
}
|
2013-12-03 22:44:38 +04:00
|
|
|
|
2017-01-31 05:58:52 +03:00
|
|
|
void GLContextEGL::GetWSIInfo(nsCString* const out) const {
|
|
|
|
out->AppendLiteral("EGL_VENDOR: ");
|
2018-06-04 05:09:52 +03:00
|
|
|
out->Append((const char*)mEgl->fQueryString(EGL_DISPLAY(), LOCAL_EGL_VENDOR));
|
2017-01-31 05:58:52 +03:00
|
|
|
|
|
|
|
out->AppendLiteral("\nEGL_VERSION: ");
|
2018-06-04 05:09:52 +03:00
|
|
|
out->Append(
|
|
|
|
(const char*)mEgl->fQueryString(EGL_DISPLAY(), LOCAL_EGL_VERSION));
|
2017-01-31 05:58:52 +03:00
|
|
|
|
|
|
|
out->AppendLiteral("\nEGL_EXTENSIONS: ");
|
2018-06-04 05:09:52 +03:00
|
|
|
out->Append(
|
|
|
|
(const char*)mEgl->fQueryString(EGL_DISPLAY(), LOCAL_EGL_EXTENSIONS));
|
2017-01-31 05:58:52 +03:00
|
|
|
|
|
|
|
#ifndef ANDROID // This query will crash some old android.
|
|
|
|
out->AppendLiteral("\nEGL_EXTENSIONS(nullptr): ");
|
2018-06-04 05:09:52 +03:00
|
|
|
out->Append((const char*)mEgl->fQueryString(nullptr, LOCAL_EGL_EXTENSIONS));
|
2017-01-31 05:58:52 +03:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
// hold a reference to the given surface
|
|
|
|
// for the lifetime of this context.
|
2016-06-11 04:37:07 +03:00
|
|
|
void GLContextEGL::HoldSurface(gfxASurface* aSurf) { mThebesSurface = aSurf; }
|
2010-07-19 09:01:14 +04:00
|
|
|
|
2018-09-27 08:31:30 +03:00
|
|
|
#define LOCAL_EGL_CONTEXT_PROVOKING_VERTEX_DONT_CARE_MOZ 0x6000
|
|
|
|
|
2015-11-27 22:02:52 +03:00
|
|
|
already_AddRefed<GLContextEGL> GLContextEGL::CreateGLContext(
|
|
|
|
CreateContextFlags flags, const SurfaceCaps& caps, bool isOffscreen,
|
2016-06-11 05:01:00 +03:00
|
|
|
EGLConfig config, EGLSurface surface, nsACString* const out_failureId) {
|
2018-06-04 05:09:29 +03:00
|
|
|
auto* egl = gl::GLLibraryEGL::Get();
|
2010-07-19 09:01:14 +04:00
|
|
|
|
2018-06-04 05:09:29 +03:00
|
|
|
if (egl->fBindAPI(LOCAL_EGL_OPENGL_ES_API) == LOCAL_EGL_FALSE) {
|
2016-06-11 05:01:00 +03:00
|
|
|
*out_failureId = NS_LITERAL_CSTRING("FEATURE_FAILURE_EGL_ES");
|
2014-01-08 00:02:18 +04:00
|
|
|
NS_WARNING("Failed to bind API to GLES!");
|
|
|
|
return nullptr;
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
|
|
|
|
2017-03-14 23:39:39 +03:00
|
|
|
std::vector<EGLint> required_attribs;
|
|
|
|
required_attribs.push_back(LOCAL_EGL_CONTEXT_CLIENT_VERSION);
|
|
|
|
if (flags & CreateContextFlags::PREFER_ES3) {
|
|
|
|
required_attribs.push_back(3);
|
|
|
|
} else {
|
|
|
|
required_attribs.push_back(2);
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
|
|
|
|
2018-07-04 01:40:58 +03:00
|
|
|
const auto debugFlags = GLContext::ChooseDebugFlags(flags);
|
|
|
|
if (!debugFlags && flags & CreateContextFlags::NO_VALIDATION &&
|
|
|
|
egl->IsExtensionSupported(GLLibraryEGL::KHR_create_context_no_error)) {
|
|
|
|
required_attribs.push_back(LOCAL_EGL_CONTEXT_OPENGL_NO_ERROR_KHR);
|
|
|
|
required_attribs.push_back(LOCAL_EGL_TRUE);
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
|
|
|
|
2018-09-27 08:31:30 +03:00
|
|
|
if (flags & CreateContextFlags::PROVOKING_VERTEX_DONT_CARE &&
|
2018-07-04 01:40:58 +03:00
|
|
|
egl->IsExtensionSupported(
|
2018-09-27 08:31:30 +03:00
|
|
|
GLLibraryEGL::MOZ_create_context_provoking_vertex_dont_care)) {
|
2018-07-04 01:40:58 +03:00
|
|
|
required_attribs.push_back(
|
2018-09-27 08:31:30 +03:00
|
|
|
LOCAL_EGL_CONTEXT_PROVOKING_VERTEX_DONT_CARE_MOZ);
|
2018-07-04 01:40:58 +03:00
|
|
|
required_attribs.push_back(LOCAL_EGL_TRUE);
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
|
|
|
|
2017-03-14 23:39:39 +03:00
|
|
|
std::vector<EGLint> robustness_attribs;
|
|
|
|
std::vector<EGLint> rbab_attribs; // RBAB: Robust Buffer Access Behavior
|
2017-03-09 23:30:01 +03:00
|
|
|
if (flags & CreateContextFlags::PREFER_ROBUSTNESS) {
|
2018-06-04 05:09:29 +03:00
|
|
|
if (egl->IsExtensionSupported(
|
|
|
|
GLLibraryEGL::EXT_create_context_robustness)) {
|
2017-03-14 23:39:39 +03:00
|
|
|
robustness_attribs = required_attribs;
|
|
|
|
robustness_attribs.push_back(
|
|
|
|
LOCAL_EGL_CONTEXT_OPENGL_RESET_NOTIFICATION_STRATEGY_EXT);
|
|
|
|
robustness_attribs.push_back(LOCAL_EGL_LOSE_CONTEXT_ON_RESET_EXT);
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-10-12 18:57:32 +03:00
|
|
|
// Don't enable robust buffer access on Adreno 630 devices.
|
|
|
|
// It causes the linking of some shaders to fail. See bug 1485441.
|
|
|
|
nsCOMPtr<nsIGfxInfo> gfxInfo = services::GetGfxInfo();
|
|
|
|
nsAutoString renderer;
|
|
|
|
gfxInfo->GetAdapterDeviceID(renderer);
|
|
|
|
if (renderer.Find("Adreno (TM) 630") == -1) {
|
|
|
|
rbab_attribs = robustness_attribs;
|
|
|
|
rbab_attribs.push_back(LOCAL_EGL_CONTEXT_OPENGL_ROBUST_ACCESS_EXT);
|
|
|
|
rbab_attribs.push_back(LOCAL_EGL_TRUE);
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
2018-07-21 04:43:33 +03:00
|
|
|
} else if (egl->IsExtensionSupported(GLLibraryEGL::KHR_create_context)) {
|
2017-03-14 23:39:39 +03:00
|
|
|
robustness_attribs = required_attribs;
|
|
|
|
robustness_attribs.push_back(
|
2018-07-21 04:43:33 +03:00
|
|
|
LOCAL_EGL_CONTEXT_OPENGL_RESET_NOTIFICATION_STRATEGY_KHR);
|
|
|
|
robustness_attribs.push_back(LOCAL_EGL_LOSE_CONTEXT_ON_RESET_KHR);
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-10-12 18:57:32 +03:00
|
|
|
rbab_attribs = robustness_attribs;
|
2017-03-14 23:39:39 +03:00
|
|
|
rbab_attribs.push_back(LOCAL_EGL_CONTEXT_FLAGS_KHR);
|
|
|
|
rbab_attribs.push_back(LOCAL_EGL_CONTEXT_OPENGL_ROBUST_ACCESS_BIT_KHR);
|
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
2017-02-10 07:31:36 +03:00
|
|
|
|
2018-07-04 01:40:58 +03:00
|
|
|
const auto fnCreate = [&](const std::vector<EGLint>& attribs) {
|
|
|
|
auto terminated_attribs = attribs;
|
|
|
|
|
2018-09-27 08:31:30 +03:00
|
|
|
for (const auto& cur : kTerminationAttribs) {
|
|
|
|
terminated_attribs.push_back(cur);
|
|
|
|
}
|
|
|
|
|
2018-06-04 05:09:29 +03:00
|
|
|
return egl->fCreateContext(EGL_DISPLAY(), config, EGL_NO_CONTEXT,
|
2018-07-21 04:43:33 +03:00
|
|
|
terminated_attribs.data());
|
|
|
|
};
|
2015-11-27 01:57:30 +03:00
|
|
|
|
2017-03-14 23:39:39 +03:00
|
|
|
EGLContext context;
|
|
|
|
do {
|
|
|
|
if (!rbab_attribs.empty()) {
|
|
|
|
context = fnCreate(rbab_attribs);
|
|
|
|
if (context) break;
|
|
|
|
NS_WARNING("Failed to create EGLContext with rbab_attribs");
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
2017-03-14 23:39:39 +03:00
|
|
|
|
|
|
|
if (!robustness_attribs.empty()) {
|
|
|
|
context = fnCreate(robustness_attribs);
|
|
|
|
if (context) break;
|
|
|
|
NS_WARNING("Failed to create EGLContext with robustness_attribs");
|
|
|
|
}
|
|
|
|
|
|
|
|
context = fnCreate(required_attribs);
|
|
|
|
if (context) break;
|
|
|
|
NS_WARNING("Failed to create EGLContext with required_attribs");
|
|
|
|
|
|
|
|
*out_failureId = NS_LITERAL_CSTRING("FEATURE_FAILURE_EGL_CREATE");
|
|
|
|
return nullptr;
|
|
|
|
} while (false);
|
|
|
|
MOZ_ASSERT(context);
|
|
|
|
|
2017-03-14 23:59:00 +03:00
|
|
|
RefPtr<GLContextEGL> glContext =
|
|
|
|
new GLContextEGL(flags, caps, isOffscreen, config, surface, context);
|
2016-06-06 23:52:42 +03:00
|
|
|
if (!glContext->Init()) {
|
2016-06-11 05:01:00 +03:00
|
|
|
*out_failureId = NS_LITERAL_CSTRING("FEATURE_FAILURE_EGL_INIT");
|
2014-01-08 00:02:18 +04:00
|
|
|
return nullptr;
|
2016-06-06 23:52:42 +03:00
|
|
|
}
|
2010-07-19 09:01:14 +04:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
return glContext.forget();
|
|
|
|
}
|
2011-02-01 00:10:57 +03:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
EGLSurface GLContextEGL::CreatePBufferSurfaceTryingPowerOfTwo(
|
|
|
|
EGLConfig config, EGLenum bindToTextureFormat,
|
2015-05-28 12:45:33 +03:00
|
|
|
mozilla::gfx::IntSize& pbsize) {
|
2014-01-08 00:02:18 +04:00
|
|
|
nsTArray<EGLint> pbattrs(16);
|
|
|
|
EGLSurface surface = nullptr;
|
2011-02-01 00:10:57 +03:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
TRY_AGAIN_POWER_OF_TWO:
|
|
|
|
pbattrs.Clear();
|
|
|
|
pbattrs.AppendElement(LOCAL_EGL_WIDTH);
|
|
|
|
pbattrs.AppendElement(pbsize.width);
|
|
|
|
pbattrs.AppendElement(LOCAL_EGL_HEIGHT);
|
|
|
|
pbattrs.AppendElement(pbsize.height);
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
if (bindToTextureFormat != LOCAL_EGL_NONE) {
|
|
|
|
pbattrs.AppendElement(LOCAL_EGL_TEXTURE_TARGET);
|
|
|
|
pbattrs.AppendElement(LOCAL_EGL_TEXTURE_2D);
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
pbattrs.AppendElement(LOCAL_EGL_TEXTURE_FORMAT);
|
|
|
|
pbattrs.AppendElement(bindToTextureFormat);
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
|
|
|
|
2017-02-10 07:31:36 +03:00
|
|
|
for (const auto& cur : kTerminationAttribs) {
|
|
|
|
pbattrs.AppendElement(cur);
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
|
|
|
|
2018-06-04 05:09:29 +03:00
|
|
|
auto* egl = gl::GLLibraryEGL::Get();
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-06-04 05:09:29 +03:00
|
|
|
surface = egl->fCreatePbufferSurface(EGL_DISPLAY(), config, &pbattrs[0]);
|
2014-01-08 00:02:18 +04:00
|
|
|
if (!surface) {
|
|
|
|
if (!is_power_of_two(pbsize.width) || !is_power_of_two(pbsize.height)) {
|
|
|
|
if (!is_power_of_two(pbsize.width))
|
|
|
|
pbsize.width = next_power_of_two(pbsize.width);
|
|
|
|
if (!is_power_of_two(pbsize.height))
|
|
|
|
pbsize.height = next_power_of_two(pbsize.height);
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
NS_WARNING("Failed to create pbuffer, trying power of two dims");
|
|
|
|
goto TRY_AGAIN_POWER_OF_TWO;
|
|
|
|
}
|
2011-02-01 00:10:57 +03:00
|
|
|
|
2017-02-10 07:31:36 +03:00
|
|
|
NS_WARNING("Failed to create pbuffer surface");
|
|
|
|
return nullptr;
|
2014-01-08 00:02:18 +04:00
|
|
|
}
|
2011-02-01 00:10:57 +03:00
|
|
|
|
2018-06-04 05:09:29 +03:00
|
|
|
return surface;
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
2018-06-04 05:09:29 +03:00
|
|
|
|
|
|
|
#if defined(MOZ_WAYLAND)
|
|
|
|
WaylandGLSurface::WaylandGLSurface(struct wl_surface* aWaylandSurface,
|
|
|
|
struct wl_egl_window* aEGLWindow)
|
2014-01-08 00:02:18 +04:00
|
|
|
: mWaylandSurface(aWaylandSurface), mEGLWindow(aEGLWindow) {}
|
|
|
|
|
|
|
|
WaylandGLSurface::~WaylandGLSurface() {
|
|
|
|
wl_egl_window_destroy(mEGLWindow);
|
|
|
|
wl_surface_destroy(mWaylandSurface);
|
2011-02-01 00:10:57 +03:00
|
|
|
}
|
|
|
|
|
2014-01-08 00:02:18 +04:00
|
|
|
EGLSurface GLContextEGL::CreateWaylandBufferSurface(
|
|
|
|
EGLConfig config, mozilla::gfx::IntSize& pbsize) {
|
|
|
|
// Available as of GTK 3.8+
|
|
|
|
static auto sGdkWaylandDisplayGetWlCompositor =
|
2018-05-22 11:34:32 +03:00
|
|
|
(wl_compositor * (*)(GdkDisplay*))
|
2014-01-08 00:02:18 +04:00
|
|
|
dlsym(RTLD_DEFAULT, "gdk_wayland_display_get_wl_compositor");
|
|
|
|
|
|
|
|
if (!sGdkWaylandDisplayGetWlCompositor) return nullptr;
|
2010-06-10 02:27:29 +04:00
|
|
|
|
2018-05-22 11:34:32 +03:00
|
|
|
struct wl_compositor* compositor =
|
|
|
|
sGdkWaylandDisplayGetWlCompositor(gdk_display_get_default());
|
|
|
|
struct wl_surface* wlsurface = wl_compositor_create_surface(compositor);
|
|
|
|
struct wl_egl_window* eglwindow =
|
|
|
|
wl_egl_window_create(wlsurface, pbsize.width, pbsize.height);
|
|
|
|
|
|
|
|
auto* egl = gl::GLLibraryEGL::Get();
|
|
|
|
EGLSurface surface =
|
|
|
|
egl->fCreateWindowSurface(EGL_DISPLAY(), config, eglwindow, 0);
|
|
|
|
|
|
|
|
if (surface) {
|
|
|
|
WaylandGLSurface* waylandData = new WaylandGLSurface(wlsurface, eglwindow);
|
|
|
|
auto entry = sWaylandGLSurface.LookupForAdd(surface);
|
|
|
|
entry.OrInsert([&waylandData]() { return waylandData; });
|
|
|
|
}
|
|
|
|
|
|
|
|
return surface;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2015-08-28 00:12:06 +03:00
|
|
|
static const EGLint kEGLConfigAttribsOffscreenPBuffer[] = {
|
|
|
|
LOCAL_EGL_SURFACE_TYPE, LOCAL_EGL_PBUFFER_BIT, LOCAL_EGL_RENDERABLE_TYPE,
|
|
|
|
LOCAL_EGL_OPENGL_ES2_BIT,
|
|
|
|
// Old versions of llvmpipe seem to need this to properly create the pbuffer
|
|
|
|
// (bug 981856)
|
|
|
|
LOCAL_EGL_RED_SIZE, 8, LOCAL_EGL_GREEN_SIZE, 8, LOCAL_EGL_BLUE_SIZE, 8,
|
|
|
|
LOCAL_EGL_ALPHA_SIZE, 0,
|
|
|
|
EGL_ATTRIBS_LIST_SAFE_TERMINATION_WORKING_AROUND_BUGS};
|
|
|
|
|
2011-09-23 09:58:15 +04:00
|
|
|
static const EGLint kEGLConfigAttribsRGB16[] = {
|
|
|
|
LOCAL_EGL_SURFACE_TYPE,
|
|
|
|
LOCAL_EGL_WINDOW_BIT,
|
|
|
|
LOCAL_EGL_RENDERABLE_TYPE,
|
|
|
|
LOCAL_EGL_OPENGL_ES2_BIT,
|
|
|
|
LOCAL_EGL_RED_SIZE,
|
|
|
|
5,
|
|
|
|
LOCAL_EGL_GREEN_SIZE,
|
|
|
|
6,
|
|
|
|
LOCAL_EGL_BLUE_SIZE,
|
|
|
|
5,
|
|
|
|
LOCAL_EGL_ALPHA_SIZE,
|
|
|
|
0,
|
2013-12-14 00:15:07 +04:00
|
|
|
EGL_ATTRIBS_LIST_SAFE_TERMINATION_WORKING_AROUND_BUGS};
|
2011-09-23 09:58:15 +04:00
|
|
|
|
2012-07-07 18:06:59 +04:00
|
|
|
static const EGLint kEGLConfigAttribsRGB24[] = {
|
|
|
|
LOCAL_EGL_SURFACE_TYPE,
|
|
|
|
LOCAL_EGL_WINDOW_BIT,
|
|
|
|
LOCAL_EGL_RENDERABLE_TYPE,
|
|
|
|
LOCAL_EGL_OPENGL_ES2_BIT,
|
|
|
|
LOCAL_EGL_RED_SIZE,
|
|
|
|
8,
|
|
|
|
LOCAL_EGL_GREEN_SIZE,
|
|
|
|
8,
|
|
|
|
LOCAL_EGL_BLUE_SIZE,
|
2013-02-14 03:26:24 +04:00
|
|
|
8,
|
|
|
|
LOCAL_EGL_ALPHA_SIZE,
|
|
|
|
0,
|
2013-12-14 00:15:07 +04:00
|
|
|
EGL_ATTRIBS_LIST_SAFE_TERMINATION_WORKING_AROUND_BUGS};
|
2011-09-23 09:58:15 +04:00
|
|
|
|
|
|
|
static const EGLint kEGLConfigAttribsRGBA32[] = {
|
|
|
|
LOCAL_EGL_SURFACE_TYPE,
|
|
|
|
LOCAL_EGL_WINDOW_BIT,
|
|
|
|
LOCAL_EGL_RENDERABLE_TYPE,
|
|
|
|
LOCAL_EGL_OPENGL_ES2_BIT,
|
|
|
|
LOCAL_EGL_RED_SIZE,
|
|
|
|
8,
|
|
|
|
LOCAL_EGL_GREEN_SIZE,
|
|
|
|
8,
|
|
|
|
LOCAL_EGL_BLUE_SIZE,
|
|
|
|
8,
|
|
|
|
LOCAL_EGL_ALPHA_SIZE,
|
2018-11-30 13:46:48 +03:00
|
|
|
8,
|
2013-12-14 00:15:07 +04:00
|
|
|
EGL_ATTRIBS_LIST_SAFE_TERMINATION_WORKING_AROUND_BUGS};
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2017-06-14 13:32:04 +03:00
|
|
|
bool CreateConfig(EGLConfig* aConfig, int32_t depth, bool aEnableDepthBuffer) {
|
2010-07-19 09:01:14 +04:00
|
|
|
EGLConfig configs[64];
|
2012-07-07 18:06:59 +04:00
|
|
|
const EGLint* attribs;
|
2012-03-09 04:01:12 +04:00
|
|
|
EGLint ncfg = ArrayLength(configs);
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2012-07-07 18:06:59 +04:00
|
|
|
switch (depth) {
|
|
|
|
case 16:
|
|
|
|
attribs = kEGLConfigAttribsRGB16;
|
|
|
|
break;
|
|
|
|
case 24:
|
|
|
|
attribs = kEGLConfigAttribsRGB24;
|
|
|
|
break;
|
|
|
|
case 32:
|
|
|
|
attribs = kEGLConfigAttribsRGBA32;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
NS_ERROR("Unknown pixel depth");
|
|
|
|
return false;
|
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-06-04 05:09:29 +03:00
|
|
|
auto* egl = gl::GLLibraryEGL::Get();
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-06-04 05:09:29 +03:00
|
|
|
if (!egl->fChooseConfig(EGL_DISPLAY(), attribs, configs, ncfg, &ncfg) ||
|
2012-03-09 04:01:12 +04:00
|
|
|
ncfg < 1) {
|
|
|
|
return false;
|
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2014-06-18 01:19:11 +04:00
|
|
|
for (int j = 0; j < ncfg; ++j) {
|
|
|
|
EGLConfig config = configs[j];
|
|
|
|
EGLint r, g, b, a;
|
2018-06-04 05:09:29 +03:00
|
|
|
if (egl->fGetConfigAttrib(EGL_DISPLAY(), config, LOCAL_EGL_RED_SIZE, &r) &&
|
|
|
|
egl->fGetConfigAttrib(EGL_DISPLAY(), config, LOCAL_EGL_GREEN_SIZE,
|
|
|
|
&g) &&
|
2015-09-24 22:21:05 +03:00
|
|
|
egl->fGetConfigAttrib(EGL_DISPLAY(), config, LOCAL_EGL_BLUE_SIZE, &b) &&
|
2018-06-04 05:09:29 +03:00
|
|
|
egl->fGetConfigAttrib(EGL_DISPLAY(), config, LOCAL_EGL_ALPHA_SIZE,
|
|
|
|
&a) &&
|
2012-03-09 04:01:12 +04:00
|
|
|
((depth == 16 && r == 5 && g == 6 && b == 5) ||
|
2012-07-07 18:06:59 +04:00
|
|
|
(depth == 24 && r == 8 && g == 8 && b == 8) ||
|
|
|
|
(depth == 32 && r == 8 && g == 8 && b == 8 && a == 8))) {
|
2017-06-14 13:32:04 +03:00
|
|
|
EGLint z;
|
|
|
|
if (aEnableDepthBuffer) {
|
2018-06-04 05:09:29 +03:00
|
|
|
if (!egl->fGetConfigAttrib(EGL_DISPLAY(), config, LOCAL_EGL_DEPTH_SIZE,
|
|
|
|
&z) ||
|
2017-06-14 13:32:04 +03:00
|
|
|
z != 24) {
|
|
|
|
continue;
|
2010-09-10 00:40:11 +04:00
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
2012-03-09 04:01:12 +04:00
|
|
|
*aConfig = config;
|
|
|
|
return true;
|
2010-07-19 09:01:14 +04:00
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
2011-09-23 09:58:15 +04:00
|
|
|
return false;
|
2010-12-14 01:36:35 +03:00
|
|
|
}
|
|
|
|
|
2012-03-30 06:47:22 +04:00
|
|
|
// Return true if a suitable EGLConfig was found and pass it out
|
|
|
|
// through aConfig. Return false otherwise.
|
|
|
|
//
|
|
|
|
// NB: It's entirely legal for the returned EGLConfig to be valid yet
|
|
|
|
// have the value null.
|
2017-06-14 13:32:04 +03:00
|
|
|
static bool CreateConfig(EGLConfig* aConfig, bool aEnableDepthBuffer) {
|
2017-03-29 17:14:19 +03:00
|
|
|
int32_t depth = gfxVars::ScreenDepth();
|
2017-06-14 13:32:04 +03:00
|
|
|
if (!CreateConfig(aConfig, depth, aEnableDepthBuffer)) {
|
2012-03-30 06:47:22 +04:00
|
|
|
#ifdef MOZ_WIDGET_ANDROID
|
|
|
|
// Bug 736005
|
|
|
|
// Android doesn't always support 16 bit so also try 24 bit
|
|
|
|
if (depth == 16) {
|
2017-06-14 13:32:04 +03:00
|
|
|
return CreateConfig(aConfig, 24, aEnableDepthBuffer);
|
2012-03-30 06:47:22 +04:00
|
|
|
}
|
2014-02-25 02:17:34 +04:00
|
|
|
// Bug 970096
|
|
|
|
// Some devices that have 24 bit screens only support 16 bit OpenGL?
|
|
|
|
if (depth == 24) {
|
2017-06-14 13:32:04 +03:00
|
|
|
return CreateConfig(aConfig, 16, aEnableDepthBuffer);
|
2012-03-30 06:47:22 +04:00
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
#endif
|
2012-03-30 06:47:22 +04:00
|
|
|
return false;
|
2018-11-30 13:46:48 +03:00
|
|
|
} else {
|
2012-03-09 04:01:12 +04:00
|
|
|
return true;
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
2012-03-30 06:47:22 +04:00
|
|
|
}
|
|
|
|
|
2014-04-15 18:57:26 +04:00
|
|
|
already_AddRefed<GLContext> GLContextProviderEGL::CreateWrappingExisting(
|
|
|
|
void* aContext, void* aSurface) {
|
2016-06-06 23:52:42 +03:00
|
|
|
nsCString discardFailureId;
|
2018-06-04 05:09:29 +03:00
|
|
|
if (!GLLibraryEGL::EnsureInitialized(false, &discardFailureId)) {
|
2017-09-19 05:26:31 +03:00
|
|
|
MOZ_CRASH("GFX: Failed to load EGL library 2!");
|
2014-04-15 18:57:26 +04:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2016-06-07 03:31:53 +03:00
|
|
|
if (!aContext || !aSurface) return nullptr;
|
2014-04-15 18:57:26 +04:00
|
|
|
|
2016-06-07 03:31:53 +03:00
|
|
|
SurfaceCaps caps = SurfaceCaps::Any();
|
|
|
|
EGLConfig config = EGL_NO_CONFIG;
|
2017-03-14 23:59:00 +03:00
|
|
|
RefPtr<GLContextEGL> gl =
|
|
|
|
new GLContextEGL(CreateContextFlags::NONE, caps, false, config,
|
|
|
|
(EGLSurface)aSurface, (EGLContext)aContext);
|
2016-06-07 03:31:53 +03:00
|
|
|
gl->SetIsDoubleBuffered(true);
|
|
|
|
gl->mOwnsContext = false;
|
2014-04-15 18:57:26 +04:00
|
|
|
|
2016-06-07 03:31:53 +03:00
|
|
|
return gl.forget();
|
2014-04-15 18:57:26 +04:00
|
|
|
}
|
|
|
|
|
2016-07-25 21:41:00 +03:00
|
|
|
already_AddRefed<GLContext> GLContextProviderEGL::CreateForCompositorWidget(
|
|
|
|
CompositorWidget* aCompositorWidget, bool aForceAccelerated) {
|
2017-01-21 03:20:59 +03:00
|
|
|
MOZ_ASSERT(aCompositorWidget);
|
2017-01-29 16:03:30 +03:00
|
|
|
return GLContextEGLFactory::Create(
|
|
|
|
GET_NATIVE_WINDOW_FROM_COMPOSITOR_WIDGET(aCompositorWidget),
|
|
|
|
aCompositorWidget->GetCompositorOptions().UseWebRender());
|
2016-07-25 21:41:00 +03:00
|
|
|
}
|
|
|
|
|
2017-01-13 23:16:52 +03:00
|
|
|
already_AddRefed<GLContext> GLContextProviderEGL::CreateForWindow(
|
|
|
|
nsIWidget* aWidget, bool aWebRender, bool aForceAccelerated) {
|
2017-01-21 03:20:59 +03:00
|
|
|
MOZ_ASSERT(aWidget);
|
2017-01-29 16:03:30 +03:00
|
|
|
return GLContextEGLFactory::Create(
|
|
|
|
GET_NATIVE_WINDOW_FROM_REAL_WIDGET(aWidget), aWebRender);
|
2010-06-10 02:27:29 +04:00
|
|
|
}
|
|
|
|
|
2017-06-07 03:26:49 +03:00
|
|
|
#if defined(MOZ_WIDGET_ANDROID)
|
|
|
|
EGLSurface GLContextEGL::CreateCompatibleSurface(void* aWindow) {
|
|
|
|
if (mConfig == EGL_NO_CONFIG) {
|
2017-09-19 05:26:31 +03:00
|
|
|
MOZ_CRASH("GFX: Failed with invalid EGLConfig 2!");
|
2017-06-07 03:26:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return GLContextProviderEGL::CreateEGLSurface(aWindow, mConfig);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* static */ EGLSurface GLContextProviderEGL::CreateEGLSurface(
|
|
|
|
void* aWindow, EGLConfig aConfig) {
|
2017-01-21 02:57:23 +03:00
|
|
|
// NOTE: aWindow is an ANativeWindow
|
2016-06-06 23:52:42 +03:00
|
|
|
nsCString discardFailureId;
|
2018-06-04 05:09:29 +03:00
|
|
|
if (!GLLibraryEGL::EnsureInitialized(false, &discardFailureId)) {
|
2017-09-19 05:26:31 +03:00
|
|
|
MOZ_CRASH("GFX: Failed to load EGL library 4!");
|
2015-04-11 05:14:00 +03:00
|
|
|
}
|
2018-06-04 05:09:29 +03:00
|
|
|
auto* egl = gl::GLLibraryEGL::Get();
|
2017-06-07 03:26:49 +03:00
|
|
|
EGLConfig config = aConfig;
|
|
|
|
if (!config && !CreateConfig(&config, /* aEnableDepthBuffer */ false)) {
|
2017-09-19 05:26:31 +03:00
|
|
|
MOZ_CRASH("GFX: Failed to create EGLConfig 2!");
|
2015-04-11 05:14:00 +03:00
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2015-04-11 05:14:00 +03:00
|
|
|
MOZ_ASSERT(aWindow);
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-06-04 05:09:29 +03:00
|
|
|
EGLSurface surface =
|
|
|
|
egl->fCreateWindowSurface(EGL_DISPLAY(), config, aWindow, 0);
|
2015-04-11 05:14:00 +03:00
|
|
|
if (surface == EGL_NO_SURFACE) {
|
2017-09-19 05:26:31 +03:00
|
|
|
MOZ_CRASH("GFX: Failed to create EGLSurface 2!");
|
2015-04-11 05:14:00 +03:00
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2015-04-11 05:14:00 +03:00
|
|
|
return surface;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* static */ void GLContextProviderEGL::DestroyEGLSurface(EGLSurface surface) {
|
2016-06-06 23:52:42 +03:00
|
|
|
nsCString discardFailureId;
|
2018-06-04 05:09:29 +03:00
|
|
|
if (!GLLibraryEGL::EnsureInitialized(false, &discardFailureId)) {
|
2017-09-19 05:26:31 +03:00
|
|
|
MOZ_CRASH("GFX: Failed to load EGL library 5!");
|
2015-04-11 05:14:00 +03:00
|
|
|
}
|
2018-06-04 05:09:29 +03:00
|
|
|
auto* egl = gl::GLLibraryEGL::Get();
|
|
|
|
egl->fDestroySurface(EGL_DISPLAY(), surface);
|
2015-04-11 05:14:00 +03:00
|
|
|
}
|
|
|
|
#endif // defined(ANDROID)
|
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
static void FillContextAttribs(bool alpha, bool depth, bool stencil, bool bpp16,
|
2015-11-27 22:02:52 +03:00
|
|
|
bool es3, nsTArray<EGLint>* out) {
|
2015-09-24 22:21:05 +03:00
|
|
|
out->AppendElement(LOCAL_EGL_SURFACE_TYPE);
|
2018-05-22 11:34:32 +03:00
|
|
|
#if defined(MOZ_WAYLAND)
|
2018-11-12 18:25:38 +03:00
|
|
|
if (!GDK_IS_X11_DISPLAY(gdk_display_get_default())) {
|
2018-05-22 11:34:32 +03:00
|
|
|
// Wayland on desktop does not support PBuffer or FBO.
|
|
|
|
// We create a dummy wl_egl_window instead.
|
|
|
|
out->AppendElement(LOCAL_EGL_WINDOW_BIT);
|
|
|
|
} else {
|
2015-09-24 22:21:05 +03:00
|
|
|
out->AppendElement(LOCAL_EGL_PBUFFER_BIT);
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
|
|
|
#else
|
2018-05-22 11:34:32 +03:00
|
|
|
out->AppendElement(LOCAL_EGL_PBUFFER_BIT);
|
|
|
|
#endif
|
2015-09-24 22:21:05 +03:00
|
|
|
|
|
|
|
out->AppendElement(LOCAL_EGL_RENDERABLE_TYPE);
|
2015-11-27 22:02:52 +03:00
|
|
|
if (es3) {
|
|
|
|
out->AppendElement(LOCAL_EGL_OPENGL_ES3_BIT_KHR);
|
|
|
|
} else {
|
|
|
|
out->AppendElement(LOCAL_EGL_OPENGL_ES2_BIT);
|
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
out->AppendElement(LOCAL_EGL_RED_SIZE);
|
|
|
|
if (bpp16) {
|
|
|
|
out->AppendElement(alpha ? 4 : 5);
|
|
|
|
} else {
|
|
|
|
out->AppendElement(8);
|
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
out->AppendElement(LOCAL_EGL_GREEN_SIZE);
|
|
|
|
if (bpp16) {
|
|
|
|
out->AppendElement(alpha ? 4 : 6);
|
|
|
|
} else {
|
|
|
|
out->AppendElement(8);
|
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
out->AppendElement(LOCAL_EGL_BLUE_SIZE);
|
|
|
|
if (bpp16) {
|
|
|
|
out->AppendElement(alpha ? 4 : 5);
|
|
|
|
} else {
|
|
|
|
out->AppendElement(8);
|
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
out->AppendElement(LOCAL_EGL_ALPHA_SIZE);
|
|
|
|
if (alpha) {
|
|
|
|
out->AppendElement(bpp16 ? 4 : 8);
|
|
|
|
} else {
|
|
|
|
out->AppendElement(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
out->AppendElement(LOCAL_EGL_DEPTH_SIZE);
|
|
|
|
out->AppendElement(depth ? 16 : 0);
|
2015-09-23 02:49:25 +03:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
out->AppendElement(LOCAL_EGL_STENCIL_SIZE);
|
|
|
|
out->AppendElement(stencil ? 8 : 0);
|
|
|
|
|
|
|
|
// EGL_ATTRIBS_LIST_SAFE_TERMINATION_WORKING_AROUND_BUGS
|
|
|
|
out->AppendElement(LOCAL_EGL_NONE);
|
|
|
|
out->AppendElement(0);
|
|
|
|
|
|
|
|
out->AppendElement(0);
|
|
|
|
out->AppendElement(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static GLint GetAttrib(GLLibraryEGL* egl, EGLConfig config, EGLint attrib) {
|
|
|
|
EGLint bits = 0;
|
|
|
|
egl->fGetConfigAttrib(egl->Display(), config, attrib, &bits);
|
|
|
|
MOZ_ASSERT(egl->fGetError() == LOCAL_EGL_SUCCESS);
|
|
|
|
|
|
|
|
return bits;
|
|
|
|
}
|
|
|
|
|
2015-11-27 22:02:52 +03:00
|
|
|
static EGLConfig ChooseConfig(GLLibraryEGL* egl, CreateContextFlags flags,
|
|
|
|
const SurfaceCaps& minCaps,
|
2015-09-24 22:21:05 +03:00
|
|
|
SurfaceCaps* const out_configCaps) {
|
|
|
|
nsTArray<EGLint> configAttribList;
|
|
|
|
FillContextAttribs(
|
|
|
|
minCaps.alpha, minCaps.depth, minCaps.stencil, minCaps.bpp16,
|
2015-11-27 22:02:52 +03:00
|
|
|
bool(flags & CreateContextFlags::PREFER_ES3), &configAttribList);
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
const EGLint* configAttribs = configAttribList.Elements();
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
// We're guaranteed to get at least minCaps, and the sorting dictated by the
|
|
|
|
// spec for eglChooseConfig reasonably assures that a reasonable 'best' config
|
|
|
|
// is on top.
|
|
|
|
const EGLint kMaxConfigs = 1;
|
|
|
|
EGLConfig configs[kMaxConfigs];
|
2013-02-14 03:26:24 +04:00
|
|
|
EGLint foundConfigs = 0;
|
2015-09-24 22:21:05 +03:00
|
|
|
if (!egl->fChooseConfig(egl->Display(), configAttribs, configs, kMaxConfigs,
|
|
|
|
&foundConfigs) ||
|
2011-02-01 00:10:54 +03:00
|
|
|
foundConfigs == 0) {
|
2015-09-24 22:21:05 +03:00
|
|
|
return EGL_NO_CONFIG;
|
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
EGLConfig config = configs[0];
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
*out_configCaps = minCaps; // Pick up any preserve, etc.
|
|
|
|
out_configCaps->color = true;
|
|
|
|
out_configCaps->alpha = bool(GetAttrib(egl, config, LOCAL_EGL_ALPHA_SIZE));
|
|
|
|
out_configCaps->depth = bool(GetAttrib(egl, config, LOCAL_EGL_DEPTH_SIZE));
|
|
|
|
out_configCaps->stencil =
|
|
|
|
bool(GetAttrib(egl, config, LOCAL_EGL_STENCIL_SIZE));
|
|
|
|
out_configCaps->bpp16 = (GetAttrib(egl, config, LOCAL_EGL_RED_SIZE) < 8);
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
return config;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*static*/ already_AddRefed<GLContextEGL>
|
2015-11-27 22:02:52 +03:00
|
|
|
GLContextEGL::CreateEGLPBufferOffscreenContext(
|
|
|
|
CreateContextFlags flags, const mozilla::gfx::IntSize& size,
|
2016-06-11 05:01:00 +03:00
|
|
|
const SurfaceCaps& minCaps, nsACString* const out_failureId) {
|
2016-04-22 02:32:18 +03:00
|
|
|
bool forceEnableHardware =
|
|
|
|
bool(flags & CreateContextFlags::FORCE_ENABLE_HARDWARE);
|
2018-06-04 05:09:29 +03:00
|
|
|
if (!GLLibraryEGL::EnsureInitialized(forceEnableHardware, out_failureId)) {
|
2016-04-22 02:32:18 +03:00
|
|
|
return nullptr;
|
2016-06-06 23:52:42 +03:00
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2018-06-04 05:09:29 +03:00
|
|
|
auto* egl = gl::GLLibraryEGL::Get();
|
2015-09-24 22:21:05 +03:00
|
|
|
SurfaceCaps configCaps;
|
2018-06-04 05:09:29 +03:00
|
|
|
EGLConfig config = ChooseConfig(egl, flags, minCaps, &configCaps);
|
2015-09-24 22:21:05 +03:00
|
|
|
if (config == EGL_NO_CONFIG) {
|
2016-06-11 05:01:00 +03:00
|
|
|
*out_failureId = NS_LITERAL_CSTRING("FEATURE_FAILURE_EGL_NO_CONFIG");
|
2015-09-24 22:21:05 +03:00
|
|
|
NS_WARNING("Failed to find a compatible config.");
|
2012-07-30 18:20:58 +04:00
|
|
|
return nullptr;
|
2010-07-19 09:01:14 +04:00
|
|
|
}
|
2010-06-10 02:27:29 +04:00
|
|
|
|
2016-04-22 02:32:18 +03:00
|
|
|
if (GLContext::ShouldSpew()) {
|
2018-06-04 05:09:29 +03:00
|
|
|
egl->DumpEGLConfig(config);
|
2016-04-22 02:32:18 +03:00
|
|
|
}
|
2010-07-19 09:01:14 +04:00
|
|
|
|
2015-05-28 12:45:33 +03:00
|
|
|
mozilla::gfx::IntSize pbSize(size);
|
2018-05-22 11:34:32 +03:00
|
|
|
EGLSurface surface = nullptr;
|
|
|
|
#if defined(MOZ_WAYLAND)
|
2018-11-12 18:25:38 +03:00
|
|
|
if (!GDK_IS_X11_DISPLAY(gdk_display_get_default())) {
|
2018-05-22 11:34:32 +03:00
|
|
|
surface = GLContextEGL::CreateWaylandBufferSurface(config, pbSize);
|
|
|
|
} else
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
surface = GLContextEGL::CreatePBufferSurfaceTryingPowerOfTwo(
|
|
|
|
config, LOCAL_EGL_NONE, pbSize);
|
|
|
|
}
|
2012-03-13 02:10:38 +04:00
|
|
|
if (!surface) {
|
2016-06-11 05:01:00 +03:00
|
|
|
*out_failureId = NS_LITERAL_CSTRING("FEATURE_FAILURE_EGL_POT");
|
2012-03-13 02:10:38 +04:00
|
|
|
NS_WARNING("Failed to create PBuffer for context!");
|
2012-07-30 18:20:58 +04:00
|
|
|
return nullptr;
|
2012-03-13 02:10:38 +04:00
|
|
|
}
|
2010-06-10 02:27:29 +04:00
|
|
|
|
2017-03-14 23:59:00 +03:00
|
|
|
RefPtr<GLContextEGL> gl = GLContextEGL::CreateGLContext(
|
|
|
|
flags, configCaps, true, config, surface, out_failureId);
|
2015-09-24 22:21:05 +03:00
|
|
|
if (!gl) {
|
2012-03-13 02:10:38 +04:00
|
|
|
NS_WARNING("Failed to create GLContext from PBuffer");
|
2018-06-04 05:09:29 +03:00
|
|
|
egl->fDestroySurface(egl->Display(), surface);
|
2018-05-22 11:34:32 +03:00
|
|
|
#if defined(MOZ_WAYLAND)
|
|
|
|
DeleteWaylandGLSurface(surface);
|
|
|
|
#endif
|
2012-07-30 18:20:58 +04:00
|
|
|
return nullptr;
|
2010-06-10 02:27:29 +04:00
|
|
|
}
|
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
return gl.forget();
|
2010-07-19 09:01:14 +04:00
|
|
|
}
|
|
|
|
|
2016-06-11 05:01:00 +03:00
|
|
|
/*static*/ already_AddRefed<GLContext> GLContextProviderEGL::CreateHeadless(
|
|
|
|
CreateContextFlags flags, nsACString* const out_failureId) {
|
2015-05-28 12:45:33 +03:00
|
|
|
mozilla::gfx::IntSize dummySize = mozilla::gfx::IntSize(16, 16);
|
2015-09-24 22:21:05 +03:00
|
|
|
SurfaceCaps dummyCaps = SurfaceCaps::Any();
|
2016-06-06 23:52:42 +03:00
|
|
|
return GLContextEGL::CreateEGLPBufferOffscreenContext(
|
|
|
|
flags, dummySize, dummyCaps, out_failureId);
|
2014-08-28 03:16:22 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Under EGL, on Android, pbuffers are supported fine, though
|
|
|
|
// often without the ability to texture from them directly.
|
2015-05-28 12:45:33 +03:00
|
|
|
/*static*/ already_AddRefed<GLContext> GLContextProviderEGL::CreateOffscreen(
|
|
|
|
const mozilla::gfx::IntSize& size, const SurfaceCaps& minCaps,
|
2016-06-11 05:01:00 +03:00
|
|
|
CreateContextFlags flags, nsACString* const out_failureId) {
|
2016-04-20 04:03:17 +03:00
|
|
|
bool forceEnableHardware =
|
|
|
|
bool(flags & CreateContextFlags::FORCE_ENABLE_HARDWARE);
|
2018-06-04 05:09:29 +03:00
|
|
|
if (!GLLibraryEGL::EnsureInitialized(
|
|
|
|
forceEnableHardware, out_failureId)) { // Needed for IsANGLE().
|
2016-04-20 04:03:17 +03:00
|
|
|
return nullptr;
|
2015-09-24 22:21:05 +03:00
|
|
|
}
|
|
|
|
|
2017-06-07 03:26:49 +03:00
|
|
|
auto* egl = gl::GLLibraryEGL::Get();
|
|
|
|
bool canOffscreenUseHeadless = true;
|
|
|
|
if (egl->IsANGLE()) {
|
|
|
|
// ANGLE needs to use PBuffers.
|
|
|
|
canOffscreenUseHeadless = false;
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
2017-06-07 03:26:49 +03:00
|
|
|
|
2015-10-18 08:24:48 +03:00
|
|
|
#if defined(MOZ_WIDGET_ANDROID)
|
2015-09-29 01:53:37 +03:00
|
|
|
// Using a headless context loses the SurfaceCaps
|
|
|
|
// which can cause a loss of depth and/or stencil
|
|
|
|
canOffscreenUseHeadless = false;
|
|
|
|
#endif // defined(MOZ_WIDGET_ANDROID)
|
2015-09-24 22:21:05 +03:00
|
|
|
|
|
|
|
RefPtr<GLContext> gl;
|
2015-09-29 01:53:37 +03:00
|
|
|
SurfaceCaps minOffscreenCaps = minCaps;
|
2015-09-24 22:21:05 +03:00
|
|
|
|
2016-06-07 03:31:53 +03:00
|
|
|
if (canOffscreenUseHeadless) {
|
|
|
|
gl = CreateHeadless(flags, out_failureId);
|
|
|
|
if (!gl) {
|
2015-09-24 22:21:05 +03:00
|
|
|
return nullptr;
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
2015-09-29 01:53:37 +03:00
|
|
|
} else {
|
|
|
|
SurfaceCaps minBackbufferCaps = minOffscreenCaps;
|
|
|
|
if (minOffscreenCaps.antialias) {
|
|
|
|
minBackbufferCaps.antialias = false;
|
|
|
|
minBackbufferCaps.depth = false;
|
|
|
|
minBackbufferCaps.stencil = false;
|
2015-09-24 22:21:05 +03:00
|
|
|
}
|
|
|
|
|
2015-09-29 01:53:37 +03:00
|
|
|
gl = GLContextEGL::CreateEGLPBufferOffscreenContext(
|
2016-06-11 05:01:00 +03:00
|
|
|
flags, size, minBackbufferCaps, out_failureId);
|
2015-09-29 01:53:37 +03:00
|
|
|
if (!gl) return nullptr;
|
2018-11-30 13:46:48 +03:00
|
|
|
|
2015-09-29 01:53:37 +03:00
|
|
|
// Pull the actual resulting caps to ensure that our offscreen matches our
|
|
|
|
// backbuffer.
|
2016-06-06 23:52:42 +03:00
|
|
|
minOffscreenCaps.alpha = gl->Caps().alpha;
|
|
|
|
if (!minOffscreenCaps.antialias) {
|
2016-06-11 05:01:00 +03:00
|
|
|
// Only update these if we don't have AA. If we do have AA, we ignore
|
|
|
|
// backbuffer depth/stencil.
|
|
|
|
minOffscreenCaps.depth = gl->Caps().depth;
|
|
|
|
minOffscreenCaps.stencil = gl->Caps().stencil;
|
2016-06-06 23:52:42 +03:00
|
|
|
}
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
|
|
|
|
2015-09-29 01:53:37 +03:00
|
|
|
// Init the offscreen with the updated offscreen caps.
|
|
|
|
if (!gl->InitOffscreen(size, minOffscreenCaps)) {
|
2016-06-11 05:01:00 +03:00
|
|
|
*out_failureId = NS_LITERAL_CSTRING("FEATURE_FAILURE_EGL_OFFSCREEN");
|
2012-07-30 18:20:58 +04:00
|
|
|
return nullptr;
|
2018-11-30 13:46:48 +03:00
|
|
|
}
|
2012-03-02 21:28:06 +04:00
|
|
|
|
2015-09-24 22:21:05 +03:00
|
|
|
return gl.forget();
|
2010-06-10 02:27:29 +04:00
|
|
|
}
|
|
|
|
|
2012-06-14 19:59:40 +04:00
|
|
|
// Don't want a global context on Android as 1) share groups across 2 threads
|
|
|
|
// fail on many Tegra drivers (bug 759225) and 2) some mobile devices have a
|
|
|
|
// very strict limit on global number of GL contexts (bug 754257) and 3) each
|
2012-11-21 06:45:13 +04:00
|
|
|
// EGL context eats 750k on B2G (bug 813783)
|
2014-01-10 22:55:23 +04:00
|
|
|
/*static*/ GLContext* GLContextProviderEGL::GetGlobalContext() {
|
2012-07-30 18:20:58 +04:00
|
|
|
return nullptr;
|
2010-07-19 09:01:14 +04:00
|
|
|
}
|
|
|
|
|
2010-07-20 08:05:42 +04:00
|
|
|
/*static*/ void GLContextProviderEGL::Shutdown() {
|
2018-06-04 05:09:52 +03:00
|
|
|
const RefPtr<GLLibraryEGL> egl = GLLibraryEGL::Get();
|
|
|
|
if (egl) {
|
|
|
|
egl->Shutdown();
|
|
|
|
}
|
2010-07-20 08:05:42 +04:00
|
|
|
}
|
|
|
|
|
2010-06-10 02:27:29 +04:00
|
|
|
} /* namespace gl */
|
|
|
|
} /* namespace mozilla */
|
|
|
|
|
2013-12-14 00:15:07 +04:00
|
|
|
#undef EGL_ATTRIBS_LIST_SAFE_TERMINATION_WORKING_AROUND_BUGS
|