blob: 327289949add844131d0d2d626ff58530bd121db [file] [log] [blame]
jfbastien@apple.com148b8d92016-12-08 21:09:06 +00001/*
mark.lam@apple.com3a72ba22021-05-10 23:21:41 +00002 * Copyright (C) 2016-2021 Apple Inc. All rights reserved.
jfbastien@apple.com148b8d92016-12-08 21:09:06 +00003 *
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 "WasmBinding.h"
28
29#if ENABLE(WEBASSEMBLY)
30
sbarati@apple.com6128a582017-01-26 02:38:41 +000031#include "CCallHelpers.h"
commit-queue@webkit.org0f5197e2022-06-10 09:37:34 +000032#include "DisallowMacroScratchRegisterUsage.h"
jfbastien@apple.com148b8d92016-12-08 21:09:06 +000033#include "LinkBuffer.h"
aperez@igalia.com0d1c4f22019-05-23 15:47:10 +000034#include "WasmCallingConvention.h"
jfbastien@apple.com89177d32017-10-26 15:33:55 +000035#include "WasmInstance.h"
jfbastien@apple.com148b8d92016-12-08 21:09:06 +000036
37namespace JSC { namespace Wasm {
38
sbarati@apple.comafec36b2017-03-29 06:15:23 +000039using JIT = CCallHelpers;
jfbastien@apple.com373aa052017-01-03 01:57:40 +000040
mark.lam@apple.comde0dba72018-04-18 03:31:09 +000041Expected<MacroAssemblerCodeRef<WasmEntryPtrTag>, BindingFailure> wasmToWasm(unsigned importIndex)
jfbastien@apple.com373aa052017-01-03 01:57:40 +000042{
mark.lam@apple.coma61c6c72018-03-30 16:31:06 +000043 // FIXME: Consider uniquify the stubs based on signature + index to see if this saves memory.
44 // https://bugs.webkit.org/show_bug.cgi?id=184157
sbarati@apple.comafec36b2017-03-29 06:15:23 +000045 JIT jit;
jfbastien@apple.com373aa052017-01-03 01:57:40 +000046
keith_miller@apple.com8e7bd482019-10-01 16:38:26 +000047 GPRReg scratch = wasmCallingConvention().prologueScratchGPRs[0];
commit-queue@webkit.org0f5197e2022-06-10 09:37:34 +000048#if USE(JSVALUE32_64)
49 GPRReg baseMemory = wasmCallingConvention().prologueScratchGPRs[1];
50 GPRReg sizeRegAsScratch = wasmCallingConvention().prologueScratchGPRs[2];
51#else
52 const PinnedRegisterInfo& pinnedRegs = PinnedRegisterInfo::get();
jfbastien@apple.com373aa052017-01-03 01:57:40 +000053 GPRReg baseMemory = pinnedRegs.baseMemoryPointer;
mark.lam@apple.comb058b152022-05-27 23:09:30 +000054 GPRReg sizeRegAsScratch = pinnedRegs.boundsCheckingSizeRegister;
commit-queue@webkit.org0f5197e2022-06-10 09:37:34 +000055#endif
56 ASSERT(baseMemory != GPRReg::InvalidGPRReg);
keith_miller@apple.com8e7bd482019-10-01 16:38:26 +000057 ASSERT(sizeRegAsScratch != GPRReg::InvalidGPRReg);
commit-queue@webkit.org0f5197e2022-06-10 09:37:34 +000058 ASSERT(scratch != GPRReg::InvalidGPRReg);
59 ASSERT(noOverlap(scratch, baseMemory, sizeRegAsScratch));
sbarati@apple.com83ea2fc2017-05-18 19:38:10 +000060
sbarati@apple.com83ea2fc2017-05-18 19:38:10 +000061 // B3's call codegen ensures that the JSCell is a WebAssemblyFunction.
jfbastien@apple.comd9f999e2017-10-20 02:23:29 +000062 jit.loadWasmContextInstance(sizeRegAsScratch); // Old Instance*
jfbastien@apple.com89177d32017-10-26 15:33:55 +000063 // Get the callee's Wasm::Instance and set it as WasmContext's instance. The caller will take care of restoring its own Instance.
64 jit.loadPtr(JIT::Address(sizeRegAsScratch, Instance::offsetOfTargetInstance(importIndex)), baseMemory); // Instance*.
commit-queue@webkit.org381ffdf2019-09-18 21:13:26 +000065 // While we're accessing that cacheline, also get the wasm entrypoint so we can tail call to it below.
mark.lam@apple.com52fa2cc2018-03-30 05:04:44 +000066 jit.loadPtr(JIT::Address(sizeRegAsScratch, Instance::offsetOfWasmEntrypointLoadLocation(importIndex)), scratch);
jfbastien@apple.comd9f999e2017-10-20 02:23:29 +000067 jit.storeWasmContextInstance(baseMemory);
sbarati@apple.com83ea2fc2017-05-18 19:38:10 +000068
jfbastien@apple.com89177d32017-10-26 15:33:55 +000069 jit.loadPtr(JIT::Address(sizeRegAsScratch, Instance::offsetOfCachedStackLimit()), sizeRegAsScratch);
70 jit.storePtr(sizeRegAsScratch, JIT::Address(baseMemory, Instance::offsetOfCachedStackLimit()));
sbarati@apple.com83ea2fc2017-05-18 19:38:10 +000071
commit-queue@webkit.org0f5197e2022-06-10 09:37:34 +000072#if !CPU(ARM) // ARM has no pinned registers for Wasm Memory, so no need to set them up
jfbastien@apple.com89177d32017-10-26 15:33:55 +000073 // FIXME the following code assumes that all Wasm::Instance have the same pinned registers. https://bugs.webkit.org/show_bug.cgi?id=162952
sbarati@apple.com83ea2fc2017-05-18 19:38:10 +000074 // Set up the callee's baseMemory register as well as the memory size registers.
commit-queue@webkit.orgf79caa42019-06-12 18:40:56 +000075 {
ysuzuki@apple.comb4cae912020-11-18 20:22:16 +000076 jit.loadPtr(JIT::Address(baseMemory, Wasm::Instance::offsetOfCachedBoundsCheckingSize()), pinnedRegs.boundsCheckingSizeRegister); // Bound checking size.
commit-queue@webkit.orgf79caa42019-06-12 18:40:56 +000077 jit.loadPtr(JIT::Address(baseMemory, Wasm::Instance::offsetOfCachedMemory()), baseMemory); // Wasm::Memory::TaggedArrayStoragePtr<void> (void*).
sbarati@apple.com161d52c2021-03-26 05:25:40 +000078 jit.cageConditionallyAndUntag(Gigacage::Primitive, baseMemory, pinnedRegs.boundsCheckingSizeRegister, wasmCallingConvention().prologueScratchGPRs[1]);
commit-queue@webkit.orgf79caa42019-06-12 18:40:56 +000079 }
commit-queue@webkit.org0f5197e2022-06-10 09:37:34 +000080#endif
jfbastien@apple.com373aa052017-01-03 01:57:40 +000081
82 // Tail call into the callee WebAssembly function.
sbarati@apple.com6118bf42021-10-12 06:32:55 +000083 jit.loadPtr(JIT::Address(scratch), scratch);
ysuzuki@apple.com49871512019-08-14 20:15:04 +000084 jit.farJump(scratch, WasmEntryPtrTag);
jfbastien@apple.com373aa052017-01-03 01:57:40 +000085
mark.lam@apple.com55ae8492021-05-23 06:07:01 +000086 LinkBuffer patchBuffer(jit, GLOBAL_THUNK_ID, LinkBuffer::Profile::WasmThunk, JITCompilationCanFail);
jfbastien@apple.com817d14322017-06-28 06:42:13 +000087 if (UNLIKELY(patchBuffer.didFailToAllocate()))
88 return makeUnexpected(BindingFailure::OutOfMemory);
89
keith_miller@apple.com8e7bd482019-10-01 16:38:26 +000090 return FINALIZE_WASM_CODE(patchBuffer, WasmEntryPtrTag, "WebAssembly->WebAssembly import[%i]", importIndex);
jfbastien@apple.com373aa052017-01-03 01:57:40 +000091}
92
jfbastien@apple.com148b8d92016-12-08 21:09:06 +000093} } // namespace JSC::Wasm
94
95#endif // ENABLE(WEBASSEMBLY)