This source file includes following definitions.
- m_executionScheduledToTerminate
- create
- postCleanup
- initializeContextIfNeeded
- evaluate
- evaluate
- scheduleExecutionTermination
- isExecutionTerminating
- forbidExecution
- isExecutionForbidden
- disableEval
- rethrowExceptionFromImportedScript
#include "config.h"
#include "bindings/v8/WorkerScriptController.h"
#include "V8DedicatedWorkerGlobalScope.h"
#include "V8ServiceWorkerGlobalScope.h"
#include "V8SharedWorkerGlobalScope.h"
#include "V8WorkerGlobalScope.h"
#include "bindings/v8/ScriptSourceCode.h"
#include "bindings/v8/ScriptValue.h"
#include "bindings/v8/V8ErrorHandler.h"
#include "bindings/v8/V8GCController.h"
#include "bindings/v8/V8Initializer.h"
#include "bindings/v8/V8ObjectConstructor.h"
#include "bindings/v8/V8ScriptRunner.h"
#include "bindings/v8/WrapperTypeInfo.h"
#include "core/inspector/ScriptCallStack.h"
#include "core/frame/DOMTimer.h"
#include "core/workers/SharedWorkerGlobalScope.h"
#include "core/workers/WorkerGlobalScope.h"
#include "core/workers/WorkerObjectProxy.h"
#include "core/workers/WorkerThread.h"
#include "platform/heap/ThreadState.h"
#include <v8.h>
#include "public/platform/Platform.h"
#include "public/platform/WebWorkerRunLoop.h"
namespace WebCore {
WorkerScriptController::WorkerScriptController(WorkerGlobalScope& workerGlobalScope)
: m_isolate(v8::Isolate::New())
, m_workerGlobalScope(workerGlobalScope)
, m_executionForbidden(false)
, m_executionScheduledToTerminate(false)
{
m_isolate->Enter();
V8Initializer::initializeWorker(m_isolate);
v8::V8::Initialize();
V8PerIsolateData::ensureInitialized(m_isolate);
m_world = DOMWrapperWorld::create(WorkerWorldId);
m_interruptor = adoptPtr(new V8IsolateInterruptor(m_isolate));
ThreadState::current()->addInterruptor(m_interruptor.get());
}
class IsolateCleanupTask : public ThreadState::CleanupTask {
public:
static PassOwnPtr<IsolateCleanupTask> create(v8::Isolate* isolate)
{
return adoptPtr(new IsolateCleanupTask(isolate));
}
virtual void postCleanup()
{
V8PerIsolateData::dispose(m_isolate);
m_isolate->Exit();
m_isolate->Dispose();
}
private:
explicit IsolateCleanupTask(v8::Isolate* isolate) : m_isolate(isolate) { }
v8::Isolate* m_isolate;
};
WorkerScriptController::~WorkerScriptController()
{
ThreadState::current()->removeInterruptor(m_interruptor.get());
m_world->dispose();
blink::Platform::current()->didStopWorkerRunLoop(blink::WebWorkerRunLoop(&m_workerGlobalScope.thread()->runLoop()));
if (isContextInitialized())
m_scriptState->disposePerContextData();
ThreadState::current()->addCleanupTask(IsolateCleanupTask::create(m_isolate));
}
bool WorkerScriptController::initializeContextIfNeeded()
{
if (isContextInitialized())
return true;
v8::Handle<v8::Context> context = v8::Context::New(m_isolate);
if (context.IsEmpty())
return false;
m_scriptState = NewScriptState::create(context, m_world);
v8::Context::Scope scope(context);
context->SetEmbedderData(0, v8AtomicString(m_isolate, "worker"));
const WrapperTypeInfo* contextType = &V8DedicatedWorkerGlobalScope::wrapperTypeInfo;
if (m_workerGlobalScope.isServiceWorkerGlobalScope())
contextType = &V8ServiceWorkerGlobalScope::wrapperTypeInfo;
else if (!m_workerGlobalScope.isDedicatedWorkerGlobalScope())
contextType = &V8SharedWorkerGlobalScope::wrapperTypeInfo;
v8::Handle<v8::Function> workerGlobalScopeConstructor = m_scriptState->perContextData()->constructorForType(contextType);
v8::Local<v8::Object> jsWorkerGlobalScope = V8ObjectConstructor::newInstance(m_isolate, workerGlobalScopeConstructor);
if (jsWorkerGlobalScope.IsEmpty()) {
m_scriptState->disposePerContextData();
return false;
}
V8DOMWrapper::associateObjectWithWrapper<V8WorkerGlobalScope>(PassRefPtrWillBeRawPtr<WorkerGlobalScope>(&m_workerGlobalScope), contextType, jsWorkerGlobalScope, m_isolate, WrapperConfiguration::Dependent);
v8::Handle<v8::Object> globalObject = v8::Handle<v8::Object>::Cast(m_scriptState->context()->Global()->GetPrototype());
globalObject->SetPrototype(jsWorkerGlobalScope);
return true;
}
ScriptValue WorkerScriptController::evaluate(const String& script, const String& fileName, const TextPosition& scriptStartPosition, WorkerGlobalScopeExecutionState* state)
{
v8::HandleScope handleScope(m_isolate);
if (!initializeContextIfNeeded())
return ScriptValue();
v8::Handle<v8::Context> context = m_scriptState->context();
if (!m_disableEvalPending.isEmpty()) {
context->AllowCodeGenerationFromStrings(false);
context->SetErrorMessageForCodeGenerationFromStrings(v8String(m_isolate, m_disableEvalPending));
m_disableEvalPending = String();
}
v8::Context::Scope scope(context);
v8::TryCatch block;
v8::Handle<v8::String> scriptString = v8String(m_isolate, script);
v8::Handle<v8::Script> compiledScript = V8ScriptRunner::compileScript(scriptString, fileName, scriptStartPosition, 0, m_isolate);
v8::Local<v8::Value> result = V8ScriptRunner::runCompiledScript(compiledScript, &m_workerGlobalScope, m_isolate);
if (!block.CanContinue()) {
m_workerGlobalScope.script()->forbidExecution();
return ScriptValue();
}
if (block.HasCaught()) {
v8::Local<v8::Message> message = block.Message();
state->hadException = true;
state->errorMessage = toCoreString(message->Get());
state->lineNumber = message->GetLineNumber();
state->columnNumber = message->GetStartColumn() + 1;
V8TRYCATCH_FOR_V8STRINGRESOURCE_RETURN(V8StringResource<>, sourceURL, message->GetScriptResourceName(), ScriptValue());
state->sourceURL = sourceURL;
state->exception = ScriptValue(block.Exception(), m_isolate);
block.Reset();
} else
state->hadException = false;
if (result.IsEmpty() || result->IsUndefined())
return ScriptValue();
return ScriptValue(result, m_isolate);
}
void WorkerScriptController::evaluate(const ScriptSourceCode& sourceCode, RefPtrWillBeRawPtr<ErrorEvent>* errorEvent)
{
if (isExecutionForbidden())
return;
WorkerGlobalScopeExecutionState state;
evaluate(sourceCode.source(), sourceCode.url().string(), sourceCode.startPosition(), &state);
if (state.hadException) {
if (errorEvent) {
*errorEvent = m_workerGlobalScope.shouldSanitizeScriptError(state.sourceURL, NotSharableCrossOrigin) ?
ErrorEvent::createSanitizedError(m_world.get()) : ErrorEvent::create(state.errorMessage, state.sourceURL, state.lineNumber, state.columnNumber, m_world.get());
V8ErrorHandler::storeExceptionOnErrorEventWrapper(errorEvent->get(), state.exception.v8Value(), m_isolate);
} else {
ASSERT(!m_workerGlobalScope.shouldSanitizeScriptError(state.sourceURL, NotSharableCrossOrigin));
RefPtrWillBeRawPtr<ErrorEvent> event = nullptr;
if (m_errorEventFromImportedScript) {
event = m_errorEventFromImportedScript.release();
} else {
event = ErrorEvent::create(state.errorMessage, state.sourceURL, state.lineNumber, state.columnNumber, m_world.get());
}
m_workerGlobalScope.reportException(event, nullptr, NotSharableCrossOrigin);
}
}
}
void WorkerScriptController::scheduleExecutionTermination()
{
{
MutexLocker locker(m_scheduledTerminationMutex);
m_executionScheduledToTerminate = true;
}
v8::V8::TerminateExecution(m_isolate);
}
bool WorkerScriptController::isExecutionTerminating() const
{
MutexLocker locker(m_scheduledTerminationMutex);
return m_executionScheduledToTerminate;
}
void WorkerScriptController::forbidExecution()
{
ASSERT(m_workerGlobalScope.isContextThread());
m_executionForbidden = true;
}
bool WorkerScriptController::isExecutionForbidden() const
{
ASSERT(m_workerGlobalScope.isContextThread());
return m_executionForbidden;
}
void WorkerScriptController::disableEval(const String& errorMessage)
{
m_disableEvalPending = errorMessage;
}
void WorkerScriptController::rethrowExceptionFromImportedScript(PassRefPtrWillBeRawPtr<ErrorEvent> errorEvent)
{
m_errorEventFromImportedScript = errorEvent;
throwError(V8ThrowException::createError(v8GeneralError, m_errorEventFromImportedScript->message(), m_isolate), m_isolate);
}
}