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/JavaScriptCore/runtime/SetConstructor.cpp | |
parent | 32761a6cee1d0dee366b885b7b9c777e67885688 (diff) | |
download | WebKitGtk-tarball-master.tar.gz |
webkitgtk-2.16.5HEADwebkitgtk-2.16.5master
Diffstat (limited to 'Source/JavaScriptCore/runtime/SetConstructor.cpp')
-rw-r--r-- | Source/JavaScriptCore/runtime/SetConstructor.cpp | 75 |
1 files changed, 44 insertions, 31 deletions
diff --git a/Source/JavaScriptCore/runtime/SetConstructor.cpp b/Source/JavaScriptCore/runtime/SetConstructor.cpp index b203888d3..04ae6bb58 100644 --- a/Source/JavaScriptCore/runtime/SetConstructor.cpp +++ b/Source/JavaScriptCore/runtime/SetConstructor.cpp @@ -1,5 +1,5 @@ /* - * Copyright (C) 2013 Apple Inc. All rights reserved. + * Copyright (C) 2013, 2016 Apple Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions @@ -27,62 +27,75 @@ #include "SetConstructor.h" #include "Error.h" -#include "JSCJSValueInlines.h" -#include "JSCellInlines.h" +#include "GetterSetter.h" +#include "IteratorOperations.h" +#include "JSCInlines.h" #include "JSGlobalObject.h" +#include "JSObjectInlines.h" #include "JSSet.h" -#include "MapData.h" #include "SetPrototype.h" namespace JSC { -const ClassInfo SetConstructor::s_info = { "Function", &Base::s_info, 0, 0, CREATE_METHOD_TABLE(SetConstructor) }; +const ClassInfo SetConstructor::s_info = { "Function", &Base::s_info, 0, CREATE_METHOD_TABLE(SetConstructor) }; -void SetConstructor::finishCreation(VM& vm, SetPrototype* setPrototype) +void SetConstructor::finishCreation(VM& vm, SetPrototype* setPrototype, GetterSetter* speciesSymbol) { - Base::finishCreation(vm, setPrototype->classInfo()->className); + Base::finishCreation(vm, setPrototype->classInfo(vm)->className); putDirectWithoutTransition(vm, vm.propertyNames->prototype, setPrototype, DontEnum | DontDelete | ReadOnly); - putDirectWithoutTransition(vm, vm.propertyNames->length, jsNumber(0), ReadOnly | DontEnum | DontDelete); + putDirectWithoutTransition(vm, vm.propertyNames->length, jsNumber(0), DontEnum | ReadOnly); + putDirectNonIndexAccessor(vm, vm.propertyNames->speciesSymbol, speciesSymbol, Accessor | ReadOnly | DontEnum); } -static EncodedJSValue JSC_HOST_CALL callSet(CallFrame* callFrame) +static EncodedJSValue JSC_HOST_CALL callSet(ExecState* exec) { - // Until we have iterators we throw if we've been given - // any arguments that could require us to throw. - if (!callFrame->argument(0).isUndefinedOrNull()) - return JSValue::encode(throwTypeError(callFrame, ASCIILiteral("Set does not accept arguments when called as a function"))); - if (!callFrame->argument(1).isUndefined()) - return throwVMError(callFrame, createRangeError(callFrame, WTF::ASCIILiteral("Invalid comparator function"))); - - JSGlobalObject* globalObject = asInternalFunction(callFrame->callee())->globalObject(); - Structure* setStructure = globalObject->setStructure(); - return JSValue::encode(JSSet::create(callFrame, setStructure)); + VM& vm = exec->vm(); + auto scope = DECLARE_THROW_SCOPE(vm); + return JSValue::encode(throwConstructorCannotBeCalledAsFunctionTypeError(exec, scope, "Set")); } -static EncodedJSValue JSC_HOST_CALL constructSet(CallFrame* callFrame) +static EncodedJSValue JSC_HOST_CALL constructSet(ExecState* exec) { - JSGlobalObject* globalObject = asInternalFunction(callFrame->callee())->globalObject(); - Structure* setStructure = globalObject->setStructure(); - JSSet* set = JSSet::create(callFrame, setStructure); - MapData* mapData = set->mapData(); - size_t count = callFrame->argumentCount(); - for (size_t i = 0; i < count; i++) { - JSValue item = callFrame->uncheckedArgument(i); - mapData->set(callFrame, item, item); - } + VM& vm = exec->vm(); + auto scope = DECLARE_THROW_SCOPE(vm); + + JSGlobalObject* globalObject = asInternalFunction(exec->jsCallee())->globalObject(); + Structure* setStructure = InternalFunction::createSubclassStructure(exec, exec->newTarget(), globalObject->setStructure()); + RETURN_IF_EXCEPTION(scope, encodedJSValue()); + JSSet* set = JSSet::create(exec, vm, setStructure); + RETURN_IF_EXCEPTION(scope, encodedJSValue()); + JSValue iterable = exec->argument(0); + if (iterable.isUndefinedOrNull()) + return JSValue::encode(set); + + JSValue adderFunction = set->get(exec, vm.propertyNames->add); + RETURN_IF_EXCEPTION(scope, encodedJSValue()); + + CallData adderFunctionCallData; + CallType adderFunctionCallType = getCallData(adderFunction, adderFunctionCallData); + if (UNLIKELY(adderFunctionCallType == CallType::None)) + return JSValue::encode(throwTypeError(exec, scope)); + + scope.release(); + forEachInIterable(exec, iterable, [&](VM&, ExecState* exec, JSValue nextValue) { + MarkedArgumentBuffer arguments; + arguments.append(nextValue); + call(exec, adderFunction, adderFunctionCallType, adderFunctionCallData, set, arguments); + }); + return JSValue::encode(set); } ConstructType SetConstructor::getConstructData(JSCell*, ConstructData& constructData) { constructData.native.function = constructSet; - return ConstructTypeHost; + return ConstructType::Host; } CallType SetConstructor::getCallData(JSCell*, CallData& callData) { callData.native.function = callSet; - return CallTypeHost; + return CallType::Host; } } |