1/*
2 * Copyright (C) 2017-2019 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 "WasmInstance.h"
28
29#if ENABLE(WEBASSEMBLY)
30
31#include "JSCInlines.h"
32#include "JSWebAssemblyHelpers.h"
33#include "JSWebAssemblyInstance.h"
34#include "Register.h"
35#include "WasmModuleInformation.h"
36#include <wtf/CheckedArithmetic.h>
37
38namespace JSC { namespace Wasm {
39
40namespace {
41size_t globalMemoryByteSize(Module& module)
42{
43 return (Checked<size_t>(module.moduleInformation().globals.size()) * sizeof(Register)).unsafeGet();
44}
45}
46
47Instance::Instance(Context* context, Ref<Module>&& module, EntryFrame** pointerToTopEntryFrame, void** pointerToActualStackLimit, StoreTopCallFrameCallback&& storeTopCallFrame)
48 : m_context(context)
49 , m_module(WTFMove(module))
50 , m_globals(MallocPtr<GlobalValue>::malloc(globalMemoryByteSize(m_module.get())))
51 , m_globalsToMark(m_module.get().moduleInformation().globals.size())
52 , m_pointerToTopEntryFrame(pointerToTopEntryFrame)
53 , m_pointerToActualStackLimit(pointerToActualStackLimit)
54 , m_storeTopCallFrame(WTFMove(storeTopCallFrame))
55 , m_numImportFunctions(m_module->moduleInformation().importFunctionCount())
56{
57 for (unsigned i = 0; i < m_numImportFunctions; ++i)
58 new (importFunctionInfo(i)) ImportFunctionInfo();
59 memset(static_cast<void*>(m_globals.get()), 0, globalMemoryByteSize(m_module.get()));
60 for (unsigned i = 0; i < m_module->moduleInformation().globals.size(); ++i) {
61 if (isSubtype(m_module.get().moduleInformation().globals[i].type, Anyref))
62 m_globalsToMark.set(i);
63 }
64 memset(bitwise_cast<char*>(this) + offsetOfTablePtr(m_numImportFunctions, 0), 0, m_module->moduleInformation().tableCount() * sizeof(Table*));
65}
66
67Ref<Instance> Instance::create(Context* context, Ref<Module>&& module, EntryFrame** pointerToTopEntryFrame, void** pointerToActualStackLimit, StoreTopCallFrameCallback&& storeTopCallFrame)
68{
69 return adoptRef(*new (NotNull, fastMalloc(allocationSize(module->moduleInformation().importFunctionCount(), module->moduleInformation().tableCount()))) Instance(context, WTFMove(module), pointerToTopEntryFrame, pointerToActualStackLimit, WTFMove(storeTopCallFrame)));
70}
71
72Instance::~Instance() { }
73
74size_t Instance::extraMemoryAllocated() const
75{
76 return globalMemoryByteSize(m_module.get()) + allocationSize(m_numImportFunctions, m_module->moduleInformation().tableCount());
77}
78
79void Instance::setGlobal(unsigned i, JSValue value)
80{
81 ASSERT(m_owner);
82 m_globals.get()[i].anyref.set(owner<JSWebAssemblyInstance>()->vm(), owner<JSWebAssemblyInstance>(), value);
83}
84
85JSValue Instance::getFunctionWrapper(unsigned i) const
86{
87 JSValue value = m_functionWrappers.get(i).get();
88 if (value.isEmpty())
89 return jsNull();
90 return value;
91}
92
93void Instance::setFunctionWrapper(unsigned i, JSValue value)
94{
95 ASSERT(m_owner);
96 ASSERT(value.isFunction(owner<JSWebAssemblyInstance>()->vm()));
97 ASSERT(!m_functionWrappers.contains(i));
98 auto locker = holdLock(owner<JSWebAssemblyInstance>()->cellLock());
99 m_functionWrappers.set(i, WriteBarrier<Unknown>(owner<JSWebAssemblyInstance>()->vm(), owner<JSWebAssemblyInstance>(), value));
100 ASSERT(getFunctionWrapper(i) == value);
101}
102
103Table* Instance::table(unsigned i)
104{
105 RELEASE_ASSERT(i < m_module->moduleInformation().tableCount());
106 return *bitwise_cast<Table**>(bitwise_cast<char*>(this) + offsetOfTablePtr(m_numImportFunctions, i));
107}
108
109void Instance::setTable(unsigned i, Ref<Table>&& table)
110{
111 RELEASE_ASSERT(i < m_module->moduleInformation().tableCount());
112 ASSERT(!this->table(i));
113 *bitwise_cast<Table**>(bitwise_cast<char*>(this) + offsetOfTablePtr(m_numImportFunctions, i)) = &table.leakRef();
114}
115
116} } // namespace JSC::Wasm
117
118#endif // ENABLE(WEBASSEMBLY)
119