WeakMapConstructor.cpp
1 /* 2 * Copyright (C) 2013-2017 Apple, Inc. All rights reserved. 3 * 4 * Redistribution and use in source and binary forms, with or without 5 * modification, are permitted provided that the following conditions 6 * are met: 7 * 1. Redistributions of source code must retain the above copyright 8 * notice, this list of conditions and the following disclaimer. 9 * 2. Redistributions in binary form must reproduce the above copyright 10 * notice, this list of conditions and the following disclaimer in the 11 * documentation and/or other materials provided with the distribution. 12 * 13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY 14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR 17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, 18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, 19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY 21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 24 */ 25 26 #include "config.h" 27 #include "WeakMapConstructor.h" 28 29 #include "IteratorOperations.h" 30 #include "JSCInlines.h" 31 #include "JSWeakMap.h" 32 #include "WeakMapPrototype.h" 33 34 namespace JSC { 35 36 const ClassInfo WeakMapConstructor::s_info = { "Function", &Base::s_info, nullptr, nullptr, CREATE_METHOD_TABLE(WeakMapConstructor) }; 37 38 void WeakMapConstructor::finishCreation(VM& vm, WeakMapPrototype* prototype) 39 { 40 Base::finishCreation(vm, 0, "WeakMap"_s, PropertyAdditionMode::WithoutStructureTransition); 41 putDirectWithoutTransition(vm, vm.propertyNames->prototype, prototype, PropertyAttribute::DontEnum | PropertyAttribute::DontDelete | PropertyAttribute::ReadOnly); 42 } 43 44 static JSC_DECLARE_HOST_FUNCTION(callWeakMap); 45 static JSC_DECLARE_HOST_FUNCTION(constructWeakMap); 46 47 WeakMapConstructor::WeakMapConstructor(VM& vm, Structure* structure) 48 : Base(vm, structure, callWeakMap, constructWeakMap) 49 { 50 } 51 52 JSC_DEFINE_HOST_FUNCTION(callWeakMap, (JSGlobalObject* globalObject, CallFrame*)) 53 { 54 VM& vm = globalObject->vm(); 55 auto scope = DECLARE_THROW_SCOPE(vm); 56 return JSValue::encode(throwConstructorCannotBeCalledAsFunctionTypeError(globalObject, scope, "WeakMap")); 57 } 58 59 JSC_DEFINE_HOST_FUNCTION(constructWeakMap, (JSGlobalObject* globalObject, CallFrame* callFrame)) 60 { 61 VM& vm = globalObject->vm(); 62 auto scope = DECLARE_THROW_SCOPE(vm); 63 64 JSObject* newTarget = asObject(callFrame->newTarget()); 65 Structure* weakMapStructure = newTarget == callFrame->jsCallee() 66 ? globalObject->weakMapStructure() 67 : InternalFunction::createSubclassStructure(globalObject, newTarget, getFunctionRealm(vm, newTarget)->weakMapStructure()); 68 RETURN_IF_EXCEPTION(scope, { }); 69 70 JSWeakMap* weakMap = JSWeakMap::create(vm, weakMapStructure); 71 JSValue iterable = callFrame->argument(0); 72 if (iterable.isUndefinedOrNull()) 73 return JSValue::encode(weakMap); 74 75 JSValue adderFunction = weakMap->JSObject::get(globalObject, vm.propertyNames->set); 76 RETURN_IF_EXCEPTION(scope, encodedJSValue()); 77 78 auto adderFunctionCallData = getCallData(vm, adderFunction); 79 if (adderFunctionCallData.type == CallData::Type::None) 80 return JSValue::encode(throwTypeError(globalObject, scope)); 81 82 scope.release(); 83 forEachInIterable(globalObject, iterable, [&](VM& vm, JSGlobalObject* globalObject, JSValue nextItem) { 84 auto scope = DECLARE_THROW_SCOPE(vm); 85 if (!nextItem.isObject()) { 86 throwTypeError(globalObject, scope); 87 return; 88 } 89 90 JSValue key = nextItem.get(globalObject, static_cast<unsigned>(0)); 91 RETURN_IF_EXCEPTION(scope, void()); 92 93 JSValue value = nextItem.get(globalObject, static_cast<unsigned>(1)); 94 RETURN_IF_EXCEPTION(scope, void()); 95 96 MarkedArgumentBuffer arguments; 97 arguments.append(key); 98 arguments.append(value); 99 ASSERT(!arguments.hasOverflowed()); 100 scope.release(); 101 call(globalObject, adderFunction, adderFunctionCallData, weakMap, arguments); 102 }); 103 104 return JSValue::encode(weakMap); 105 } 106 107 }