blob: 5f8ae85ea2312958b79d2fc18d195dbed36d0ab6 [file] [log] [blame]
fpizlo@apple.comf74e4ae2015-11-18 23:05:34 +00001/*
fpizlo@apple.com45e45652016-01-07 21:20:37 +00002 * Copyright (C) 2015-2016 Apple Inc. All rights reserved.
fpizlo@apple.comf74e4ae2015-11-18 23:05:34 +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 "FTLCompile.h"
28
fpizlo@apple.comef5e13652016-02-18 00:11:21 +000029#if ENABLE(FTL_JIT)
fpizlo@apple.comf74e4ae2015-11-18 23:05:34 +000030
fpizlo@apple.comc6ca1042016-01-19 19:20:35 +000031#include "AirCode.h"
fpizlo@apple.comf74e4ae2015-11-18 23:05:34 +000032#include "B3Generate.h"
33#include "B3ProcedureInlines.h"
fpizlo@apple.com215f8f02016-01-26 22:05:28 +000034#include "B3StackSlot.h"
fpizlo@apple.comf74e4ae2015-11-18 23:05:34 +000035#include "CodeBlockWithJITType.h"
36#include "CCallHelpers.h"
37#include "DFGCommon.h"
38#include "DFGGraphSafepoint.h"
39#include "DFGOperations.h"
40#include "DataView.h"
41#include "Disassembler.h"
fpizlo@apple.comf74e4ae2015-11-18 23:05:34 +000042#include "FTLJITCode.h"
43#include "FTLThunks.h"
fpizlo@apple.comf74e4ae2015-11-18 23:05:34 +000044#include "JITSubGenerator.h"
45#include "LinkBuffer.h"
sbarati@apple.comd3d0c002016-01-30 01:11:05 +000046#include "PCToCodeOriginMap.h"
fpizlo@apple.comf74e4ae2015-11-18 23:05:34 +000047#include "ScratchRegisterAllocator.h"
48
49namespace JSC { namespace FTL {
50
51using namespace DFG;
52
53void compile(State& state, Safepoint::Result& safepointResult)
54{
55 Graph& graph = state.graph;
56 CodeBlock* codeBlock = graph.m_codeBlock;
57 VM& vm = graph.m_vm;
fpizlo@apple.comc6ca1042016-01-19 19:20:35 +000058
fpizlo@apple.comf74e4ae2015-11-18 23:05:34 +000059 {
60 GraphSafepoint safepoint(state.graph, safepointResult);
61
62 B3::prepareForGeneration(*state.proc);
63 }
64
65 if (safepointResult.didGetCancelled())
66 return;
67 RELEASE_ASSERT(!state.graph.m_vm.heap.isCollecting());
68
69 if (state.allocationFailed)
70 return;
71
72 std::unique_ptr<RegisterAtOffsetList> registerOffsets =
73 std::make_unique<RegisterAtOffsetList>(state.proc->calleeSaveRegisters());
74 if (shouldDumpDisassembly()) {
75 dataLog("Unwind info for ", CodeBlockWithJITType(state.graph.m_codeBlock, JITCode::FTLJIT), ":\n");
76 dataLog(" ", *registerOffsets, "\n");
77 }
aestes@apple.com13aae082016-01-02 08:03:08 +000078 state.graph.m_codeBlock->setCalleeSaveRegisters(WTFMove(registerOffsets));
fpizlo@apple.come362fbc2015-12-03 20:01:57 +000079 ASSERT(!(state.proc->frameSize() % sizeof(EncodedJSValue)));
80 state.jitCode->common.frameRegisterCount = state.proc->frameSize() / sizeof(EncodedJSValue);
fpizlo@apple.comf74e4ae2015-11-18 23:05:34 +000081
fpizlo@apple.com373ed6b2015-12-15 04:52:55 +000082 int localsOffset =
83 state.capturedValue->offsetFromFP() / sizeof(EncodedJSValue) + graph.m_nextMachineLocal;
fpizlo@apple.com215f8f02016-01-26 22:05:28 +000084 if (shouldDumpDisassembly()) {
85 dataLog(
86 "localsOffset = ", localsOffset, " for stack slot: ",
87 pointerDump(state.capturedValue), " at ", RawPointer(state.capturedValue), "\n");
88 }
89
fpizlo@apple.com373ed6b2015-12-15 04:52:55 +000090 for (unsigned i = graph.m_inlineVariableData.size(); i--;) {
91 InlineCallFrame* inlineCallFrame = graph.m_inlineVariableData[i].inlineCallFrame;
92
93 if (inlineCallFrame->argumentCountRegister.isValid())
94 inlineCallFrame->argumentCountRegister += localsOffset;
95
96 for (unsigned argument = inlineCallFrame->arguments.size(); argument-- > 1;) {
97 inlineCallFrame->arguments[argument] =
98 inlineCallFrame->arguments[argument].withLocalsOffset(localsOffset);
99 }
100
101 if (inlineCallFrame->isClosureCall) {
102 inlineCallFrame->calleeRecovery =
103 inlineCallFrame->calleeRecovery.withLocalsOffset(localsOffset);
104 }
commit-queue@webkit.org6c5b09b2016-03-23 22:33:17 +0000105
fpizlo@apple.com373ed6b2015-12-15 04:52:55 +0000106 }
sbarati@apple.come1437f12016-05-14 02:03:10 +0000107
fpizlo@apple.com48bf58d2016-07-18 19:32:34 +0000108 // Note that the scope register could be invalid here if the original code had CallEval but it
109 // got killed. That's because it takes the CallEval to cause the scope register to be kept alive
110 // unless the debugger is also enabled.
111 if (graph.needsScopeRegister() && codeBlock->scopeRegister().isValid())
sbarati@apple.come1437f12016-05-14 02:03:10 +0000112 codeBlock->setScopeRegister(codeBlock->scopeRegister() + localsOffset);
113
fpizlo@apple.com373ed6b2015-12-15 04:52:55 +0000114 for (OSRExitDescriptor& descriptor : state.jitCode->osrExitDescriptors) {
115 for (unsigned i = descriptor.m_values.size(); i--;)
116 descriptor.m_values[i] = descriptor.m_values[i].withLocalsOffset(localsOffset);
117 for (ExitTimeObjectMaterialization* materialization : descriptor.m_materializations)
118 materialization->accountForLocalsOffset(localsOffset);
119 }
120
fpizlo@apple.com21014282016-01-08 21:42:23 +0000121 // We will add exception handlers while generating.
122 codeBlock->clearExceptionHandlers();
123
fpizlo@apple.comf74e4ae2015-11-18 23:05:34 +0000124 CCallHelpers jit(&vm, codeBlock);
125 B3::generate(*state.proc, jit);
126
fpizlo@apple.com45e45652016-01-07 21:20:37 +0000127 // Emit the exception handler.
128 *state.exceptionHandler = jit.label();
mark.lam@apple.com491c2f52016-05-13 20:16:29 +0000129 jit.copyCalleeSavesToVMEntryFrameCalleeSavesBuffer();
fpizlo@apple.com45e45652016-01-07 21:20:37 +0000130 jit.move(MacroAssembler::TrustedImmPtr(jit.vm()), GPRInfo::argumentGPR0);
131 jit.move(GPRInfo::callFrameRegister, GPRInfo::argumentGPR1);
132 CCallHelpers::Call call = jit.call();
133 jit.jumpToExceptionHandler();
134 jit.addLinkTask(
135 [=] (LinkBuffer& linkBuffer) {
136 linkBuffer.link(call, FunctionPtr(lookupExceptionHandler));
137 });
138
fpizlo@apple.comf74e4ae2015-11-18 23:05:34 +0000139 state.finalizer->b3CodeLinkBuffer = std::make_unique<LinkBuffer>(
140 vm, jit, codeBlock, JITCompilationCanFail);
141 if (state.finalizer->b3CodeLinkBuffer->didFailToAllocate()) {
142 state.allocationFailed = true;
143 return;
144 }
sbarati@apple.comd3d0c002016-01-30 01:11:05 +0000145
146 B3::PCToOriginMap originMap = state.proc->releasePCToOriginMap();
147 if (vm.shouldBuilderPCToCodeOriginMapping())
148 codeBlock->setPCToCodeOriginMap(std::make_unique<PCToCodeOriginMap>(PCToCodeOriginMapBuilder(vm, WTFMove(originMap)), *state.finalizer->b3CodeLinkBuffer));
fpizlo@apple.comf74e4ae2015-11-18 23:05:34 +0000149
150 state.generatedFunction = bitwise_cast<GeneratedFunction>(
151 state.finalizer->b3CodeLinkBuffer->entrypoint().executableAddress());
fpizlo@apple.come362fbc2015-12-03 20:01:57 +0000152 state.jitCode->initializeB3Byproducts(state.proc->releaseByproducts());
fpizlo@apple.comf74e4ae2015-11-18 23:05:34 +0000153}
154
155} } // namespace JSC::FTL
156
fpizlo@apple.comef5e13652016-02-18 00:11:21 +0000157#endif // ENABLE(FTL_JIT)
fpizlo@apple.comf74e4ae2015-11-18 23:05:34 +0000158