Unreviewed, rolling out r222791 and r222873.
[WebKit-https.git] / Source / JavaScriptCore / wasm / js / WebAssemblyMemoryConstructor.cpp
1 /*
2  * Copyright (C) 2016-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 "WebAssemblyMemoryConstructor.h"
28
29 #if ENABLE(WEBASSEMBLY)
30
31 #include "FunctionPrototype.h"
32 #include "JSCInlines.h"
33 #include "JSWebAssemblyHelpers.h"
34 #include "JSWebAssemblyMemory.h"
35 #include "WasmMemory.h"
36 #include "WasmPageCount.h"
37 #include "WebAssemblyMemoryPrototype.h"
38 #include <wtf/Optional.h>
39
40 #include "WebAssemblyMemoryConstructor.lut.h"
41
42 namespace JSC {
43
44 const ClassInfo WebAssemblyMemoryConstructor::s_info = { "Function", &Base::s_info, &constructorTableWebAssemblyMemory, nullptr, CREATE_METHOD_TABLE(WebAssemblyMemoryConstructor) };
45
46 /* Source for WebAssemblyMemoryConstructor.lut.h
47  @begin constructorTableWebAssemblyMemory
48  @end
49  */
50
51 static EncodedJSValue JSC_HOST_CALL constructJSWebAssemblyMemory(ExecState* exec)
52 {
53     VM& vm = exec->vm();
54     auto throwScope = DECLARE_THROW_SCOPE(vm);
55     if (exec->argumentCount() != 1)
56         return JSValue::encode(throwException(exec, throwScope, createTypeError(exec, ASCIILiteral("WebAssembly.Memory expects exactly one argument"))));
57
58     JSObject* memoryDescriptor;
59     {
60         JSValue argument = exec->argument(0);
61         if (!argument.isObject())
62             return JSValue::encode(throwException(exec, throwScope, createTypeError(exec, ASCIILiteral("WebAssembly.Memory expects its first argument to be an object"))));
63         memoryDescriptor = jsCast<JSObject*>(argument);
64     }
65
66     Wasm::PageCount initialPageCount;
67     {
68         Identifier initial = Identifier::fromString(&vm, "initial");
69         JSValue minSizeValue = memoryDescriptor->get(exec, initial);
70         RETURN_IF_EXCEPTION(throwScope, encodedJSValue());
71         uint32_t size = toNonWrappingUint32(exec, minSizeValue);
72         RETURN_IF_EXCEPTION(throwScope, encodedJSValue());
73         if (!Wasm::PageCount::isValid(size))
74             return JSValue::encode(throwException(exec, throwScope, createRangeError(exec, ASCIILiteral("WebAssembly.Memory 'initial' page count is too large"))));
75         initialPageCount = Wasm::PageCount(size);
76     }
77
78     Wasm::PageCount maximumPageCount;
79     {
80         Identifier maximum = Identifier::fromString(&vm, "maximum");
81         bool hasProperty = memoryDescriptor->hasProperty(exec, maximum);
82         RETURN_IF_EXCEPTION(throwScope, encodedJSValue());
83         if (hasProperty) {
84             JSValue maxSizeValue = memoryDescriptor->get(exec, maximum);
85             RETURN_IF_EXCEPTION(throwScope, encodedJSValue());
86             uint32_t size = toNonWrappingUint32(exec, maxSizeValue);
87             RETURN_IF_EXCEPTION(throwScope, encodedJSValue());
88             if (!Wasm::PageCount::isValid(size))
89                 return JSValue::encode(throwException(exec, throwScope, createRangeError(exec, ASCIILiteral("WebAssembly.Memory 'maximum' page count is too large"))));
90             maximumPageCount = Wasm::PageCount(size);
91
92             if (initialPageCount > maximumPageCount) {
93                 return JSValue::encode(throwException(exec, throwScope,
94                     createRangeError(exec, ASCIILiteral("'maximum' page count must be than greater than or equal to the 'initial' page count"))));
95             }
96         }
97     }
98
99     RefPtr<Wasm::Memory> memory = Wasm::Memory::create(vm, initialPageCount, maximumPageCount);
100     if (!memory)
101         return JSValue::encode(throwException(exec, throwScope, createOutOfMemoryError(exec)));
102
103     auto* jsMemory = JSWebAssemblyMemory::create(exec, vm, exec->lexicalGlobalObject()->WebAssemblyMemoryStructure(), adoptRef(*memory.leakRef()));
104     RETURN_IF_EXCEPTION(throwScope, encodedJSValue());
105
106     return JSValue::encode(jsMemory);
107 }
108
109 static EncodedJSValue JSC_HOST_CALL callJSWebAssemblyMemory(ExecState* exec)
110 {
111     VM& vm = exec->vm();
112     auto throwScope = DECLARE_THROW_SCOPE(vm);
113     return JSValue::encode(throwConstructorCannotBeCalledAsFunctionTypeError(exec, throwScope, "WebAssembly.Memory"));
114 }
115
116 WebAssemblyMemoryConstructor* WebAssemblyMemoryConstructor::create(VM& vm, Structure* structure, WebAssemblyMemoryPrototype* thisPrototype)
117 {
118     auto* constructor = new (NotNull, allocateCell<WebAssemblyMemoryConstructor>(vm.heap)) WebAssemblyMemoryConstructor(vm, structure);
119     constructor->finishCreation(vm, thisPrototype);
120     return constructor;
121 }
122
123 Structure* WebAssemblyMemoryConstructor::createStructure(VM& vm, JSGlobalObject* globalObject, JSValue prototype)
124 {
125     return Structure::create(vm, globalObject, prototype, TypeInfo(ObjectType, StructureFlags), info());
126 }
127
128 void WebAssemblyMemoryConstructor::finishCreation(VM& vm, WebAssemblyMemoryPrototype* prototype)
129 {
130     Base::finishCreation(vm, ASCIILiteral("Memory"));
131     putDirectWithoutTransition(vm, vm.propertyNames->prototype, prototype, PropertyAttribute::DontEnum | PropertyAttribute::DontDelete | PropertyAttribute::ReadOnly);
132     putDirectWithoutTransition(vm, vm.propertyNames->length, jsNumber(1), PropertyAttribute::ReadOnly | PropertyAttribute::DontEnum | PropertyAttribute::DontDelete);
133 }
134
135 WebAssemblyMemoryConstructor::WebAssemblyMemoryConstructor(VM& vm, Structure* structure)
136     : Base(vm, structure)
137 {
138 }
139
140 ConstructType WebAssemblyMemoryConstructor::getConstructData(JSCell*, ConstructData& constructData)
141 {
142     constructData.native.function = constructJSWebAssemblyMemory;
143     return ConstructType::Host;
144 }
145
146 CallType WebAssemblyMemoryConstructor::getCallData(JSCell*, CallData& callData)
147 {
148     callData.native.function = callJSWebAssemblyMemory;
149     return CallType::Host;
150 }
151
152 } // namespace JSC
153
154 #endif // ENABLE(WEBASSEMBLY)
155