blob: 368b5981b02dfc9c9213b410058eaa4208babf1d [file] [log] [blame]
/*
* Copyright (C) 2013 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "JSPromiseResolver.h"
#if ENABLE(PROMISES)
#include "JSCJSValueInlines.h"
#include "JSCellInlines.h"
#include "JSGlobalObject.h"
#include "JSPromise.h"
#include "JSPromiseCallback.h"
#include "SlotVisitorInlines.h"
#include "StructureInlines.h"
namespace JSC {
const ClassInfo JSPromiseResolver::s_info = { "PromiseResolver", &Base::s_info, 0, 0, CREATE_METHOD_TABLE(JSPromiseResolver) };
JSPromiseResolver* JSPromiseResolver::create(VM& vm, Structure* structure, JSPromise* promise)
{
JSPromiseResolver* object = new (NotNull, allocateCell<JSPromiseResolver>(vm.heap)) JSPromiseResolver(vm, structure);
object->finishCreation(vm, promise);
return object;
}
Structure* JSPromiseResolver::createStructure(VM& vm, JSGlobalObject* globalObject, JSValue prototype)
{
return Structure::create(vm, globalObject, prototype, TypeInfo(ObjectType, StructureFlags), info());
}
JSPromiseResolver::JSPromiseResolver(VM& vm, Structure* structure)
: JSNonFinalObject(vm, structure)
, m_isResolved(false)
{
}
void JSPromiseResolver::finishCreation(VM& vm, JSPromise* promise)
{
Base::finishCreation(vm);
ASSERT(inherits(info()));
m_promise.set(vm, this, promise);
}
void JSPromiseResolver::visitChildren(JSCell* cell, SlotVisitor& visitor)
{
JSPromiseResolver* thisObject = jsCast<JSPromiseResolver*>(cell);
ASSERT_GC_OBJECT_INHERITS(thisObject, info());
COMPILE_ASSERT(StructureFlags & OverridesVisitChildren, OverridesVisitChildrenWithoutSettingFlag);
ASSERT(thisObject->structure()->typeInfo().overridesVisitChildren());
Base::visitChildren(thisObject, visitor);
visitor.append(&thisObject->m_promise);
}
JSPromise* JSPromiseResolver::promise() const
{
return m_promise.get();
}
void JSPromiseResolver::fulfillIfNotResolved(ExecState* exec, JSValue value)
{
if (!m_isResolved) {
m_isResolved = true;
fulfill(exec, value);
}
}
void JSPromiseResolver::resolveIfNotResolved(ExecState* exec, JSValue value)
{
if (!m_isResolved) {
m_isResolved = true;
resolve(exec, value);
}
}
void JSPromiseResolver::rejectIfNotResolved(ExecState* exec, JSValue value)
{
if (!m_isResolved) {
m_isResolved = true;
reject(exec, value);
}
}
void JSPromiseResolver::fulfill(ExecState* exec, JSValue value, ResolverMode mode)
{
// 1. Let promise be the context object's associated promise.
// 2. Set promise's state to fulfilled.
m_promise->setState(JSPromise::Fulfilled);
// 3. Set promise's result to value.
m_promise->setResult(exec->vm(), value);
// 4. If the synchronous flag is set, process promise's fulfill callbacks with value.
if (mode == ResolveSynchronously) {
m_promise->processFulfillCallbacksWithValue(exec, value);
return;
}
// 5. Otherwise, the synchronous flag is unset, queue a task to process promise's fulfill callbacks with value.
m_promise->queueTaskToProcessFulfillCallbacks(exec);
}
void JSPromiseResolver::resolve(ExecState* exec, JSValue value, ResolverMode mode)
{
// 1. Let then be null.
JSValue then = jsNull();
// 2. If value is a JavaScript Object, set then to the result of calling the JavaScript [[Get]] internal
// method of value with property name then.
if (value.isObject()) {
then = value.get(exec, exec->propertyNames().then);
// 3. If calling the [[Get]] internal method threw an exception, catch it and run reject with the thrown
// exception and the synchronous flag if set, and then terminate these steps.
if (exec->hadException()) {
JSValue exception = exec->exception();
exec->clearException();
reject(exec, exception, mode);
return;
}
// 4. If JavaScript IsCallable(then) is true, run these substeps and then terminate these steps:
CallData callData;
CallType callType = JSC::getCallData(then, callData);
if (callType != CallTypeNone) {
VM& vm = exec->vm();
// 4.1. Let fulfillCallback be a promise callback for the context object and its resolve algorithm.
JSPromiseCallback* fulfillCallback = JSPromiseCallback::create(vm, globalObject()->promiseCallbackStructure(), this, JSPromiseCallback::Resolve);
// 4.2. Let rejectCallback be a promise callback for the context object and its reject algorithm.
JSPromiseCallback* rejectCallback = JSPromiseCallback::create(vm, globalObject()->promiseCallbackStructure(), this, JSPromiseCallback::Reject);
// 4.3. Call the JavaScript [[Call]] internal method of then with this value value and fulfillCallback
// and rejectCallback as arguments.
MarkedArgumentBuffer thenArguments;
thenArguments.append(fulfillCallback);
thenArguments.append(rejectCallback);
call(exec, then, callType, callData, value, thenArguments);
// 4.4 If calling the [[Call]] internal method threw an exception, catch it and run context object's
// reject with the thrown exception and the synchronous flag if set.
if (exec->hadException()) {
JSValue exception = exec->exception();
exec->clearException();
reject(exec, exception, mode);
return;
}
return;
}
}
// 5. Run context object's fulfill with value and the synchronous flag if set.
fulfill(exec, value, mode);
}
void JSPromiseResolver::reject(ExecState* exec, JSValue value, ResolverMode mode)
{
// 1. Let promise be the context object's associated promise.
// 2. Set promise's state to rejected.
m_promise->setState(JSPromise::Rejected);
// 3. Set promise's result to value.
m_promise->setResult(exec->vm(), value);
// 4. If the synchronous flag is set, process promise's reject callbacks with value.
if (mode == ResolveSynchronously) {
m_promise->processRejectCallbacksWithValue(exec, value);
return;
}
// 5. Otherwise, the synchronous flag is unset, queue a task to process promise's reject callbacks with value.
m_promise->queueTaskToProcessRejectCallbacks(exec);
}
} // namespace JSC
#endif // ENABLE(PROMISES)