• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2011 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 #ifndef DFGNonSpeculativeJIT_h
27 #define DFGNonSpeculativeJIT_h
28 
29 #if ENABLE(DFG_JIT)
30 
31 #include <dfg/DFGJITCodeGenerator.h>
32 
33 namespace JSC { namespace DFG {
34 
35 class SpeculationCheckIndexIterator;
36 
37 // === EntryLocation ===
38 //
39 // This structure describes an entry point into the non-speculative
40 // code path. This is used in linking bail-outs from the speculative path.
41 struct EntryLocation {
42     EntryLocation(MacroAssembler::Label, NonSpeculativeJIT*);
43 
44     // The node this entry point corresponds to, and the label
45     // marking the start of code for the given node.
46     MacroAssembler::Label m_entry;
47     NodeIndex m_nodeIndex;
48 
49     // For every entry point we record a map recording for every
50     // machine register which, if any, values it contains. For
51     // GPR registers we must also record the format of the value.
52     struct RegisterInfo {
53         NodeIndex nodeIndex;
54         DataFormat format;
55     };
56     RegisterInfo m_gprInfo[numberOfGPRs];
57     NodeIndex m_fprInfo[numberOfFPRs];
58 };
59 
60 // === NonSpeculativeJIT ===
61 //
62 // This class is used to generate code for the non-speculative path.
63 // Code generation will take advantage of static information available
64 // in the dataflow to perform safe optimizations - for example, avoiding
65 // boxing numeric values between arithmetic operations, but will not
66 // perform any unsafe optimizations that would render the code unable
67 // to produce the correct results for any possible input.
68 class NonSpeculativeJIT : public JITCodeGenerator {
69     friend struct EntryLocation;
70 public:
NonSpeculativeJIT(JITCompiler & jit)71     NonSpeculativeJIT(JITCompiler& jit)
72         : JITCodeGenerator(jit, false)
73     {
74     }
75 
76     void compile(SpeculationCheckIndexIterator&);
77 
78     typedef SegmentedVector<EntryLocation, 16> EntryLocationVector;
entryLocations()79     EntryLocationVector& entryLocations() { return m_entryLocations; }
80 
81 private:
82     void compile(SpeculationCheckIndexIterator&, Node&);
83     void compile(SpeculationCheckIndexIterator&, BasicBlock&);
84 
85     bool isKnownInteger(NodeIndex);
86     bool isKnownNumeric(NodeIndex);
87 
88     // These methods are used when generating 'unexpected'
89     // calls out from JIT code to C++ helper routines -
90     // they spill all live values to the appropriate
91     // slots in the RegisterFile without changing any state
92     // in the GenerationInfo.
93     void silentSpillGPR(VirtualRegister spillMe, GPRReg exclude = InvalidGPRReg)
94     {
95         GenerationInfo& info = m_generationInfo[spillMe];
96         ASSERT(info.registerFormat() != DataFormatNone && info.registerFormat() != DataFormatDouble);
97 
98         if (!info.needsSpill() || (info.gpr() == exclude))
99             return;
100 
101         DataFormat registerFormat = info.registerFormat();
102         JITCompiler::RegisterID reg = JITCompiler::gprToRegisterID(info.gpr());
103 
104         if (registerFormat == DataFormatInteger) {
105             m_jit.orPtr(JITCompiler::tagTypeNumberRegister, reg);
106             m_jit.storePtr(reg, JITCompiler::addressFor(spillMe));
107         } else {
108             ASSERT(registerFormat & DataFormatJS || registerFormat == DataFormatCell);
109             m_jit.storePtr(reg, JITCompiler::addressFor(spillMe));
110         }
111     }
112     void silentSpillFPR(VirtualRegister spillMe, GPRReg canTrample, FPRReg exclude = InvalidFPRReg)
113     {
114         GenerationInfo& info = m_generationInfo[spillMe];
115         ASSERT(info.registerFormat() == DataFormatDouble);
116 
117         if (!info.needsSpill() || (info.fpr() == exclude))
118             return;
119 
120         boxDouble(info.fpr(), canTrample);
121         m_jit.storePtr(JITCompiler::gprToRegisterID(canTrample), JITCompiler::addressFor(spillMe));
122     }
123 
124     void silentFillGPR(VirtualRegister spillMe, GPRReg exclude = InvalidGPRReg)
125     {
126         GenerationInfo& info = m_generationInfo[spillMe];
127         if (info.gpr() == exclude)
128             return;
129 
130         NodeIndex nodeIndex = info.nodeIndex();
131         Node& node = m_jit.graph()[nodeIndex];
132         ASSERT(info.registerFormat() != DataFormatNone && info.registerFormat() != DataFormatDouble);
133         DataFormat registerFormat = info.registerFormat();
134         JITCompiler::RegisterID reg = JITCompiler::gprToRegisterID(info.gpr());
135 
136         if (registerFormat == DataFormatInteger) {
137             if (node.isConstant()) {
138                 ASSERT(isInt32Constant(nodeIndex));
139                 m_jit.move(Imm32(valueOfInt32Constant(nodeIndex)), reg);
140             } else
141                 m_jit.load32(JITCompiler::addressFor(spillMe), reg);
142             return;
143         }
144 
145         if (node.isConstant())
146             m_jit.move(constantAsJSValueAsImmPtr(nodeIndex), reg);
147         else {
148             ASSERT(registerFormat & DataFormatJS || registerFormat == DataFormatCell);
149             m_jit.loadPtr(JITCompiler::addressFor(spillMe), reg);
150         }
151     }
152     void silentFillFPR(VirtualRegister spillMe, GPRReg canTrample, FPRReg exclude = InvalidFPRReg)
153     {
154         GenerationInfo& info = m_generationInfo[spillMe];
155         if (info.fpr() == exclude)
156             return;
157 
158         NodeIndex nodeIndex = info.nodeIndex();
159         Node& node = m_jit.graph()[nodeIndex];
160         ASSERT(info.registerFormat() == DataFormatDouble);
161 
162         if (node.isConstant()) {
163             JITCompiler::RegisterID reg = JITCompiler::gprToRegisterID(info.gpr());
164             m_jit.move(constantAsJSValueAsImmPtr(nodeIndex), reg);
165         } else {
166             m_jit.loadPtr(JITCompiler::addressFor(spillMe), JITCompiler::gprToRegisterID(canTrample));
167             unboxDouble(canTrample, info.fpr());
168         }
169     }
170 
171     void silentSpillAllRegisters(GPRReg exclude, GPRReg preserve = InvalidGPRReg)
172     {
173         GPRReg canTrample = (preserve == gpr0) ? gpr1 : gpr0;
174 
175         for (GPRReg gpr = gpr0; gpr < numberOfGPRs; next(gpr)) {
176             VirtualRegister name = m_gprs.name(gpr);
177             if (name != InvalidVirtualRegister)
178                 silentSpillGPR(name, exclude);
179         }
180         for (FPRReg fpr = fpr0; fpr < numberOfFPRs; next(fpr)) {
181             VirtualRegister name = m_fprs.name(fpr);
182             if (name != InvalidVirtualRegister)
183                 silentSpillFPR(name, canTrample);
184         }
185     }
186     void silentSpillAllRegisters(FPRReg exclude, GPRReg preserve = InvalidGPRReg)
187     {
188         GPRReg canTrample = (preserve == gpr0) ? gpr1 : gpr0;
189 
190         for (GPRReg gpr = gpr0; gpr < numberOfGPRs; next(gpr)) {
191             VirtualRegister name = m_gprs.name(gpr);
192             if (name != InvalidVirtualRegister)
193                 silentSpillGPR(name);
194         }
195         for (FPRReg fpr = fpr0; fpr < numberOfFPRs; next(fpr)) {
196             VirtualRegister name = m_fprs.name(fpr);
197             if (name != InvalidVirtualRegister)
198                 silentSpillFPR(name, canTrample, exclude);
199         }
200     }
silentFillAllRegisters(GPRReg exclude)201     void silentFillAllRegisters(GPRReg exclude)
202     {
203         GPRReg canTrample = (exclude == gpr0) ? gpr1 : gpr0;
204 
205         for (FPRReg fpr = fpr0; fpr < numberOfFPRs; next(fpr)) {
206             VirtualRegister name = m_fprs.name(fpr);
207             if (name != InvalidVirtualRegister)
208                 silentFillFPR(name, canTrample);
209         }
210         for (GPRReg gpr = gpr0; gpr < numberOfGPRs; next(gpr)) {
211             VirtualRegister name = m_gprs.name(gpr);
212             if (name != InvalidVirtualRegister)
213                 silentFillGPR(name, exclude);
214         }
215     }
silentFillAllRegisters(FPRReg exclude)216     void silentFillAllRegisters(FPRReg exclude)
217     {
218         GPRReg canTrample = gpr0;
219 
220         for (FPRReg fpr = fpr0; fpr < numberOfFPRs; next(fpr)) {
221             VirtualRegister name = m_fprs.name(fpr);
222             if (name != InvalidVirtualRegister) {
223 #ifndef NDEBUG
224                 ASSERT(fpr != exclude);
225 #else
226                 UNUSED_PARAM(exclude);
227 #endif
228                 silentFillFPR(name, canTrample, exclude);
229             }
230         }
231         for (GPRReg gpr = gpr0; gpr < numberOfGPRs; next(gpr)) {
232             VirtualRegister name = m_gprs.name(gpr);
233             if (name != InvalidVirtualRegister)
234                 silentFillGPR(name);
235         }
236     }
237 
238     // These methods are used to plant calls out to C++
239     // helper routines to convert between types.
240     void valueToNumber(JSValueOperand&, FPRReg result);
241     void valueToInt32(JSValueOperand&, GPRReg result);
242     void numberToInt32(FPRReg, GPRReg result);
243 
244     // Record an entry location into the non-speculative code path;
245     // for every bail-out on the speculative path we record information
246     // to be able to re-enter into the non-speculative one.
trackEntry(MacroAssembler::Label entry)247     void trackEntry(MacroAssembler::Label entry)
248     {
249         m_entryLocations.append(EntryLocation(entry, this));
250     }
251 
252     EntryLocationVector m_entryLocations;
253 };
254 
255 } } // namespace JSC::DFG
256 
257 #endif
258 #endif
259 
260