2018-11-30 22:52:05 +03:00
|
|
|
/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
2015-09-29 01:09:56 +03: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/. */
|
|
|
|
|
|
|
|
#include "GLContextProvider.h"
|
|
|
|
#include "GLContextEAGL.h"
|
|
|
|
#include "nsDebug.h"
|
|
|
|
#include "nsIWidget.h"
|
|
|
|
#include "gfxFailure.h"
|
|
|
|
#include "prenv.h"
|
|
|
|
#include "mozilla/Preferences.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"
|
2015-09-29 01:09:56 +03:00
|
|
|
#include "GeckoProfiler.h"
|
|
|
|
|
|
|
|
#import <UIKit/UIKit.h>
|
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
namespace gl {
|
|
|
|
|
2016-07-25 21:41:00 +03:00
|
|
|
using namespace mozilla::widget;
|
|
|
|
|
2016-06-07 02:27:43 +03:00
|
|
|
GLContextEAGL::GLContextEAGL(CreateContextFlags flags, const SurfaceCaps& caps,
|
2019-01-21 20:18:16 +03:00
|
|
|
EAGLContext* context, GLContext* sharedContext, bool isOffscreen)
|
|
|
|
: GLContext(flags, caps, sharedContext, isOffscreen), mContext(context) {}
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
GLContextEAGL::~GLContextEAGL() {
|
|
|
|
MakeCurrent();
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
if (mBackbufferFB) {
|
|
|
|
fDeleteFramebuffers(1, &mBackbufferFB);
|
|
|
|
}
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
if (mBackbufferRB) {
|
|
|
|
fDeleteRenderbuffers(1, &mBackbufferRB);
|
|
|
|
}
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
MarkDestroyed();
|
2018-10-26 19:30:48 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
if (mLayer) {
|
|
|
|
mLayer = nil;
|
|
|
|
}
|
2018-10-26 19:30:48 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
if (mContext) {
|
|
|
|
[EAGLContext setCurrentContext:nil];
|
|
|
|
[mContext release];
|
|
|
|
}
|
2015-09-29 01:09:56 +03:00
|
|
|
}
|
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
bool GLContextEAGL::AttachToWindow(nsIWidget* aWidget) {
|
|
|
|
// This should only be called once
|
|
|
|
MOZ_ASSERT(!mBackbufferFB && !mBackbufferRB);
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
UIView* view = reinterpret_cast<UIView*>(aWidget->GetNativeData(NS_NATIVE_WIDGET));
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
if (!view) {
|
|
|
|
MOZ_CRASH("no view!");
|
|
|
|
}
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
mLayer = [view layer];
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
fGenFramebuffers(1, &mBackbufferFB);
|
|
|
|
return RecreateRB();
|
2015-09-29 01:09:56 +03:00
|
|
|
}
|
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
bool GLContextEAGL::RecreateRB() {
|
|
|
|
MakeCurrent();
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
CAEAGLLayer* layer = (CAEAGLLayer*)mLayer;
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
if (mBackbufferRB) {
|
|
|
|
// It doesn't seem to be enough to just call renderbufferStorage: below,
|
|
|
|
// we apparently have to recreate the RB.
|
|
|
|
fDeleteRenderbuffers(1, &mBackbufferRB);
|
|
|
|
mBackbufferRB = 0;
|
|
|
|
}
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
fGenRenderbuffers(1, &mBackbufferRB);
|
|
|
|
fBindRenderbuffer(LOCAL_GL_RENDERBUFFER, mBackbufferRB);
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
[mContext renderbufferStorage:LOCAL_GL_RENDERBUFFER fromDrawable:layer];
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
fBindFramebuffer(LOCAL_GL_FRAMEBUFFER, mBackbufferFB);
|
|
|
|
fFramebufferRenderbuffer(LOCAL_GL_FRAMEBUFFER, LOCAL_GL_COLOR_ATTACHMENT0, LOCAL_GL_RENDERBUFFER,
|
|
|
|
mBackbufferRB);
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
return LOCAL_GL_FRAMEBUFFER_COMPLETE == fCheckFramebufferStatus(LOCAL_GL_FRAMEBUFFER);
|
2015-09-29 01:09:56 +03:00
|
|
|
}
|
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
bool GLContextEAGL::MakeCurrentImpl() const {
|
|
|
|
if (mContext) {
|
|
|
|
if (![EAGLContext setCurrentContext:mContext]) {
|
|
|
|
return false;
|
2015-09-29 01:09:56 +03:00
|
|
|
}
|
2019-01-21 20:18:16 +03:00
|
|
|
}
|
|
|
|
return true;
|
2015-09-29 01:09:56 +03:00
|
|
|
}
|
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
bool GLContextEAGL::IsCurrentImpl() const { return [EAGLContext currentContext] == mContext; }
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-02-23 00:17:28 +03:00
|
|
|
static PRFuncPtr GLAPIENTRY GetLoadedProcAddress(const char* const name) {
|
|
|
|
PRLibrary* lib = nullptr;
|
|
|
|
const auto& ret = PR_FindFunctionSymbolAndLibrary(name, &leakedLibRef);
|
|
|
|
if (lib) {
|
|
|
|
PR_UnloadLibrary(lib);
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
Maybe<SymbolLoader> GLContextEAGL::GetSymbolLoader() const {
|
|
|
|
return Some(SymbolLoader(&GetLoadedProcAddress));
|
|
|
|
}
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
bool GLContextEAGL::IsDoubleBuffered() const { return true; }
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
bool GLContextEAGL::SwapBuffers() {
|
Bug 1375392 - Tweak the PROFILER_LABEL* macros. r=mstange.
This patch makes the following changes to the macros.
- Removes PROFILER_LABEL_FUNC. It's only suitable for use in functions outside
classes, due to PROFILER_FUNCTION_NAME not getting class names, and it was
mostly misused.
- Removes PROFILER_FUNCTION_NAME. It's no longer used, and __func__ is
universally available now anyway.
- Combines the first two string literal arguments of PROFILER_LABEL and
PROFILER_LABEL_DYNAMIC into a single argument. There was no good reason for
them to be separate, and it forced a '::' in the label, which isn't always
appropriate. Also, the meaning of the "name_space" argument was interpreted
in an interesting variety of ways.
- Adds an "AUTO_" prefix to PROFILER_LABEL and PROFILER_LABEL_DYNAMIC, to make
it clearer they construct RAII objects rather than just being function calls.
(I myself have screwed up the scoping because of this in the past.)
- Fills in the 'js::ProfileEntry::Category::' qualifier within the macro, so
the caller doesn't need to. This makes a *lot* more of the uses fit onto a
single line.
The patch also makes the following changes to the macro uses (beyond those
required by the changes described above).
- Fixes a bunch of labels that had gotten out of sync with the name of the
class and/or function that encloses them.
- Removes a useless PROFILER_LABEL use within a trivial scope in
EventStateManager::DispatchMouseOrPointerEvent(). It clearly wasn't serving
any useful purpose. It also serves as extra evidence that the AUTO_ prefix is
a good idea.
- Tweaks DecodePool::SyncRunIf{Preferred,Possible} so that the labelling is
done within them, instead of at their callsites, because that's a more
standard way of doing things.
--HG--
extra : rebase_source : 318d1bc6fc1425a94aacbf489dd46e4f83211de4
2017-06-22 10:08:53 +03:00
|
|
|
AUTO_PROFILER_LABEL("GLContextEAGL::SwapBuffers", GRAPHICS);
|
2015-09-29 01:09:56 +03:00
|
|
|
|
|
|
|
[mContext presentRenderbuffer:LOCAL_GL_RENDERBUFFER];
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
void GLContextEAGL::GetWSIInfo(nsCString* const out) const { out->AppendLiteral("EAGL"); }
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
already_AddRefed<GLContext> GLContextProviderEAGL::CreateWrappingExisting(void*, void*) {
|
|
|
|
return nullptr;
|
2015-09-29 01:09:56 +03:00
|
|
|
}
|
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
static GLContextEAGL* GetGlobalContextEAGL() {
|
|
|
|
return static_cast<GLContextEAGL*>(GLContextProviderEAGL::GetGlobalContext());
|
2015-09-29 01:09:56 +03:00
|
|
|
}
|
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
static already_AddRefed<GLContext> CreateEAGLContext(CreateContextFlags flags, bool aOffscreen,
|
|
|
|
GLContextEAGL* sharedContext) {
|
|
|
|
EAGLRenderingAPI apis[] = {kEAGLRenderingAPIOpenGLES3, kEAGLRenderingAPIOpenGLES2};
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
// Try to create a GLES3 context if we can, otherwise fall back to GLES2
|
|
|
|
EAGLContext* context = nullptr;
|
|
|
|
for (EAGLRenderingAPI api : apis) {
|
|
|
|
if (sharedContext) {
|
|
|
|
context = [[EAGLContext alloc] initWithAPI:api
|
|
|
|
sharegroup:sharedContext->GetEAGLContext().sharegroup];
|
|
|
|
} else {
|
|
|
|
context = [[EAGLContext alloc] initWithAPI:api];
|
2015-09-29 01:09:56 +03:00
|
|
|
}
|
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
if (context) {
|
|
|
|
break;
|
2015-09-29 01:09:56 +03:00
|
|
|
}
|
2019-01-21 20:18:16 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!context) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
RefPtr<GLContextEAGL> glContext =
|
|
|
|
new GLContextEAGL(flags, SurfaceCaps::ForRGBA(), context, sharedContext, aOffscreen);
|
|
|
|
if (!glContext->Init()) {
|
|
|
|
glContext = nullptr;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return glContext.forget();
|
2015-09-29 01:09:56 +03:00
|
|
|
}
|
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
already_AddRefed<GLContext> GLContextProviderEAGL::CreateForCompositorWidget(
|
2019-03-13 03:17:01 +03:00
|
|
|
CompositorWidget* aCompositorWidget, bool aWebRender, bool aForceAccelerated) {
|
|
|
|
if (!aCompositorWidget) {
|
|
|
|
MOZ_ASSERT(false);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
return CreateForWindow(aCompositorWidget->RealWidget(), aWebRender, aForceAccelerated);
|
2016-07-25 21:41:00 +03:00
|
|
|
}
|
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
already_AddRefed<GLContext> GLContextProviderEAGL::CreateForWindow(nsIWidget* aWidget,
|
|
|
|
bool aWebRender,
|
|
|
|
bool aForceAccelerated) {
|
|
|
|
RefPtr<GLContext> glContext =
|
|
|
|
CreateEAGLContext(CreateContextFlags::NONE, false, GetGlobalContextEAGL());
|
|
|
|
if (!glContext) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
if (!GLContextEAGL::Cast(glContext)->AttachToWindow(aWidget)) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
return glContext.forget();
|
2015-09-29 01:09:56 +03:00
|
|
|
}
|
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
already_AddRefed<GLContext> GLContextProviderEAGL::CreateHeadless(CreateContextFlags flags,
|
|
|
|
nsACString* const out_failureId) {
|
|
|
|
return CreateEAGLContext(flags, true, GetGlobalContextEAGL());
|
2015-09-29 01:09:56 +03:00
|
|
|
}
|
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
already_AddRefed<GLContext> GLContextProviderEAGL::CreateOffscreen(
|
|
|
|
const mozilla::gfx::IntSize& size, const SurfaceCaps& caps, CreateContextFlags flags,
|
|
|
|
nsACString* const out_failureId) {
|
|
|
|
RefPtr<GLContext> glContext = CreateHeadless(flags, out_failureId);
|
|
|
|
if (!glContext->InitOffscreen(size, caps)) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
return glContext.forget();
|
2015-09-29 01:09:56 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static RefPtr<GLContext> gGlobalContext;
|
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
GLContext* GLContextProviderEAGL::GetGlobalContext() {
|
|
|
|
static bool triedToCreateContext = false;
|
|
|
|
if (!triedToCreateContext) {
|
|
|
|
triedToCreateContext = true;
|
2016-04-22 02:32:18 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
MOZ_RELEASE_ASSERT(!gGlobalContext, "GFX: Global GL context already initialized.");
|
|
|
|
RefPtr<GLContext> temp = CreateHeadless(CreateContextFlags::NONE);
|
|
|
|
gGlobalContext = temp;
|
2016-04-22 02:32:18 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
if (!gGlobalContext) {
|
|
|
|
MOZ_CRASH("Failed to create global context");
|
2015-09-29 01:09:56 +03:00
|
|
|
}
|
2019-01-21 20:18:16 +03:00
|
|
|
}
|
2015-09-29 01:09:56 +03:00
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
return gGlobalContext;
|
2015-09-29 01:09:56 +03:00
|
|
|
}
|
|
|
|
|
2019-01-21 20:18:16 +03:00
|
|
|
void GLContextProviderEAGL::Shutdown() { gGlobalContext = nullptr; }
|
2015-09-29 01:09:56 +03:00
|
|
|
|
|
|
|
} /* namespace gl */
|
|
|
|
} /* namespace mozilla */
|