]> git.saurik.com Git - apple/javascriptcore.git/blob - ftl/FTLLocation.cpp
JavaScriptCore-7601.1.46.3.tar.gz
[apple/javascriptcore.git] / ftl / FTLLocation.cpp
1 /*
2 * Copyright (C) 2013, 2014 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 "FTLLocation.h"
28
29 #if ENABLE(FTL_JIT)
30
31 #include "FTLSaveRestore.h"
32 #include "RegisterSet.h"
33 #include <wtf/CommaPrinter.h>
34 #include <wtf/DataLog.h>
35 #include <wtf/ListDump.h>
36
37 namespace JSC { namespace FTL {
38
39 Location Location::forStackmaps(const StackMaps* stackmaps, const StackMaps::Location& location)
40 {
41 switch (location.kind) {
42 case StackMaps::Location::Unprocessed:
43 RELEASE_ASSERT_NOT_REACHED();
44 break;
45
46 case StackMaps::Location::Register:
47 case StackMaps::Location::Direct:
48 return forRegister(location.dwarfReg, location.offset);
49
50 case StackMaps::Location::Indirect:
51 return forIndirect(location.dwarfReg, location.offset);
52
53 case StackMaps::Location::Constant:
54 return forConstant(location.offset);
55
56 case StackMaps::Location::ConstantIndex:
57 ASSERT(stackmaps);
58 return forConstant(stackmaps->constants[location.offset].integer);
59 }
60
61 RELEASE_ASSERT_NOT_REACHED();
62
63 return Location();
64 }
65
66 void Location::dump(PrintStream& out) const
67 {
68 out.print("(", kind());
69 if (hasDwarfReg())
70 out.print(", ", dwarfReg());
71 if (hasOffset())
72 out.print(", ", offset());
73 if (hasAddend())
74 out.print(", ", addend());
75 if (hasConstant())
76 out.print(", ", constant());
77 out.print(")");
78 }
79
80 bool Location::involvesGPR() const
81 {
82 return isGPR() || kind() == Indirect;
83 }
84
85 bool Location::isGPR() const
86 {
87 return kind() == Register && dwarfReg().reg().isGPR();
88 }
89
90 GPRReg Location::gpr() const
91 {
92 return dwarfReg().reg().gpr();
93 }
94
95 bool Location::isFPR() const
96 {
97 return kind() == Register && dwarfReg().reg().isFPR();
98 }
99
100 FPRReg Location::fpr() const
101 {
102 return dwarfReg().reg().fpr();
103 }
104
105 void Location::restoreInto(MacroAssembler& jit, char* savedRegisters, GPRReg result, unsigned numFramesToPop) const
106 {
107 if (involvesGPR() && RegisterSet::stackRegisters().get(gpr())) {
108 // Make the result GPR contain the appropriate stack register.
109 if (numFramesToPop) {
110 jit.move(MacroAssembler::framePointerRegister, result);
111
112 for (unsigned i = numFramesToPop - 1; i--;)
113 jit.loadPtr(result, result);
114
115 if (gpr() == MacroAssembler::framePointerRegister)
116 jit.loadPtr(result, result);
117 else
118 jit.addPtr(MacroAssembler::TrustedImmPtr(sizeof(void*) * 2), result);
119 } else
120 jit.move(gpr(), result);
121 }
122
123 if (isGPR()) {
124 if (RegisterSet::stackRegisters().get(gpr())) {
125 // Already restored into result.
126 } else
127 jit.load64(savedRegisters + offsetOfGPR(gpr()), result);
128
129 if (addend())
130 jit.add64(MacroAssembler::TrustedImm32(addend()), result);
131 return;
132 }
133
134 if (isFPR()) {
135 jit.load64(savedRegisters + offsetOfFPR(fpr()), result);
136 ASSERT(!addend());
137 return;
138 }
139
140 switch (kind()) {
141 case Register:
142 // LLVM used some register that we don't know about!
143 dataLog("Unrecognized location: ", *this, "\n");
144 RELEASE_ASSERT_NOT_REACHED();
145 return;
146
147 case Indirect:
148 if (RegisterSet::stackRegisters().get(gpr())) {
149 // The stack register is already recovered into result.
150 jit.load64(MacroAssembler::Address(result, offset()), result);
151 return;
152 }
153
154 jit.load64(savedRegisters + offsetOfGPR(gpr()), result);
155 jit.load64(MacroAssembler::Address(result, offset()), result);
156 return;
157
158 case Constant:
159 jit.move(MacroAssembler::TrustedImm64(constant()), result);
160 return;
161
162 case Unprocessed:
163 // Should never see this - it's an enumeration entry on LLVM's side that means that
164 // it hasn't processed this location.
165 RELEASE_ASSERT_NOT_REACHED();
166 return;
167 }
168
169 RELEASE_ASSERT_NOT_REACHED();
170 }
171
172 GPRReg Location::directGPR() const
173 {
174 RELEASE_ASSERT(!addend());
175 return gpr();
176 }
177
178 } } // namespace JSC::FTL
179
180 namespace WTF {
181
182 using namespace JSC::FTL;
183
184 void printInternal(PrintStream& out, JSC::FTL::Location::Kind kind)
185 {
186 switch (kind) {
187 case Location::Unprocessed:
188 out.print("Unprocessed");
189 return;
190 case Location::Register:
191 out.print("Register");
192 return;
193 case Location::Indirect:
194 out.print("Indirect");
195 return;
196 case Location::Constant:
197 out.print("Constant");
198 return;
199 }
200 RELEASE_ASSERT_NOT_REACHED();
201 }
202
203 } // namespace WTF
204
205 #endif // ENABLE(FTL_JIT)
206