blob: 80e17ea00189a7c7be5d13bb817bd71e28eb40e1 [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 "JSPromise.h"
#if ENABLE(PROMISES)
#include "Error.h"
#include "JSCJSValueInlines.h"
#include "JSCellInlines.h"
#include "JSPromiseResolver.h"
#include "SlotVisitorInlines.h"
#include "StrongInlines.h"
#include "StructureInlines.h"
namespace JSC {
class JSPromiseTaskContext : public TaskContext {
public:
static PassRefPtr<JSPromiseTaskContext> create(VM& vm, JSPromise* promise)
{
return adoptRef(new JSPromiseTaskContext(vm, promise));
}
JSPromise& promise() const { return *m_promise.get(); }
private:
JSPromiseTaskContext(VM& vm, JSPromise* promise)
{
m_promise.set(vm, promise);
}
Strong<JSPromise> m_promise;
};
const ClassInfo JSPromise::s_info = { "Promise", &Base::s_info, 0, 0, CREATE_METHOD_TABLE(JSPromise) };
JSPromise* JSPromise::create(VM& vm, Structure* structure)
{
JSPromise* promise = new (NotNull, allocateCell<JSPromise>(vm.heap)) JSPromise(vm, structure);
promise->finishCreation(vm);
return promise;
}
JSPromise* JSPromise::createWithResolver(VM& vm, JSGlobalObject* globalObject)
{
JSPromise* promise = new (NotNull, allocateCell<JSPromise>(vm.heap)) JSPromise(vm, globalObject->promiseStructure());
promise->finishCreation(vm);
JSPromiseResolver* resolver = JSPromiseResolver::create(vm, globalObject->promiseResolverStructure(), promise);
promise->setResolver(vm, resolver);
return promise;
}
Structure* JSPromise::createStructure(VM& vm, JSGlobalObject* globalObject, JSValue prototype)
{
return Structure::create(vm, globalObject, prototype, TypeInfo(ObjectType, StructureFlags), info());
}
JSPromise::JSPromise(VM& vm, Structure* structure)
: JSDestructibleObject(vm, structure)
, m_state(Pending)
{
}
void JSPromise::finishCreation(VM& vm)
{
Base::finishCreation(vm);
ASSERT(inherits(info()));
}
void JSPromise::destroy(JSCell* cell)
{
static_cast<JSPromise*>(cell)->JSPromise::~JSPromise();
}
void JSPromise::visitChildren(JSCell* cell, SlotVisitor& visitor)
{
JSPromise* thisObject = jsCast<JSPromise*>(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_resolver);
visitor.append(&thisObject->m_result);
for (size_t i = 0; i < thisObject->m_fulfillCallbacks.size(); ++i)
visitor.append(&thisObject->m_fulfillCallbacks[i]);
for (size_t i = 0; i < thisObject->m_rejectCallbacks.size(); ++i)
visitor.append(&thisObject->m_rejectCallbacks[i]);
}
void JSPromise::setResolver(VM& vm, JSPromiseResolver* resolver)
{
m_resolver.set(vm, this, resolver);
}
JSPromiseResolver* JSPromise::resolver() const
{
return m_resolver.get();
}
void JSPromise::setResult(VM& vm, JSValue result)
{
m_result.set(vm, this, result);
}
JSValue JSPromise::result() const
{
return m_result.get();
}
void JSPromise::setState(JSPromise::State state)
{
ASSERT(m_state == Pending);
m_state = state;
}
JSPromise::State JSPromise::state() const
{
return m_state;
}
void JSPromise::appendCallbacks(ExecState* exec, InternalFunction* fulfillCallback, InternalFunction* rejectCallback)
{
// 1. Append fulfillCallback to promise's fulfill callbacks.
m_fulfillCallbacks.append(WriteBarrier<InternalFunction>(exec->vm(), this, fulfillCallback));
// 2. Append rejectCallback to promise' reject callbacks.
m_rejectCallbacks.append(WriteBarrier<InternalFunction>(exec->vm(), this, rejectCallback));
// 3. If promise's state is fulfilled, queue a task to process promise's fulfill callbacks with promise's result.
if (m_state == Fulfilled) {
queueTaskToProcessFulfillCallbacks(exec);
return;
}
// 4. If promise's state is rejected, queue a task to process promise's reject callbacks with promise's result.
if (m_state == Rejected) {
queueTaskToProcessRejectCallbacks(exec);
return;
}
}
void JSPromise::queueTaskToProcessFulfillCallbacks(ExecState* exec)
{
JSGlobalObject* globalObject = this->globalObject();
if (globalObject->globalObjectMethodTable()->queueTaskToEventLoop)
globalObject->globalObjectMethodTable()->queueTaskToEventLoop(globalObject, processFulfillCallbacksForTask, JSPromiseTaskContext::create(exec->vm(), this));
else
WTFLogAlways("ERROR: Event loop not supported.");
}
void JSPromise::queueTaskToProcessRejectCallbacks(ExecState* exec)
{
JSGlobalObject* globalObject = this->globalObject();
if (globalObject->globalObjectMethodTable()->queueTaskToEventLoop)
globalObject->globalObjectMethodTable()->queueTaskToEventLoop(globalObject, processRejectCallbacksForTask, JSPromiseTaskContext::create(exec->vm(), this));
else
WTFLogAlways("ERROR: Event loop not supported.");
}
void JSPromise::processFulfillCallbacksForTask(ExecState* exec, TaskContext* taskContext)
{
JSPromiseTaskContext* promiseTaskContext = static_cast<JSPromiseTaskContext*>(taskContext);
JSPromise& promise = promiseTaskContext->promise();
promise.processFulfillCallbacksWithValue(exec, promise.result());
}
void JSPromise::processRejectCallbacksForTask(ExecState* exec, TaskContext* taskContext)
{
JSPromiseTaskContext* promiseTaskContext = static_cast<JSPromiseTaskContext*>(taskContext);
JSPromise& promise = promiseTaskContext->promise();
promise.processRejectCallbacksWithValue(exec, promise.result());
}
void JSPromise::processFulfillCallbacksWithValue(ExecState* exec, JSValue value)
{
ASSERT(m_state == Fulfilled);
for (size_t i = 0; i < m_fulfillCallbacks.size(); ++i) {
JSValue callback = m_fulfillCallbacks[i].get();
CallData callData;
CallType callType = JSC::getCallData(callback, callData);
ASSERT(callType != CallTypeNone);
MarkedArgumentBuffer arguments;
arguments.append(value);
call(exec, callback, callType, callData, this, arguments);
}
m_fulfillCallbacks.clear();
}
void JSPromise::processRejectCallbacksWithValue(ExecState* exec, JSValue value)
{
ASSERT(m_state == Rejected);
for (size_t i = 0; i < m_rejectCallbacks.size(); ++i) {
JSValue callback = m_rejectCallbacks[i].get();
CallData callData;
CallType callType = JSC::getCallData(callback, callData);
ASSERT(callType != CallTypeNone);
MarkedArgumentBuffer arguments;
arguments.append(value);
call(exec, callback, callType, callData, this, arguments);
}
m_rejectCallbacks.clear();
}
} // namespace JSC
#endif // ENABLE(PROMISES)