• 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 DFGJITCodeGenerator_h
27 #define DFGJITCodeGenerator_h
28 
29 #if ENABLE(DFG_JIT)
30 
31 #include "CodeBlock.h"
32 #include <dfg/DFGGenerationInfo.h>
33 #include <dfg/DFGGraph.h>
34 #include <dfg/DFGJITCompiler.h>
35 #include <dfg/DFGOperations.h>
36 #include <dfg/DFGRegisterBank.h>
37 
38 namespace JSC { namespace DFG {
39 
40 class SpeculateIntegerOperand;
41 class SpeculateStrictInt32Operand;
42 class SpeculateCellOperand;
43 
44 
45 // === JITCodeGenerator ===
46 //
47 // This class provides common infrastructure used by the speculative &
48 // non-speculative JITs. Provides common mechanisms for virtual and
49 // physical register management, calls out from JIT code to helper
50 // functions, etc.
51 class JITCodeGenerator {
52 protected:
53     typedef MacroAssembler::TrustedImm32 TrustedImm32;
54     typedef MacroAssembler::Imm32 Imm32;
55 
56     // These constants are used to set priorities for spill order for
57     // the register allocator.
58     enum SpillOrder {
59         SpillOrderNone,
60         SpillOrderConstant = 1, // no spill, and cheap fill
61         SpillOrderSpilled = 2,  // no spill
62         SpillOrderJS = 4,       // needs spill
63         SpillOrderCell = 4,     // needs spill
64         SpillOrderInteger = 5,  // needs spill and box
65         SpillOrderDouble = 6,   // needs spill and convert
66         SpillOrderMax
67     };
68 
69 
70 public:
71     GPRReg fillInteger(NodeIndex, DataFormat& returnFormat);
72     FPRReg fillDouble(NodeIndex);
73     GPRReg fillJSValue(NodeIndex);
74 
75     // lock and unlock GPR & FPR registers.
lock(GPRReg reg)76     void lock(GPRReg reg)
77     {
78         m_gprs.lock(reg);
79     }
lock(FPRReg reg)80     void lock(FPRReg reg)
81     {
82         m_fprs.lock(reg);
83     }
unlock(GPRReg reg)84     void unlock(GPRReg reg)
85     {
86         m_gprs.unlock(reg);
87     }
unlock(FPRReg reg)88     void unlock(FPRReg reg)
89     {
90         m_fprs.unlock(reg);
91     }
92 
93     // Used to check whether a child node is on its last use,
94     // and its machine registers may be reused.
canReuse(NodeIndex nodeIndex)95     bool canReuse(NodeIndex nodeIndex)
96     {
97         VirtualRegister virtualRegister = m_jit.graph()[nodeIndex].virtualRegister;
98         GenerationInfo& info = m_generationInfo[virtualRegister];
99         return info.canReuse();
100     }
reuse(GPRReg reg)101     GPRReg reuse(GPRReg reg)
102     {
103         m_gprs.lock(reg);
104         return reg;
105     }
reuse(FPRReg reg)106     FPRReg reuse(FPRReg reg)
107     {
108         m_fprs.lock(reg);
109         return reg;
110     }
111 
112     // Allocate a gpr/fpr.
allocate()113     GPRReg allocate()
114     {
115         VirtualRegister spillMe;
116         GPRReg gpr = m_gprs.allocate(spillMe);
117         if (spillMe != InvalidVirtualRegister)
118             spill(spillMe);
119         return gpr;
120     }
fprAllocate()121     FPRReg fprAllocate()
122     {
123         VirtualRegister spillMe;
124         FPRReg fpr = m_fprs.allocate(spillMe);
125         if (spillMe != InvalidVirtualRegister)
126             spill(spillMe);
127         return fpr;
128     }
129 
130     // Check whether a VirtualRegsiter is currently in a machine register.
131     // We use this when filling operands to fill those that are already in
132     // machine registers first (by locking VirtualRegsiters that are already
133     // in machine register before filling those that are not we attempt to
134     // avoid spilling values we will need immediately).
isFilled(NodeIndex nodeIndex)135     bool isFilled(NodeIndex nodeIndex)
136     {
137         VirtualRegister virtualRegister = m_jit.graph()[nodeIndex].virtualRegister;
138         GenerationInfo& info = m_generationInfo[virtualRegister];
139         return info.registerFormat() != DataFormatNone;
140     }
isFilledDouble(NodeIndex nodeIndex)141     bool isFilledDouble(NodeIndex nodeIndex)
142     {
143         VirtualRegister virtualRegister = m_jit.graph()[nodeIndex].virtualRegister;
144         GenerationInfo& info = m_generationInfo[virtualRegister];
145         return info.registerFormat() == DataFormatDouble;
146     }
147 
148 protected:
JITCodeGenerator(JITCompiler & jit,bool isSpeculative)149     JITCodeGenerator(JITCompiler& jit, bool isSpeculative)
150         : m_jit(jit)
151         , m_isSpeculative(isSpeculative)
152         , m_compileIndex(0)
153         , m_generationInfo(m_jit.codeBlock()->m_numCalleeRegisters)
154         , m_blockHeads(jit.graph().m_blocks.size())
155     {
156     }
157 
158     // These methods convert between doubles, and doubles boxed and JSValues.
boxDouble(FPRReg fpr,GPRReg gpr)159     GPRReg boxDouble(FPRReg fpr, GPRReg gpr)
160     {
161         JITCompiler::FPRegisterID fpReg = JITCompiler::fprToRegisterID(fpr);
162         JITCompiler::RegisterID reg = JITCompiler::gprToRegisterID(gpr);
163         m_jit.moveDoubleToPtr(fpReg, reg);
164         m_jit.subPtr(JITCompiler::tagTypeNumberRegister, reg);
165         return gpr;
166     }
unboxDouble(GPRReg gpr,FPRReg fpr)167     FPRReg unboxDouble(GPRReg gpr, FPRReg fpr)
168     {
169         JITCompiler::RegisterID reg = JITCompiler::gprToRegisterID(gpr);
170         JITCompiler::FPRegisterID fpReg = JITCompiler::fprToRegisterID(fpr);
171         m_jit.addPtr(JITCompiler::tagTypeNumberRegister, reg);
172         m_jit.movePtrToDouble(reg, fpReg);
173         return fpr;
174     }
boxDouble(FPRReg fpr)175     GPRReg boxDouble(FPRReg fpr)
176     {
177         return boxDouble(fpr, allocate());
178     }
unboxDouble(GPRReg gpr)179     FPRReg unboxDouble(GPRReg gpr)
180     {
181         return unboxDouble(gpr, fprAllocate());
182     }
183 
184     // Called on an operand once it has been consumed by a parent node.
use(NodeIndex nodeIndex)185     void use(NodeIndex nodeIndex)
186     {
187         VirtualRegister virtualRegister = m_jit.graph()[nodeIndex].virtualRegister;
188         GenerationInfo& info = m_generationInfo[virtualRegister];
189 
190         // use() returns true when the value becomes dead, and any
191         // associated resources may be freed.
192         if (!info.use())
193             return;
194 
195         // Release the associated machine registers.
196         DataFormat registerFormat = info.registerFormat();
197         if (registerFormat == DataFormatDouble)
198             m_fprs.release(info.fpr());
199         else if (registerFormat != DataFormatNone)
200             m_gprs.release(info.gpr());
201     }
202 
203     // Spill a VirtualRegister to the RegisterFile.
spill(VirtualRegister spillMe)204     void spill(VirtualRegister spillMe)
205     {
206         GenerationInfo& info = m_generationInfo[spillMe];
207 
208         // Check the GenerationInfo to see if this value need writing
209         // to the RegisterFile - if not, mark it as spilled & return.
210         if (!info.needsSpill()) {
211             info.setSpilled();
212             return;
213         }
214 
215         DataFormat spillFormat = info.registerFormat();
216         if (spillFormat == DataFormatDouble) {
217             // All values are spilled as JSValues, so box the double via a temporary gpr.
218             GPRReg gpr = boxDouble(info.fpr());
219             m_jit.storePtr(JITCompiler::gprToRegisterID(gpr), JITCompiler::addressFor(spillMe));
220             unlock(gpr);
221             info.spill(DataFormatJSDouble);
222             return;
223         }
224 
225         // The following code handles JSValues, int32s, and cells.
226         ASSERT(spillFormat == DataFormatInteger || spillFormat == DataFormatCell || spillFormat & DataFormatJS);
227 
228         JITCompiler::RegisterID reg = JITCompiler::gprToRegisterID(info.gpr());
229         // We need to box int32 and cell values ...
230         // but on JSVALUE64 boxing a cell is a no-op!
231         if (spillFormat == DataFormatInteger)
232             m_jit.orPtr(JITCompiler::tagTypeNumberRegister, reg);
233 
234         // Spill the value, and record it as spilled in its boxed form.
235         m_jit.storePtr(reg, JITCompiler::addressFor(spillMe));
236         info.spill((DataFormat)(spillFormat | DataFormatJS));
237     }
238 
239     // Checks/accessors for constant values.
isConstant(NodeIndex nodeIndex)240     bool isConstant(NodeIndex nodeIndex) { return m_jit.isConstant(nodeIndex); }
isInt32Constant(NodeIndex nodeIndex)241     bool isInt32Constant(NodeIndex nodeIndex) { return m_jit.isInt32Constant(nodeIndex); }
isDoubleConstant(NodeIndex nodeIndex)242     bool isDoubleConstant(NodeIndex nodeIndex) { return m_jit.isDoubleConstant(nodeIndex); }
isJSConstant(NodeIndex nodeIndex)243     bool isJSConstant(NodeIndex nodeIndex) { return m_jit.isJSConstant(nodeIndex); }
valueOfInt32Constant(NodeIndex nodeIndex)244     int32_t valueOfInt32Constant(NodeIndex nodeIndex) { return m_jit.valueOfInt32Constant(nodeIndex); }
valueOfDoubleConstant(NodeIndex nodeIndex)245     double valueOfDoubleConstant(NodeIndex nodeIndex) { return m_jit.valueOfDoubleConstant(nodeIndex); }
valueOfJSConstant(NodeIndex nodeIndex)246     JSValue valueOfJSConstant(NodeIndex nodeIndex) { return m_jit.valueOfJSConstant(nodeIndex); }
247 
identifier(unsigned index)248     Identifier* identifier(unsigned index)
249     {
250         return &m_jit.codeBlock()->identifier(index);
251     }
252 
253     // Spill all VirtualRegisters back to the RegisterFile.
flushRegisters()254     void flushRegisters()
255     {
256         for (GPRReg gpr = gpr0; gpr < numberOfGPRs; next(gpr)) {
257             VirtualRegister name = m_gprs.name(gpr);
258             if (name != InvalidVirtualRegister) {
259                 spill(name);
260                 m_gprs.release(gpr);
261             }
262         }
263         for (FPRReg fpr = fpr0; fpr < numberOfFPRs; next(fpr)) {
264             VirtualRegister name = m_fprs.name(fpr);
265             if (name != InvalidVirtualRegister) {
266                 spill(name);
267                 m_fprs.release(fpr);
268             }
269         }
270     }
271 
272 #ifndef NDEBUG
273     // Used to ASSERT flushRegisters() has been called prior to
274     // calling out from JIT code to a C helper function.
isFlushed()275     bool isFlushed()
276     {
277         for (GPRReg gpr = gpr0; gpr < numberOfGPRs; next(gpr)) {
278             VirtualRegister name = m_gprs.name(gpr);
279             if (name != InvalidVirtualRegister)
280                 return false;
281         }
282         for (FPRReg fpr = fpr0; fpr < numberOfFPRs; next(fpr)) {
283             VirtualRegister name = m_fprs.name(fpr);
284             if (name != InvalidVirtualRegister)
285                 return false;
286         }
287         return true;
288     }
289 #endif
290 
291     // Get the JSValue representation of a constant.
constantAsJSValue(NodeIndex nodeIndex)292     JSValue constantAsJSValue(NodeIndex nodeIndex)
293     {
294         Node& node = m_jit.graph()[nodeIndex];
295         if (isInt32Constant(nodeIndex))
296             return jsNumber(node.int32Constant());
297         if (isDoubleConstant(nodeIndex))
298             return JSValue(JSValue::EncodeAsDouble, node.numericConstant());
299         ASSERT(isJSConstant(nodeIndex));
300         return valueOfJSConstant(nodeIndex);
301     }
constantAsJSValueAsImmPtr(NodeIndex nodeIndex)302     MacroAssembler::ImmPtr constantAsJSValueAsImmPtr(NodeIndex nodeIndex)
303     {
304         return MacroAssembler::ImmPtr(JSValue::encode(constantAsJSValue(nodeIndex)));
305     }
306 
307     // Helper functions to enable code sharing in implementations of bit/shift ops.
bitOp(NodeType op,int32_t imm,MacroAssembler::RegisterID op1,MacroAssembler::RegisterID result)308     void bitOp(NodeType op, int32_t imm, MacroAssembler::RegisterID op1, MacroAssembler::RegisterID result)
309     {
310         switch (op) {
311         case BitAnd:
312             m_jit.and32(Imm32(imm), op1, result);
313             break;
314         case BitOr:
315             m_jit.or32(Imm32(imm), op1, result);
316             break;
317         case BitXor:
318             m_jit.xor32(Imm32(imm), op1, result);
319             break;
320         default:
321             ASSERT_NOT_REACHED();
322         }
323     }
bitOp(NodeType op,MacroAssembler::RegisterID op1,MacroAssembler::RegisterID op2,MacroAssembler::RegisterID result)324     void bitOp(NodeType op, MacroAssembler::RegisterID op1, MacroAssembler::RegisterID op2, MacroAssembler::RegisterID result)
325     {
326         switch (op) {
327         case BitAnd:
328             m_jit.and32(op1, op2, result);
329             break;
330         case BitOr:
331             m_jit.or32(op1, op2, result);
332             break;
333         case BitXor:
334             m_jit.xor32(op1, op2, result);
335             break;
336         default:
337             ASSERT_NOT_REACHED();
338         }
339     }
shiftOp(NodeType op,MacroAssembler::RegisterID op1,int32_t shiftAmount,MacroAssembler::RegisterID result)340     void shiftOp(NodeType op, MacroAssembler::RegisterID op1, int32_t shiftAmount, MacroAssembler::RegisterID result)
341     {
342         switch (op) {
343         case BitRShift:
344             m_jit.rshift32(op1, Imm32(shiftAmount), result);
345             break;
346         case BitLShift:
347             m_jit.lshift32(op1, Imm32(shiftAmount), result);
348             break;
349         case BitURShift:
350             m_jit.urshift32(op1, Imm32(shiftAmount), result);
351             break;
352         default:
353             ASSERT_NOT_REACHED();
354         }
355     }
shiftOp(NodeType op,MacroAssembler::RegisterID op1,MacroAssembler::RegisterID shiftAmount,MacroAssembler::RegisterID result)356     void shiftOp(NodeType op, MacroAssembler::RegisterID op1, MacroAssembler::RegisterID shiftAmount, MacroAssembler::RegisterID result)
357     {
358         switch (op) {
359         case BitRShift:
360             m_jit.rshift32(op1, shiftAmount, result);
361             break;
362         case BitLShift:
363             m_jit.lshift32(op1, shiftAmount, result);
364             break;
365         case BitURShift:
366             m_jit.urshift32(op1, shiftAmount, result);
367             break;
368         default:
369             ASSERT_NOT_REACHED();
370         }
371     }
372 
373     // Called once a node has completed code generation but prior to setting
374     // its result, to free up its children. (This must happen prior to setting
375     // the nodes result, since the node may have the same VirtualRegister as
376     // a child, and as such will use the same GeneratioInfo).
377     void useChildren(Node&);
378 
379     // These method called to initialize the the GenerationInfo
380     // to describe the result of an operation.
381     void integerResult(GPRReg reg, NodeIndex nodeIndex, DataFormat format = DataFormatInteger)
382     {
383         Node& node = m_jit.graph()[nodeIndex];
384         useChildren(node);
385 
386         VirtualRegister virtualRegister = node.virtualRegister;
387         GenerationInfo& info = m_generationInfo[virtualRegister];
388 
389         if (format == DataFormatInteger) {
390             m_jit.jitAssertIsInt32(reg);
391             m_gprs.retain(reg, virtualRegister, SpillOrderInteger);
392             info.initInteger(nodeIndex, node.refCount, reg);
393         } else {
394             ASSERT(format == DataFormatJSInteger);
395             m_jit.jitAssertIsJSInt32(reg);
396             m_gprs.retain(reg, virtualRegister, SpillOrderJS);
397             info.initJSValue(nodeIndex, node.refCount, reg, format);
398         }
399     }
noResult(NodeIndex nodeIndex)400     void noResult(NodeIndex nodeIndex)
401     {
402         Node& node = m_jit.graph()[nodeIndex];
403         useChildren(node);
404 
405         VirtualRegister virtualRegister = node.virtualRegister;
406         GenerationInfo& info = m_generationInfo[virtualRegister];
407         info.initNone(nodeIndex, node.refCount);
408     }
cellResult(GPRReg reg,NodeIndex nodeIndex)409     void cellResult(GPRReg reg, NodeIndex nodeIndex)
410     {
411         Node& node = m_jit.graph()[nodeIndex];
412         useChildren(node);
413 
414         VirtualRegister virtualRegister = node.virtualRegister;
415         m_gprs.retain(reg, virtualRegister, SpillOrderCell);
416         GenerationInfo& info = m_generationInfo[virtualRegister];
417         info.initCell(nodeIndex, node.refCount, reg);
418     }
419     void jsValueResult(GPRReg reg, NodeIndex nodeIndex, DataFormat format = DataFormatJS)
420     {
421         if (format == DataFormatJSInteger)
422             m_jit.jitAssertIsJSInt32(reg);
423 
424         Node& node = m_jit.graph()[nodeIndex];
425         useChildren(node);
426 
427         VirtualRegister virtualRegister = node.virtualRegister;
428         m_gprs.retain(reg, virtualRegister, SpillOrderJS);
429         GenerationInfo& info = m_generationInfo[virtualRegister];
430         info.initJSValue(nodeIndex, node.refCount, reg, format);
431     }
doubleResult(FPRReg reg,NodeIndex nodeIndex)432     void doubleResult(FPRReg reg, NodeIndex nodeIndex)
433     {
434         Node& node = m_jit.graph()[nodeIndex];
435         useChildren(node);
436 
437         VirtualRegister virtualRegister = node.virtualRegister;
438         m_fprs.retain(reg, virtualRegister, SpillOrderDouble);
439         GenerationInfo& info = m_generationInfo[virtualRegister];
440         info.initDouble(nodeIndex, node.refCount, reg);
441     }
initConstantInfo(NodeIndex nodeIndex)442     void initConstantInfo(NodeIndex nodeIndex)
443     {
444         ASSERT(isInt32Constant(nodeIndex) || isDoubleConstant(nodeIndex) || isJSConstant(nodeIndex));
445         Node& node = m_jit.graph()[nodeIndex];
446         m_generationInfo[node.virtualRegister].initConstant(nodeIndex, node.refCount);
447     }
448 
449     // These methods used to sort arguments into the correct registers.
450     template<GPRReg destA, GPRReg destB>
setupTwoStubArgs(GPRReg srcA,GPRReg srcB)451     void setupTwoStubArgs(GPRReg srcA, GPRReg srcB)
452     {
453         // Assuming that srcA != srcB, there are 7 interesting states the registers may be in:
454         // (1) both are already in arg regs, the right way around.
455         // (2) both are already in arg regs, the wrong way around.
456         // (3) neither are currently in arg registers.
457         // (4) srcA in in its correct reg.
458         // (5) srcA in in the incorrect reg.
459         // (6) srcB in in its correct reg.
460         // (7) srcB in in the incorrect reg.
461         //
462         // The trivial approach is to simply emit two moves, to put srcA in place then srcB in
463         // place (the MacroAssembler will omit redundant moves). This apporach will be safe in
464         // cases 1, 3, 4, 5, 6, and in cases where srcA==srcB. The two problem cases are 2
465         // (requires a swap) and 7 (must move srcB first, to avoid trampling.)
466 
467         if (srcB != destA) {
468             // Handle the easy cases - two simple moves.
469             m_jit.move(JITCompiler::gprToRegisterID(srcA), JITCompiler::gprToRegisterID(destA));
470             m_jit.move(JITCompiler::gprToRegisterID(srcB), JITCompiler::gprToRegisterID(destB));
471         } else if (srcA != destB) {
472             // Handle the non-swap case - just put srcB in place first.
473             m_jit.move(JITCompiler::gprToRegisterID(srcB), JITCompiler::gprToRegisterID(destB));
474             m_jit.move(JITCompiler::gprToRegisterID(srcA), JITCompiler::gprToRegisterID(destA));
475         } else
476             m_jit.swap(JITCompiler::gprToRegisterID(destB), JITCompiler::gprToRegisterID(destB));
477     }
478     template<FPRReg destA, FPRReg destB>
setupTwoStubArgs(FPRReg srcA,FPRReg srcB)479     void setupTwoStubArgs(FPRReg srcA, FPRReg srcB)
480     {
481         // Assuming that srcA != srcB, there are 7 interesting states the registers may be in:
482         // (1) both are already in arg regs, the right way around.
483         // (2) both are already in arg regs, the wrong way around.
484         // (3) neither are currently in arg registers.
485         // (4) srcA in in its correct reg.
486         // (5) srcA in in the incorrect reg.
487         // (6) srcB in in its correct reg.
488         // (7) srcB in in the incorrect reg.
489         //
490         // The trivial approach is to simply emit two moves, to put srcA in place then srcB in
491         // place (the MacroAssembler will omit redundant moves). This apporach will be safe in
492         // cases 1, 3, 4, 5, 6, and in cases where srcA==srcB. The two problem cases are 2
493         // (requires a swap) and 7 (must move srcB first, to avoid trampling.)
494 
495         if (srcB != destA) {
496             // Handle the easy cases - two simple moves.
497             m_jit.moveDouble(JITCompiler::fprToRegisterID(srcA), JITCompiler::fprToRegisterID(destA));
498             m_jit.moveDouble(JITCompiler::fprToRegisterID(srcB), JITCompiler::fprToRegisterID(destB));
499             return;
500         }
501 
502         if (srcA != destB) {
503             // Handle the non-swap case - just put srcB in place first.
504             m_jit.moveDouble(JITCompiler::fprToRegisterID(srcB), JITCompiler::fprToRegisterID(destB));
505             m_jit.moveDouble(JITCompiler::fprToRegisterID(srcA), JITCompiler::fprToRegisterID(destA));
506             return;
507         }
508 
509         ASSERT(srcB == destA && srcA == destB);
510         // Need to swap; pick a temporary register.
511         FPRReg temp;
512         if (destA != JITCompiler::argumentFPR3 && destA != JITCompiler::argumentFPR3)
513             temp = JITCompiler::argumentFPR3;
514         else if (destA != JITCompiler::argumentFPR2 && destA != JITCompiler::argumentFPR2)
515             temp = JITCompiler::argumentFPR2;
516         else {
517             ASSERT(destA != JITCompiler::argumentFPR1 && destA != JITCompiler::argumentFPR1);
518             temp = JITCompiler::argumentFPR1;
519         }
520         m_jit.moveDouble(JITCompiler::fprToRegisterID(destA), JITCompiler::fprToRegisterID(temp));
521         m_jit.moveDouble(JITCompiler::fprToRegisterID(destB), JITCompiler::fprToRegisterID(destA));
522         m_jit.moveDouble(JITCompiler::fprToRegisterID(temp), JITCompiler::fprToRegisterID(destB));
523     }
setupStubArguments(GPRReg arg1,GPRReg arg2)524     void setupStubArguments(GPRReg arg1, GPRReg arg2)
525     {
526         setupTwoStubArgs<JITCompiler::argumentGPR1, JITCompiler::argumentGPR2>(arg1, arg2);
527     }
setupStubArguments(GPRReg arg1,GPRReg arg2,GPRReg arg3)528     void setupStubArguments(GPRReg arg1, GPRReg arg2, GPRReg arg3)
529     {
530         // If neither of arg2/arg3 are in our way, then we can move arg1 into place.
531         // Then we can use setupTwoStubArgs to fix arg2/arg3.
532         if (arg2 != JITCompiler::argumentGPR1 && arg3 != JITCompiler::argumentGPR1) {
533             m_jit.move(JITCompiler::gprToRegisterID(arg1), JITCompiler::argumentRegister1);
534             setupTwoStubArgs<JITCompiler::argumentGPR2, JITCompiler::argumentGPR3>(arg2, arg3);
535             return;
536         }
537 
538         // If neither of arg1/arg3 are in our way, then we can move arg2 into place.
539         // Then we can use setupTwoStubArgs to fix arg1/arg3.
540         if (arg1 != JITCompiler::argumentGPR2 && arg3 != JITCompiler::argumentGPR2) {
541             m_jit.move(JITCompiler::gprToRegisterID(arg2), JITCompiler::argumentRegister2);
542             setupTwoStubArgs<JITCompiler::argumentGPR1, JITCompiler::argumentGPR3>(arg1, arg3);
543             return;
544         }
545 
546         // If neither of arg1/arg2 are in our way, then we can move arg3 into place.
547         // Then we can use setupTwoStubArgs to fix arg1/arg2.
548         if (arg1 != JITCompiler::argumentGPR3 && arg2 != JITCompiler::argumentGPR3) {
549             m_jit.move(JITCompiler::gprToRegisterID(arg3), JITCompiler::argumentRegister3);
550             setupTwoStubArgs<JITCompiler::argumentGPR1, JITCompiler::argumentGPR2>(arg1, arg2);
551             return;
552         }
553 
554         // If we get here, we haven't been able to move any of arg1/arg2/arg3.
555         // Since all three are blocked, then all three must already be in the argument register.
556         // But are they in the right ones?
557 
558         // First, ensure arg1 is in place.
559         if (arg1 != JITCompiler::argumentGPR1) {
560             m_jit.swap(JITCompiler::gprToRegisterID(arg1), JITCompiler::argumentRegister1);
561 
562             // If arg1 wasn't in argumentGPR1, one of arg2/arg3 must be.
563             ASSERT(arg2 == JITCompiler::argumentGPR1 || arg3 == JITCompiler::argumentGPR1);
564             // If arg2 was in argumentGPR1 it no longer is (due to the swap).
565             // Otherwise arg3 must have been. Mark him as moved.
566             if (arg2 == JITCompiler::argumentGPR1)
567                 arg2 = arg1;
568             else
569                 arg3 = arg1;
570         }
571 
572         // Either arg2 & arg3 need swapping, or we're all done.
573         ASSERT((arg2 == JITCompiler::argumentGPR2 || arg3 == JITCompiler::argumentGPR3)
574             || (arg2 == JITCompiler::argumentGPR3 || arg3 == JITCompiler::argumentGPR2));
575 
576         if (arg2 != JITCompiler::argumentGPR2)
577             m_jit.swap(JITCompiler::argumentRegister2, JITCompiler::argumentRegister3);
578     }
579 
580     // These methods add calls to C++ helper functions.
callOperation(J_DFGOperation_EJP operation,GPRReg result,GPRReg arg1,void * pointer)581     void callOperation(J_DFGOperation_EJP operation, GPRReg result, GPRReg arg1, void* pointer)
582     {
583         ASSERT(isFlushed());
584 
585         m_jit.move(JITCompiler::gprToRegisterID(arg1), JITCompiler::argumentRegister1);
586         m_jit.move(JITCompiler::TrustedImmPtr(pointer), JITCompiler::argumentRegister2);
587         m_jit.move(JITCompiler::callFrameRegister, JITCompiler::argumentRegister0);
588 
589         appendCallWithExceptionCheck(operation);
590         m_jit.move(JITCompiler::returnValueRegister, JITCompiler::gprToRegisterID(result));
591     }
callOperation(J_DFGOperation_EJI operation,GPRReg result,GPRReg arg1,Identifier * identifier)592     void callOperation(J_DFGOperation_EJI operation, GPRReg result, GPRReg arg1, Identifier* identifier)
593     {
594         callOperation((J_DFGOperation_EJP)operation, result, arg1, identifier);
595     }
callOperation(J_DFGOperation_EJ operation,GPRReg result,GPRReg arg1)596     void callOperation(J_DFGOperation_EJ operation, GPRReg result, GPRReg arg1)
597     {
598         ASSERT(isFlushed());
599 
600         m_jit.move(JITCompiler::gprToRegisterID(arg1), JITCompiler::argumentRegister1);
601         m_jit.move(JITCompiler::callFrameRegister, JITCompiler::argumentRegister0);
602 
603         appendCallWithExceptionCheck(operation);
604         m_jit.move(JITCompiler::returnValueRegister, JITCompiler::gprToRegisterID(result));
605     }
callOperation(Z_DFGOperation_EJ operation,GPRReg result,GPRReg arg1)606     void callOperation(Z_DFGOperation_EJ operation, GPRReg result, GPRReg arg1)
607     {
608         ASSERT(isFlushed());
609 
610         m_jit.move(JITCompiler::gprToRegisterID(arg1), JITCompiler::argumentRegister1);
611         m_jit.move(JITCompiler::callFrameRegister, JITCompiler::argumentRegister0);
612 
613         appendCallWithExceptionCheck(operation);
614         m_jit.move(JITCompiler::returnValueRegister, JITCompiler::gprToRegisterID(result));
615     }
callOperation(Z_DFGOperation_EJJ operation,GPRReg result,GPRReg arg1,GPRReg arg2)616     void callOperation(Z_DFGOperation_EJJ operation, GPRReg result, GPRReg arg1, GPRReg arg2)
617     {
618         ASSERT(isFlushed());
619 
620         setupStubArguments(arg1, arg2);
621         m_jit.move(JITCompiler::callFrameRegister, JITCompiler::argumentRegister0);
622 
623         appendCallWithExceptionCheck(operation);
624         m_jit.move(JITCompiler::returnValueRegister, JITCompiler::gprToRegisterID(result));
625     }
callOperation(J_DFGOperation_EJJ operation,GPRReg result,GPRReg arg1,GPRReg arg2)626     void callOperation(J_DFGOperation_EJJ operation, GPRReg result, GPRReg arg1, GPRReg arg2)
627     {
628         ASSERT(isFlushed());
629 
630         setupStubArguments(arg1, arg2);
631         m_jit.move(JITCompiler::callFrameRegister, JITCompiler::argumentRegister0);
632 
633         appendCallWithExceptionCheck(operation);
634         m_jit.move(JITCompiler::returnValueRegister, JITCompiler::gprToRegisterID(result));
635     }
callOperation(V_DFGOperation_EJJP operation,GPRReg arg1,GPRReg arg2,void * pointer)636     void callOperation(V_DFGOperation_EJJP operation, GPRReg arg1, GPRReg arg2, void* pointer)
637     {
638         ASSERT(isFlushed());
639 
640         setupStubArguments(arg1, arg2);
641         m_jit.move(JITCompiler::TrustedImmPtr(pointer), JITCompiler::argumentRegister3);
642         m_jit.move(JITCompiler::callFrameRegister, JITCompiler::argumentRegister0);
643 
644         appendCallWithExceptionCheck(operation);
645     }
callOperation(V_DFGOperation_EJJI operation,GPRReg arg1,GPRReg arg2,Identifier * identifier)646     void callOperation(V_DFGOperation_EJJI operation, GPRReg arg1, GPRReg arg2, Identifier* identifier)
647     {
648         callOperation((V_DFGOperation_EJJP)operation, arg1, arg2, identifier);
649     }
callOperation(V_DFGOperation_EJJJ operation,GPRReg arg1,GPRReg arg2,GPRReg arg3)650     void callOperation(V_DFGOperation_EJJJ operation, GPRReg arg1, GPRReg arg2, GPRReg arg3)
651     {
652         ASSERT(isFlushed());
653 
654         setupStubArguments(arg1, arg2, arg3);
655         m_jit.move(JITCompiler::callFrameRegister, JITCompiler::argumentRegister0);
656 
657         appendCallWithExceptionCheck(operation);
658     }
callOperation(D_DFGOperation_DD operation,FPRReg result,FPRReg arg1,FPRReg arg2)659     void callOperation(D_DFGOperation_DD operation, FPRReg result, FPRReg arg1, FPRReg arg2)
660     {
661         ASSERT(isFlushed());
662 
663         setupTwoStubArgs<JITCompiler::argumentFPR0, JITCompiler::argumentFPR1>(arg1, arg2);
664 
665         m_jit.appendCall(operation);
666         m_jit.moveDouble(JITCompiler::fpReturnValueRegister, JITCompiler::fprToRegisterID(result));
667     }
668 
appendCallWithExceptionCheck(const FunctionPtr & function)669     void appendCallWithExceptionCheck(const FunctionPtr& function)
670     {
671         m_jit.appendCallWithExceptionCheck(function, m_jit.graph()[m_compileIndex].exceptionInfo);
672     }
673 
addBranch(const MacroAssembler::Jump & jump,BlockIndex destination)674     void addBranch(const MacroAssembler::Jump& jump, BlockIndex destination)
675     {
676         m_branches.append(BranchRecord(jump, destination));
677     }
678 
linkBranches()679     void linkBranches()
680     {
681         for (size_t i = 0; i < m_branches.size(); ++i) {
682             BranchRecord& branch = m_branches[i];
683             branch.jump.linkTo(m_blockHeads[branch.destination], &m_jit);
684         }
685     }
686 
687 #ifndef NDEBUG
688     void dump(const char* label = 0);
689 #endif
690 
691 #if DFG_CONSISTENCY_CHECK
692     void checkConsistency();
693 #else
checkConsistency()694     void checkConsistency() {}
695 #endif
696 
697     // The JIT, while also provides MacroAssembler functionality.
698     JITCompiler& m_jit;
699     // This flag is used to distinguish speculative and non-speculative
700     // code generation. This is significant when filling spilled values
701     // from the RegisterFile. When spilling we attempt to store information
702     // as to the type of boxed value being stored (int32, double, cell), and
703     // when filling on the speculative path we will retrieve this type info
704     // where available. On the non-speculative path, however, we cannot rely
705     // on the spill format info, since the a value being loaded might have
706     // been spilled by either the speculative or non-speculative paths (where
707     // we entered the non-speculative path on an intervening bail-out), and
708     // the value may have been boxed differently on the two paths.
709     bool m_isSpeculative;
710     // The current node being generated.
711     BlockIndex m_block;
712     NodeIndex m_compileIndex;
713     // Virtual and physical register maps.
714     Vector<GenerationInfo, 32> m_generationInfo;
715     RegisterBank<GPRReg, numberOfGPRs, SpillOrder, SpillOrderNone, SpillOrderMax> m_gprs;
716     RegisterBank<FPRReg, numberOfFPRs, SpillOrder, SpillOrderNone, SpillOrderMax> m_fprs;
717 
718     Vector<MacroAssembler::Label> m_blockHeads;
719     struct BranchRecord {
BranchRecordBranchRecord720         BranchRecord(MacroAssembler::Jump jump, BlockIndex destination)
721             : jump(jump)
722             , destination(destination)
723         {
724         }
725 
726         MacroAssembler::Jump jump;
727         BlockIndex destination;
728     };
729     Vector<BranchRecord, 8> m_branches;
730 };
731 
732 // === Operand types ===
733 //
734 // IntegerOperand, DoubleOperand and JSValueOperand.
735 //
736 // These classes are used to lock the operands to a node into machine
737 // registers. These classes implement of pattern of locking a value
738 // into register at the point of construction only if it is already in
739 // registers, and otherwise loading it lazily at the point it is first
740 // used. We do so in order to attempt to avoid spilling one operand
741 // in order to make space available for another.
742 
743 class IntegerOperand {
744 public:
IntegerOperand(JITCodeGenerator * jit,NodeIndex index)745     explicit IntegerOperand(JITCodeGenerator* jit, NodeIndex index)
746         : m_jit(jit)
747         , m_index(index)
748         , m_gprOrInvalid(InvalidGPRReg)
749 #ifndef NDEBUG
750         , m_format(DataFormatNone)
751 #endif
752     {
753         ASSERT(m_jit);
754         if (jit->isFilled(index))
755             gpr();
756     }
757 
~IntegerOperand()758     ~IntegerOperand()
759     {
760         ASSERT(m_gprOrInvalid != InvalidGPRReg);
761         m_jit->unlock(m_gprOrInvalid);
762     }
763 
index()764     NodeIndex index() const
765     {
766         return m_index;
767     }
768 
gpr()769     GPRReg gpr()
770     {
771         if (m_gprOrInvalid == InvalidGPRReg)
772             m_gprOrInvalid = m_jit->fillInteger(index(), m_format);
773         return m_gprOrInvalid;
774     }
775 
format()776     DataFormat format()
777     {
778         gpr(); // m_format is set when m_gpr is locked.
779         ASSERT(m_format == DataFormatInteger || m_format == DataFormatJSInteger);
780         return m_format;
781     }
782 
registerID()783     MacroAssembler::RegisterID registerID()
784     {
785         return JITCompiler::gprToRegisterID(gpr());
786     }
787 
788 private:
789     JITCodeGenerator* m_jit;
790     NodeIndex m_index;
791     GPRReg m_gprOrInvalid;
792     DataFormat m_format;
793 };
794 
795 class DoubleOperand {
796 public:
DoubleOperand(JITCodeGenerator * jit,NodeIndex index)797     explicit DoubleOperand(JITCodeGenerator* jit, NodeIndex index)
798         : m_jit(jit)
799         , m_index(index)
800         , m_fprOrInvalid(InvalidFPRReg)
801     {
802         ASSERT(m_jit);
803         if (jit->isFilledDouble(index))
804             fpr();
805     }
806 
~DoubleOperand()807     ~DoubleOperand()
808     {
809         ASSERT(m_fprOrInvalid != InvalidFPRReg);
810         m_jit->unlock(m_fprOrInvalid);
811     }
812 
index()813     NodeIndex index() const
814     {
815         return m_index;
816     }
817 
fpr()818     FPRReg fpr()
819     {
820         if (m_fprOrInvalid == InvalidFPRReg)
821             m_fprOrInvalid = m_jit->fillDouble(index());
822         return m_fprOrInvalid;
823     }
824 
registerID()825     MacroAssembler::FPRegisterID registerID()
826     {
827         return JITCompiler::fprToRegisterID(fpr());
828     }
829 
830 private:
831     JITCodeGenerator* m_jit;
832     NodeIndex m_index;
833     FPRReg m_fprOrInvalid;
834 };
835 
836 class JSValueOperand {
837 public:
JSValueOperand(JITCodeGenerator * jit,NodeIndex index)838     explicit JSValueOperand(JITCodeGenerator* jit, NodeIndex index)
839         : m_jit(jit)
840         , m_index(index)
841         , m_gprOrInvalid(InvalidGPRReg)
842     {
843         ASSERT(m_jit);
844         if (jit->isFilled(index))
845             gpr();
846     }
847 
~JSValueOperand()848     ~JSValueOperand()
849     {
850         ASSERT(m_gprOrInvalid != InvalidGPRReg);
851         m_jit->unlock(m_gprOrInvalid);
852     }
853 
index()854     NodeIndex index() const
855     {
856         return m_index;
857     }
858 
gpr()859     GPRReg gpr()
860     {
861         if (m_gprOrInvalid == InvalidGPRReg)
862             m_gprOrInvalid = m_jit->fillJSValue(index());
863         return m_gprOrInvalid;
864     }
865 
registerID()866     MacroAssembler::RegisterID registerID()
867     {
868         return JITCompiler::gprToRegisterID(gpr());
869     }
870 
871 private:
872     JITCodeGenerator* m_jit;
873     NodeIndex m_index;
874     GPRReg m_gprOrInvalid;
875 };
876 
877 
878 // === Temporaries ===
879 //
880 // These classes are used to allocate temporary registers.
881 // A mechanism is provided to attempt to reuse the registers
882 // currently allocated to child nodes whose value is consumed
883 // by, and not live after, this operation.
884 
885 class GPRTemporary {
886 public:
887     GPRTemporary(JITCodeGenerator*);
888     GPRTemporary(JITCodeGenerator*, SpeculateIntegerOperand&);
889     GPRTemporary(JITCodeGenerator*, SpeculateIntegerOperand&, SpeculateIntegerOperand&);
890     GPRTemporary(JITCodeGenerator*, IntegerOperand&);
891     GPRTemporary(JITCodeGenerator*, IntegerOperand&, IntegerOperand&);
892     GPRTemporary(JITCodeGenerator*, SpeculateCellOperand&);
893     GPRTemporary(JITCodeGenerator*, JSValueOperand&);
894 
~GPRTemporary()895     ~GPRTemporary()
896     {
897         m_jit->unlock(gpr());
898     }
899 
gpr()900     GPRReg gpr() const
901     {
902         ASSERT(m_gpr != InvalidGPRReg);
903         return m_gpr;
904     }
905 
registerID()906     MacroAssembler::RegisterID registerID()
907     {
908         ASSERT(m_gpr != InvalidGPRReg);
909         return JITCompiler::gprToRegisterID(m_gpr);
910     }
911 
912 protected:
GPRTemporary(JITCodeGenerator * jit,GPRReg lockedGPR)913     GPRTemporary(JITCodeGenerator* jit, GPRReg lockedGPR)
914         : m_jit(jit)
915         , m_gpr(lockedGPR)
916     {
917     }
918 
919 private:
920     JITCodeGenerator* m_jit;
921     GPRReg m_gpr;
922 };
923 
924 class FPRTemporary {
925 public:
926     FPRTemporary(JITCodeGenerator*);
927     FPRTemporary(JITCodeGenerator*, DoubleOperand&);
928     FPRTemporary(JITCodeGenerator*, DoubleOperand&, DoubleOperand&);
929 
~FPRTemporary()930     ~FPRTemporary()
931     {
932         m_jit->unlock(fpr());
933     }
934 
fpr()935     FPRReg fpr() const
936     {
937         ASSERT(m_fpr != InvalidFPRReg);
938         return m_fpr;
939     }
940 
registerID()941     MacroAssembler::FPRegisterID registerID()
942     {
943         ASSERT(m_fpr != InvalidFPRReg);
944         return JITCompiler::fprToRegisterID(m_fpr);
945     }
946 
947 protected:
FPRTemporary(JITCodeGenerator * jit,FPRReg lockedFPR)948     FPRTemporary(JITCodeGenerator* jit, FPRReg lockedFPR)
949         : m_jit(jit)
950         , m_fpr(lockedFPR)
951     {
952     }
953 
954 private:
955     JITCodeGenerator* m_jit;
956     FPRReg m_fpr;
957 };
958 
959 
960 // === Results ===
961 //
962 // These classes lock the result of a call to a C++ helper function.
963 
964 class GPRResult : public GPRTemporary {
965 public:
GPRResult(JITCodeGenerator * jit)966     GPRResult(JITCodeGenerator* jit)
967         : GPRTemporary(jit, lockedResult(jit))
968     {
969     }
970 
971 private:
lockedResult(JITCodeGenerator * jit)972     static GPRReg lockedResult(JITCodeGenerator* jit)
973     {
974         jit->lock(JITCompiler::returnValueGPR);
975         return JITCompiler::returnValueGPR;
976     }
977 };
978 
979 class FPRResult : public FPRTemporary {
980 public:
FPRResult(JITCodeGenerator * jit)981     FPRResult(JITCodeGenerator* jit)
982         : FPRTemporary(jit, lockedResult(jit))
983     {
984     }
985 
986 private:
lockedResult(JITCodeGenerator * jit)987     static FPRReg lockedResult(JITCodeGenerator* jit)
988     {
989         jit->lock(JITCompiler::returnValueFPR);
990         return JITCompiler::returnValueFPR;
991     }
992 };
993 
994 } } // namespace JSC::DFG
995 
996 #endif
997 #endif
998 
999