c3a4041321d4b28ede33b41f44031877430b9f1e
[WebKit-https.git] / Source / JavaScriptCore / wasm / WasmMemoryInformation.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 "WasmMemoryInformation.h"
28
29 #if ENABLE(WEBASSEMBLY)
30
31 #include "WasmCallingConvention.h"
32 #include "WasmContextInlines.h"
33 #include "WasmMemory.h"
34 #include <wtf/NeverDestroyed.h>
35
36 namespace JSC { namespace Wasm {
37
38 static Vector<GPRReg> getPinnedRegisters(unsigned remainingPinnedRegisters)
39 {
40     Vector<GPRReg> registers;
41     jscCallingConvention().m_calleeSaveRegisters.forEach([&] (Reg reg) {
42         if (!reg.isGPR())
43             return;
44         GPRReg gpr = reg.gpr();
45         if (!remainingPinnedRegisters || RegisterSet::stackRegisters().get(reg))
46             return;
47         --remainingPinnedRegisters;
48         registers.append(gpr);
49     });
50     return registers;
51 }
52
53 const PinnedRegisterInfo& PinnedRegisterInfo::get()
54 {
55     static LazyNeverDestroyed<PinnedRegisterInfo> staticPinnedRegisterInfo;
56     static std::once_flag staticPinnedRegisterInfoFlag;
57     std::call_once(staticPinnedRegisterInfoFlag, [] () {
58         Vector<PinnedSizeRegisterInfo> sizeRegisters;
59         GPRReg baseMemoryPointer = InvalidGPRReg;
60         GPRReg wasmContextInstancePointer = InvalidGPRReg;
61
62         // FIXME: We should support more than one memory size register, and we should allow different
63         //        WebAssembly.Instance to have different pins. Right now we take a vector with only one entry.
64         //        If we have more than one size register, we can have one for each load size class.
65         //        see: https://bugs.webkit.org/show_bug.cgi?id=162952
66         Vector<unsigned> pinnedSizes = { 0 };
67         unsigned numberOfPinnedRegisters = pinnedSizes.size() + 1;
68         if (!Context::useFastTLS())
69             ++numberOfPinnedRegisters;
70         Vector<GPRReg> pinnedRegs = getPinnedRegisters(numberOfPinnedRegisters);
71
72         baseMemoryPointer = pinnedRegs.takeLast();
73         if (!Context::useFastTLS())
74             wasmContextInstancePointer = pinnedRegs.takeLast();
75
76         ASSERT(pinnedSizes.size() == pinnedRegs.size());
77         for (unsigned i = 0; i < pinnedSizes.size(); ++i)
78             sizeRegisters.append({ pinnedRegs[i], pinnedSizes[i] });
79         staticPinnedRegisterInfo.construct(WTFMove(sizeRegisters), baseMemoryPointer, wasmContextInstancePointer);
80     });
81
82     return staticPinnedRegisterInfo.get();
83 }
84
85 PinnedRegisterInfo::PinnedRegisterInfo(Vector<PinnedSizeRegisterInfo>&& sizeRegisters, GPRReg baseMemoryPointer, GPRReg wasmContextInstancePointer)
86     : sizeRegisters(WTFMove(sizeRegisters))
87     , baseMemoryPointer(baseMemoryPointer)
88     , wasmContextInstancePointer(wasmContextInstancePointer)
89 {
90 }
91
92 MemoryInformation::MemoryInformation(PageCount initial, PageCount maximum, bool isImport)
93     : m_initial(initial)
94     , m_maximum(maximum)
95     , m_isImport(isImport)
96 {
97     RELEASE_ASSERT(!!m_initial);
98     RELEASE_ASSERT(!m_maximum || m_maximum >= m_initial);
99     ASSERT(!!*this);
100 }
101
102 } } // namespace JSC::Wasm
103
104 #endif // ENABLE(WEBASSEMBLY)