| /* |
| * Copyright (C) 2015 Yusuke Suzuki <utatane.tea@gmail.com>. |
| * |
| * 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. ``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 |
| * 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 "IteratorOperations.h" |
| |
| #include "Error.h" |
| #include "JSCInlines.h" |
| #include "ObjectConstructor.h" |
| |
| using namespace WTF; |
| |
| namespace JSC { |
| |
| JSValue iteratorNext(ExecState* exec, JSValue iterator, JSValue value) |
| { |
| JSValue nextFunction = iterator.get(exec, exec->vm().propertyNames->next); |
| if (exec->hadException()) |
| return jsUndefined(); |
| |
| CallData nextFunctionCallData; |
| CallType nextFunctionCallType = getCallData(nextFunction, nextFunctionCallData); |
| if (nextFunctionCallType == CallTypeNone) |
| return throwTypeError(exec); |
| |
| MarkedArgumentBuffer nextFunctionArguments; |
| if (!value.isEmpty()) |
| nextFunctionArguments.append(value); |
| JSValue result = call(exec, nextFunction, nextFunctionCallType, nextFunctionCallData, iterator, nextFunctionArguments); |
| if (exec->hadException()) |
| return jsUndefined(); |
| |
| if (!result.isObject()) |
| return throwTypeError(exec, ASCIILiteral("Iterator result interface is not an object.")); |
| |
| return result; |
| } |
| |
| JSValue iteratorNext(ExecState* exec, JSValue iterator) |
| { |
| return iteratorNext(exec, iterator, JSValue()); |
| } |
| |
| JSValue iteratorValue(ExecState* exec, JSValue iterResult) |
| { |
| return iterResult.get(exec, exec->vm().propertyNames->value); |
| } |
| |
| bool iteratorComplete(ExecState* exec, JSValue iterResult) |
| { |
| JSValue done = iterResult.get(exec, exec->vm().propertyNames->done); |
| return done.toBoolean(exec); |
| } |
| |
| JSValue iteratorStep(ExecState* exec, JSValue iterator) |
| { |
| JSValue result = iteratorNext(exec, iterator); |
| if (exec->hadException()) |
| return jsUndefined(); |
| bool done = iteratorComplete(exec, result); |
| if (exec->hadException()) |
| return jsUndefined(); |
| if (done) |
| return jsBoolean(false); |
| return result; |
| } |
| |
| void iteratorClose(ExecState* exec, JSValue iterator) |
| { |
| Exception* exception = nullptr; |
| if (exec->hadException()) { |
| exception = exec->exception(); |
| exec->clearException(); |
| } |
| JSValue returnFunction = iterator.get(exec, exec->vm().propertyNames->returnKeyword); |
| if (exec->hadException()) |
| return; |
| |
| if (returnFunction.isUndefined()) { |
| if (exception) |
| exec->vm().throwException(exec, exception); |
| return; |
| } |
| |
| CallData returnFunctionCallData; |
| CallType returnFunctionCallType = getCallData(returnFunction, returnFunctionCallData); |
| if (returnFunctionCallType == CallTypeNone) { |
| if (exception) |
| exec->vm().throwException(exec, exception); |
| else |
| throwTypeError(exec); |
| return; |
| } |
| |
| MarkedArgumentBuffer returnFunctionArguments; |
| JSValue innerResult = call(exec, returnFunction, returnFunctionCallType, returnFunctionCallData, iterator, returnFunctionArguments); |
| |
| if (exception) { |
| exec->vm().throwException(exec, exception); |
| return; |
| } |
| |
| if (exec->hadException()) |
| return; |
| |
| if (!innerResult.isObject()) { |
| throwTypeError(exec, ASCIILiteral("Iterator result interface is not an object.")); |
| return; |
| } |
| } |
| |
| JSObject* createIteratorResultObject(ExecState* exec, JSValue value, bool done) |
| { |
| JSObject* resultObject = constructEmptyObject(exec); |
| resultObject->putDirect(exec->vm(), exec->propertyNames().done, jsBoolean(done)); |
| resultObject->putDirect(exec->vm(), exec->propertyNames().value, value); |
| return resultObject; |
| } |
| |
| } // namespace JSC |