Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(847)

Unified Diff: third_party/WebKit/Source/modules/webgl/WebGLRenderingContextBase.cpp

Issue 1914233006: Implement offscreenCanvas.getContext('webgl') on a worker thread (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: address kbr@'s comment Created 4 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: third_party/WebKit/Source/modules/webgl/WebGLRenderingContextBase.cpp
diff --git a/third_party/WebKit/Source/modules/webgl/WebGLRenderingContextBase.cpp b/third_party/WebKit/Source/modules/webgl/WebGLRenderingContextBase.cpp
index a2bd44a75698135a0d48e9ce905cb3a2a639ccbe..ceb485bdbf6c7e4ddd84cb242e0e556ae3c88ebf 100644
--- a/third_party/WebKit/Source/modules/webgl/WebGLRenderingContextBase.cpp
+++ b/third_party/WebKit/Source/modules/webgl/WebGLRenderingContextBase.cpp
@@ -66,7 +66,6 @@
#include "modules/webgl/WebGLCompressedTexturePVRTC.h"
#include "modules/webgl/WebGLCompressedTextureS3TC.h"
#include "modules/webgl/WebGLContextAttributeHelpers.h"
-#include "modules/webgl/WebGLContextAttributes.h"
#include "modules/webgl/WebGLContextEvent.h"
#include "modules/webgl/WebGLContextGroup.h"
#include "modules/webgl/WebGLDebugRendererInfo.h"
@@ -79,19 +78,18 @@
#include "modules/webgl/WebGLRenderbuffer.h"
#include "modules/webgl/WebGLShader.h"
#include "modules/webgl/WebGLShaderPrecisionFormat.h"
-#include "modules/webgl/WebGLTexture.h"
#include "modules/webgl/WebGLUniformLocation.h"
#include "modules/webgl/WebGLVertexArrayObject.h"
#include "modules/webgl/WebGLVertexArrayObjectOES.h"
#include "platform/CheckedInt.h"
#include "platform/RuntimeEnabledFeatures.h"
+#include "platform/ThreadSafeFunctional.h"
+#include "platform/WaitableEvent.h"
#include "platform/geometry/IntSize.h"
#include "platform/graphics/GraphicsContext.h"
#include "platform/graphics/UnacceleratedImageBufferSurface.h"
#include "platform/graphics/gpu/AcceleratedImageBufferSurface.h"
-#include "platform/graphics/gpu/DrawingBuffer.h"
#include "public/platform/Platform.h"
-#include "public/platform/WebGraphicsContext3DProvider.h"
#include "public/platform/functional/WebFunction.h"
#include "wtf/Functional.h"
#include "wtf/PassOwnPtr.h"
@@ -108,19 +106,29 @@ namespace {
const double secondsBetweenRestoreAttempts = 1.0;
const int maxGLErrorsAllowedToConsole = 256;
const unsigned maxGLActiveContexts = 16;
+const unsigned maxGLActiveContextsOnWorker = 4;
+
+unsigned currentMaxGLContexts()
+{
+ return isMainThread() ? maxGLActiveContexts : maxGLActiveContextsOnWorker;
+}
using WebGLRenderingContextBaseSet = PersistentHeapHashSet<WeakMember<WebGLRenderingContextBase>>;
WebGLRenderingContextBaseSet& activeContexts()
{
- DEFINE_STATIC_LOCAL(WebGLRenderingContextBaseSet, activeContexts, ());
- return activeContexts;
+ DEFINE_THREAD_SAFE_STATIC_LOCAL(ThreadSpecific<WebGLRenderingContextBaseSet>, activeContexts, new ThreadSpecific<WebGLRenderingContextBaseSet>());
+ if (!activeContexts.isSet())
+ activeContexts->registerAsStaticReference();
+ return *activeContexts;
}
using WebGLRenderingContextBaseMap = PersistentHeapHashMap<WeakMember<WebGLRenderingContextBase>, int>;
WebGLRenderingContextBaseMap& forciblyEvictedContexts()
{
- DEFINE_STATIC_LOCAL(WebGLRenderingContextBaseMap, forciblyEvictedContexts, ());
- return forciblyEvictedContexts;
+ DEFINE_THREAD_SAFE_STATIC_LOCAL(ThreadSpecific<WebGLRenderingContextBaseMap>, forciblyEvictedContexts, new ThreadSpecific<WebGLRenderingContextBaseMap>());
+ if (!forciblyEvictedContexts.isSet())
+ forciblyEvictedContexts->registerAsStaticReference();
+ return *forciblyEvictedContexts;
}
} // namespace
@@ -190,8 +198,9 @@ WebGLRenderingContextBase* WebGLRenderingContextBase::oldestEvictedContext()
void WebGLRenderingContextBase::activateContext(WebGLRenderingContextBase* context)
{
+ unsigned maxGLContexts = currentMaxGLContexts();
unsigned removedContexts = 0;
- while (activeContexts().size() >= maxGLActiveContexts && removedContexts < maxGLActiveContexts) {
+ while (activeContexts().size() >= maxGLContexts && removedContexts < maxGLContexts) {
forciblyLoseOldestContext("WARNING: Too many active WebGL contexts. Oldest context will be lost.");
removedContexts++;
}
@@ -223,8 +232,9 @@ void WebGLRenderingContextBase::willDestroyContext(WebGLRenderingContextBase* co
ASSERT(!forciblyEvictedContexts().contains(context));
ASSERT(!activeContexts().contains(context));
+ unsigned maxGLContexts = currentMaxGLContexts();
// Try to re-enable the oldest inactive contexts.
- while (activeContexts().size() < maxGLActiveContexts && forciblyEvictedContexts().size()) {
+ while (activeContexts().size() < maxGLContexts && forciblyEvictedContexts().size()) {
WebGLRenderingContextBase* evictedContext = oldestEvictedContext();
if (!evictedContext->m_restoreAllowed) {
forciblyEvictedContexts().remove(evictedContext);
@@ -520,17 +530,63 @@ static String extractWebGLContextCreationError(const Platform::GraphicsInfo& inf
return statusMessage;
}
-static PassOwnPtr<WebGraphicsContext3DProvider> createWebGraphicsContext3DProviderInternal(HTMLCanvasElement* canvas, ScriptState* scriptState, WebGLContextAttributes attributes, unsigned webGLVersion)
+class WebGLRenderingContextBase::ContextProviderCreationInfo {
+public:
+ ContextProviderCreationInfo(Platform::ContextAttributes contextAttributes, Platform::GraphicsInfo glInfo, ScriptState* scriptState)
+ {
+ m_contextAttributes = contextAttributes;
+ m_glInfo = glInfo;
+ m_scriptState = scriptState;
+ }
+ Platform::ContextAttributes contextAttributes() { return m_contextAttributes; }
+ Platform::GraphicsInfo glInfo() { return m_glInfo; }
+ ScriptState* scriptState() { return m_scriptState; }
+ void setContextProvider(PassOwnPtr<WebGraphicsContext3DProvider> provider) { m_provider = std::move(provider); }
+ PassOwnPtr<WebGraphicsContext3DProvider> releaseContextProvider() { return m_provider.release(); }
+private:
+ Platform::ContextAttributes m_contextAttributes;
+ Platform::GraphicsInfo m_glInfo;
+ ScriptState* m_scriptState;
+ OwnPtr<WebGraphicsContext3DProvider> m_provider;
+};
+
+void WebGLRenderingContextBase::createContextProviderOnMainThread(ContextProviderCreationInfo* creationInfo, WaitableEvent* waitableEvent)
+{
+ ASSERT(isMainThread());
+ Platform::GraphicsInfo glInfo = creationInfo->glInfo();
+ OwnPtr<WebGraphicsContext3DProvider> provider = adoptPtr(Platform::current()->createOffscreenGraphicsContext3DProvider(
+ creationInfo->contextAttributes(), creationInfo->scriptState()->getExecutionContext()->url(), 0, &glInfo, Platform::DoNotBindToCurrentThread));
+ creationInfo->setContextProvider(provider.release());
+ waitableEvent->signal();
+}
+
+PassOwnPtr<WebGraphicsContext3DProvider> WebGLRenderingContextBase::createContextProviderOnWorkerThread(Platform::ContextAttributes contextAttributes, Platform::GraphicsInfo glInfo, ScriptState* scriptState)
+{
+ WaitableEvent waitableEvent;
+ OwnPtr<ContextProviderCreationInfo> creationInfo = adoptPtr(new ContextProviderCreationInfo(contextAttributes, glInfo, scriptState));
+ WebTaskRunner* taskRunner = Platform::current()->mainThread()->getWebTaskRunner();
+ taskRunner->postTask(BLINK_FROM_HERE, threadSafeBind(&createContextProviderOnMainThread, AllowCrossThreadAccess(creationInfo.get()), AllowCrossThreadAccess(&waitableEvent)));
+ waitableEvent.wait();
+ return creationInfo->releaseContextProvider();
+}
+
+PassOwnPtr<WebGraphicsContext3DProvider> WebGLRenderingContextBase::createContextProviderInternal(HTMLCanvasElement* canvas, ScriptState* scriptState, WebGLContextAttributes attributes, unsigned webGLVersion)
{
Platform::ContextAttributes contextAttributes = toPlatformContextAttributes(attributes, webGLVersion);
Platform::GraphicsInfo glInfo;
OwnPtr<WebGraphicsContext3DProvider> contextProvider;
if (canvas) {
contextProvider = adoptPtr(Platform::current()->createOffscreenGraphicsContext3DProvider(
- contextAttributes, canvas->document().topDocument().url(), 0, &glInfo));
+ contextAttributes, canvas->document().topDocument().url(), 0, &glInfo, Platform::BindToCurrentThread));
} else {
- contextProvider = adoptPtr(Platform::current()->createOffscreenGraphicsContext3DProvider(
- contextAttributes, scriptState->getExecutionContext()->url(), 0, &glInfo));
+ if (isMainThread()) {
+ contextProvider = adoptPtr(Platform::current()->createOffscreenGraphicsContext3DProvider(
+ contextAttributes, scriptState->getExecutionContext()->url(), 0, &glInfo, Platform::BindToCurrentThread));
+ } else {
+ contextProvider = createContextProviderOnWorkerThread(contextAttributes, glInfo, scriptState);
+ if (!contextProvider->bindToCurrentThread())
+ return nullptr;
+ }
}
if (!contextProvider || shouldFailContextCreationForTesting) {
shouldFailContextCreationForTesting = false;
@@ -544,7 +600,6 @@ static PassOwnPtr<WebGraphicsContext3DProvider> createWebGraphicsContext3DProvid
canvas->dispatchEvent(WebGLContextEvent::create(EventTypeNames::webglcontextcreationerror, false, true, "OES_packed_depth_stencil support is required."));
return nullptr;
}
-
return contextProvider.release();
}
@@ -565,12 +620,12 @@ PassOwnPtr<WebGraphicsContext3DProvider> WebGLRenderingContextBase::createWebGra
return nullptr;
}
- return createWebGraphicsContext3DProviderInternal(canvas, nullptr, attributes, webGLVersion);
+ return createContextProviderInternal(canvas, nullptr, attributes, webGLVersion);
}
PassOwnPtr<WebGraphicsContext3DProvider> WebGLRenderingContextBase::createWebGraphicsContext3DProvider(ScriptState* scriptState, WebGLContextAttributes attributes, unsigned webGLVersion)
{
- return createWebGraphicsContext3DProviderInternal(nullptr, scriptState, attributes, webGLVersion);
+ return createContextProviderInternal(nullptr, scriptState, attributes, webGLVersion);
}
void WebGLRenderingContextBase::forceNextWebGLContextCreationToFail()
@@ -5968,7 +6023,7 @@ void WebGLRenderingContextBase::maybeRestoreContext(Timer<WebGLRenderingContextB
Platform::ContextAttributes attributes = toPlatformContextAttributes(m_requestedAttributes, version());
Platform::GraphicsInfo glInfo;
OwnPtr<WebGraphicsContext3DProvider> contextProvider = adoptPtr(Platform::current()->createOffscreenGraphicsContext3DProvider(
- attributes, canvas()->document().topDocument().url(), 0, &glInfo));
+ attributes, canvas()->document().topDocument().url(), 0, &glInfo, Platform::BindToCurrentThread));
RefPtr<DrawingBuffer> buffer;
if (contextProvider) {
// Construct a new drawing buffer with the new GL context.

Powered by Google App Engine
This is Rietveld 408576698