diff options
author | Lorry Tar Creator <lorry-tar-importer@lorry> | 2017-06-27 06:07:23 +0000 |
---|---|---|
committer | Lorry Tar Creator <lorry-tar-importer@lorry> | 2017-06-27 06:07:23 +0000 |
commit | 1bf1084f2b10c3b47fd1a588d85d21ed0eb41d0c (patch) | |
tree | 46dcd36c86e7fbc6e5df36deb463b33e9967a6f7 /Source/WebCore/bindings/js/JSMessageEventCustom.cpp | |
parent | 32761a6cee1d0dee366b885b7b9c777e67885688 (diff) | |
download | WebKitGtk-tarball-master.tar.gz |
webkitgtk-2.16.5HEADwebkitgtk-2.16.5master
Diffstat (limited to 'Source/WebCore/bindings/js/JSMessageEventCustom.cpp')
-rw-r--r-- | Source/WebCore/bindings/js/JSMessageEventCustom.cpp | 110 |
1 files changed, 65 insertions, 45 deletions
diff --git a/Source/WebCore/bindings/js/JSMessageEventCustom.cpp b/Source/WebCore/bindings/js/JSMessageEventCustom.cpp index c84ca0d9a..e0886ab25 100644 --- a/Source/WebCore/bindings/js/JSMessageEventCustom.cpp +++ b/Source/WebCore/bindings/js/JSMessageEventCustom.cpp @@ -33,9 +33,10 @@ #include "JSBlob.h" #include "JSDOMBinding.h" +#include "JSDOMConvert.h" #include "JSDOMWindow.h" #include "JSEventTarget.h" -#include "JSMessagePortCustom.h" +#include "JSMessagePort.h" #include "MessageEvent.h" #include <runtime/JSArray.h> #include <runtime/JSArrayBuffer.h> @@ -44,85 +45,104 @@ using namespace JSC; namespace WebCore { -JSValue JSMessageEvent::data(ExecState* exec) const +JSValue JSMessageEvent::data(ExecState& state) const { - if (JSValue cachedValue = m_data.get()) - return cachedValue; + if (JSValue cachedValue = m_data.get()) { + // We cannot use a cached object if we are in a different world than the one it was created in. + if (!cachedValue.isObject() || &worldForDOMObject(cachedValue.getObject()) == ¤tWorld(&state)) + return cachedValue; + ASSERT_NOT_REACHED(); + } - MessageEvent& event = impl(); + MessageEvent& event = wrapped(); JSValue result; switch (event.dataType()) { case MessageEvent::DataTypeScriptValue: { - Deprecated::ScriptValue scriptValue = event.dataAsScriptValue(); - if (scriptValue.hasNoValue()) + JSValue dataValue = event.dataAsScriptValue(); + if (!dataValue) result = jsNull(); - else - result = scriptValue.jsValue(); + else { + // We need to make sure MessageEvents do not leak objects in their state property across isolated DOM worlds. + // Ideally, we would check that the worlds have different privileges but that's not possible yet. + if (dataValue.isObject() && &worldForDOMObject(dataValue.getObject()) != ¤tWorld(&state)) { + RefPtr<SerializedScriptValue> serializedValue = event.trySerializeData(&state); + if (serializedValue) + result = serializedValue->deserialize(state, globalObject()); + else + result = jsNull(); + } else + result = dataValue; + } break; } case MessageEvent::DataTypeSerializedScriptValue: if (RefPtr<SerializedScriptValue> serializedValue = event.dataAsSerializedScriptValue()) { - MessagePortArray ports = impl().ports(); - result = serializedValue->deserialize(exec, globalObject(), &ports, NonThrowing); - } - else + Vector<RefPtr<MessagePort>> ports = wrapped().ports(); + // FIXME: Why does this suppress exceptions? + result = serializedValue->deserialize(state, globalObject(), ports, SerializationErrorMode::NonThrowing); + } else result = jsNull(); break; case MessageEvent::DataTypeString: - result = jsStringWithCache(exec, event.dataAsString()); + result = jsStringWithCache(&state, event.dataAsString()); break; case MessageEvent::DataTypeBlob: - result = toJS(exec, globalObject(), event.dataAsBlob()); + result = toJS(&state, globalObject(), event.dataAsBlob()); break; case MessageEvent::DataTypeArrayBuffer: - result = toJS(exec, globalObject(), event.dataAsArrayBuffer()); + result = toJS(&state, globalObject(), event.dataAsArrayBuffer()); break; } // Save the result so we don't have to deserialize the value again. - const_cast<JSMessageEvent*>(this)->m_data.set(exec->vm(), this, result); + m_data.set(state.vm(), this, result); return result; } -static JSC::JSValue handleInitMessageEvent(JSMessageEvent* jsEvent, JSC::ExecState* exec) +static JSC::JSValue handleInitMessageEvent(JSMessageEvent* jsEvent, JSC::ExecState& state) { - const String& typeArg = exec->argument(0).toString(exec)->value(exec); - bool canBubbleArg = exec->argument(1).toBoolean(exec); - bool cancelableArg = exec->argument(2).toBoolean(exec); - const String originArg = exec->argument(4).toString(exec)->value(exec); - const String lastEventIdArg = exec->argument(5).toString(exec)->value(exec); - DOMWindow* sourceArg = toDOMWindow(exec->argument(6)); - OwnPtr<MessagePortArray> messagePorts; - OwnPtr<ArrayBufferArray> arrayBuffers; - if (!exec->argument(7).isUndefinedOrNull()) { - messagePorts = adoptPtr(new MessagePortArray); - arrayBuffers = adoptPtr(new ArrayBufferArray); - fillMessagePortArray(exec, exec->argument(7), *messagePorts, *arrayBuffers); - if (exec->hadException()) - return jsUndefined(); + VM& vm = state.vm(); + auto scope = DECLARE_THROW_SCOPE(vm); + + const String& typeArg = state.argument(0).toWTFString(&state); + RETURN_IF_EXCEPTION(scope, JSValue()); + + bool canBubbleArg = state.argument(1).toBoolean(&state); + RETURN_IF_EXCEPTION(scope, JSValue()); + + bool cancelableArg = state.argument(2).toBoolean(&state); + RETURN_IF_EXCEPTION(scope, JSValue()); + + JSValue dataArg = state.argument(3); + + const String originArg = convert<IDLUSVString>(state, state.argument(4)); + RETURN_IF_EXCEPTION(scope, JSValue()); + + const String lastEventIdArg = state.argument(5).toWTFString(&state); + RETURN_IF_EXCEPTION(scope, JSValue()); + + auto sourceArg = convert<IDLNullable<IDLUnion<IDLInterface<DOMWindow>, IDLInterface<MessagePort>>>>(state, state.argument(6)); + RETURN_IF_EXCEPTION(scope, JSValue()); + + Vector<RefPtr<MessagePort>> messagePorts; + if (!state.argument(7).isUndefinedOrNull()) { + messagePorts = convert<IDLSequence<IDLInterface<MessagePort>>>(state, state.argument(7)); + RETURN_IF_EXCEPTION(scope, JSValue()); } - Deprecated::ScriptValue dataArg = Deprecated::ScriptValue(exec->vm(), exec->argument(3)); - if (exec->hadException()) - return jsUndefined(); - MessageEvent& event = jsEvent->impl(); - event.initMessageEvent(typeArg, canBubbleArg, cancelableArg, dataArg, originArg, lastEventIdArg, sourceArg, messagePorts.release()); - jsEvent->m_data.set(exec->vm(), jsEvent, dataArg.jsValue()); + MessageEvent& event = jsEvent->wrapped(); + event.initMessageEvent(state, typeArg, canBubbleArg, cancelableArg, dataArg, originArg, lastEventIdArg, WTFMove(sourceArg), WTFMove(messagePorts)); + jsEvent->m_data.set(vm, jsEvent, dataArg); return jsUndefined(); } -JSC::JSValue JSMessageEvent::initMessageEvent(JSC::ExecState* exec) -{ - return handleInitMessageEvent(this, exec); -} - -JSC::JSValue JSMessageEvent::webkitInitMessageEvent(JSC::ExecState* exec) +JSC::JSValue JSMessageEvent::webkitInitMessageEvent(JSC::ExecState& state) { - return handleInitMessageEvent(this, exec); + return handleInitMessageEvent(this, state); } } // namespace WebCore |