summaryrefslogtreecommitdiff
path: root/Source/JavaScriptCore/jit/GPRInfo.h
diff options
context:
space:
mode:
authorLorry Tar Creator <lorry-tar-importer@lorry>2017-06-27 06:07:23 +0000
committerLorry Tar Creator <lorry-tar-importer@lorry>2017-06-27 06:07:23 +0000
commit1bf1084f2b10c3b47fd1a588d85d21ed0eb41d0c (patch)
tree46dcd36c86e7fbc6e5df36deb463b33e9967a6f7 /Source/JavaScriptCore/jit/GPRInfo.h
parent32761a6cee1d0dee366b885b7b9c777e67885688 (diff)
downloadWebKitGtk-tarball-master.tar.gz
Diffstat (limited to 'Source/JavaScriptCore/jit/GPRInfo.h')
-rw-r--r--Source/JavaScriptCore/jit/GPRInfo.h392
1 files changed, 228 insertions, 164 deletions
diff --git a/Source/JavaScriptCore/jit/GPRInfo.h b/Source/JavaScriptCore/jit/GPRInfo.h
index 393a56b50..f7e4a6b2c 100644
--- a/Source/JavaScriptCore/jit/GPRInfo.h
+++ b/Source/JavaScriptCore/jit/GPRInfo.h
@@ -1,5 +1,5 @@
/*
- * Copyright (C) 2011, 2013 Apple Inc. All rights reserved.
+ * Copyright (C) 2011, 2013-2016 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
@@ -23,14 +23,21 @@
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
-#ifndef GPRInfo_h
-#define GPRInfo_h
+#pragma once
#include "MacroAssembler.h"
+#include <array>
#include <wtf/PrintStream.h>
namespace JSC {
+enum NoResultTag { NoResult };
+
+// We use the same conventions in the basline JIT as in the LLint. If you
+// change mappings in the GPRInfo, you should change them in the offlineasm
+// compiler adequately. The register naming conventions are described at the
+// top of the LowLevelInterpreter.asm file.
+
typedef MacroAssembler::RegisterID GPRReg;
#define InvalidGPRReg ((::JSC::GPRReg)-1)
@@ -54,12 +61,25 @@ public:
return JSValueRegs(gpr);
}
+ static JSValueRegs withTwoAvailableRegs(GPRReg gpr, GPRReg)
+ {
+ return JSValueRegs(gpr);
+ }
+
bool operator!() const { return m_gpr == InvalidGPRReg; }
+ explicit operator bool() const { return m_gpr != InvalidGPRReg; }
+
+ bool operator==(JSValueRegs other) { return m_gpr == other.m_gpr; }
+ bool operator!=(JSValueRegs other) { return !(*this == other); }
GPRReg gpr() const { return m_gpr; }
GPRReg tagGPR() const { return InvalidGPRReg; }
GPRReg payloadGPR() const { return m_gpr; }
+ bool uses(GPRReg gpr) const { return m_gpr == gpr; }
+
+ void dump(PrintStream&) const;
+
private:
GPRReg m_gpr;
};
@@ -98,6 +118,7 @@ public:
}
bool operator!() const { return m_base == InvalidGPRReg; }
+ explicit operator bool() const { return m_base != InvalidGPRReg; }
bool isAddress() const { return m_offset != notAddress(); }
@@ -119,6 +140,11 @@ public:
return m_base;
}
+ JSValueRegs regs() const
+ {
+ return JSValueRegs(gpr());
+ }
+
MacroAssembler::Address asAddress() const { return MacroAssembler::Address(base(), offset()); }
private:
@@ -144,16 +170,29 @@ public:
{
}
+ static JSValueRegs withTwoAvailableRegs(GPRReg gpr1, GPRReg gpr2)
+ {
+ return JSValueRegs(gpr1, gpr2);
+ }
+
static JSValueRegs payloadOnly(GPRReg gpr)
{
return JSValueRegs(InvalidGPRReg, gpr);
}
- bool operator!() const
+ bool operator!() const { return !static_cast<bool>(*this); }
+ explicit operator bool() const
{
- return static_cast<GPRReg>(m_tagGPR) == InvalidGPRReg
- && static_cast<GPRReg>(m_payloadGPR) == InvalidGPRReg;
+ return static_cast<GPRReg>(m_tagGPR) != InvalidGPRReg
+ || static_cast<GPRReg>(m_payloadGPR) != InvalidGPRReg;
}
+
+ bool operator==(JSValueRegs other) const
+ {
+ return m_tagGPR == other.m_tagGPR
+ && m_payloadGPR == other.m_payloadGPR;
+ }
+ bool operator!=(JSValueRegs other) const { return !(*this == other); }
GPRReg tagGPR() const { return static_cast<GPRReg>(m_tagGPR); }
GPRReg payloadGPR() const { return static_cast<GPRReg>(m_payloadGPR); }
@@ -169,6 +208,10 @@ public:
return tagGPR();
}
+ bool uses(GPRReg gpr) const { return m_tagGPR == gpr || m_payloadGPR == gpr; }
+
+ void dump(PrintStream&) const;
+
private:
int8_t m_tagGPR;
int8_t m_payloadGPR;
@@ -219,11 +262,12 @@ public:
result.m_tagType = static_cast<int8_t>(JSValue::CellTag);
return result;
}
-
- bool operator!() const
+
+ bool operator!() const { return !static_cast<bool>(*this); }
+ explicit operator bool() const
{
- return static_cast<GPRReg>(m_baseOrTag) == InvalidGPRReg
- && static_cast<GPRReg>(m_payload) == InvalidGPRReg;
+ return static_cast<GPRReg>(m_baseOrTag) != InvalidGPRReg
+ || static_cast<GPRReg>(m_payload) != InvalidGPRReg;
}
bool isAddress() const
@@ -268,6 +312,11 @@ public:
return static_cast<int32_t>(m_tagType);
}
+ JSValueRegs regs() const
+ {
+ return JSValueRegs(tagGPR(), payloadGPR());
+ }
+
MacroAssembler::Address asAddress(unsigned additionalOffset = 0) const { return MacroAssembler::Address(base(), offset() + additionalOffset); }
private:
@@ -280,10 +329,9 @@ private:
};
#endif // USE(JSVALUE32_64)
-// The baseline JIT requires that regT3 be callee-preserved.
-
#if CPU(X86)
#define NUMBER_OF_ARGUMENT_REGISTERS 0u
+#define NUMBER_OF_CALLEE_SAVES_REGISTERS 0u
class GPRInfo {
public:
@@ -291,25 +339,20 @@ public:
static const unsigned numberOfRegisters = 6;
static const unsigned numberOfArgumentRegisters = NUMBER_OF_ARGUMENT_REGISTERS;
- // Note: regT3 is required to be callee-preserved.
-
// Temporary registers.
static const GPRReg regT0 = X86Registers::eax;
static const GPRReg regT1 = X86Registers::edx;
static const GPRReg regT2 = X86Registers::ecx;
- static const GPRReg regT3 = X86Registers::ebx;
- static const GPRReg regT4 = X86Registers::edi;
- static const GPRReg regT5 = X86Registers::esi;
- // These registers match the baseline JIT.
- static const GPRReg cachedResultRegister = regT0;
- static const GPRReg cachedResultRegister2 = regT1;
+ static const GPRReg regT3 = X86Registers::ebx; // Callee-save
+ static const GPRReg regT4 = X86Registers::esi; // Callee-save
+ static const GPRReg regT5 = X86Registers::edi; // Callee-save
static const GPRReg callFrameRegister = X86Registers::ebp;
// These constants provide the names for the general purpose argument & return value registers.
static const GPRReg argumentGPR0 = X86Registers::ecx; // regT2
static const GPRReg argumentGPR1 = X86Registers::edx; // regT1
+ static const GPRReg argumentGPR2 = X86Registers::eax; // regT0
+ static const GPRReg argumentGPR3 = X86Registers::ebx; // regT3
static const GPRReg nonArgGPR0 = X86Registers::esi; // regT4
- static const GPRReg nonArgGPR1 = X86Registers::eax; // regT0
- static const GPRReg nonArgGPR2 = X86Registers::ebx; // regT3
static const GPRReg returnValueGPR = X86Registers::eax; // regT0
static const GPRReg returnValueGPR2 = X86Registers::edx; // regT1
static const GPRReg nonPreservedNonReturnGPR = X86Registers::ecx;
@@ -321,13 +364,18 @@ public:
return registerForIndex[index];
}
+ static GPRReg toArgumentRegister(unsigned)
+ {
+ UNREACHABLE_FOR_PLATFORM();
+ return InvalidGPRReg;
+ }
+
static unsigned toIndex(GPRReg reg)
{
ASSERT(reg != InvalidGPRReg);
ASSERT(static_cast<int>(reg) < 8);
- static const unsigned indexForRegister[8] = { 0, 2, 1, 3, InvalidIndex, InvalidIndex, 5, 4 };
+ static const unsigned indexForRegister[8] = { 0, 2, 1, 3, InvalidIndex, InvalidIndex, 4, 5 };
unsigned result = indexForRegister[reg];
- ASSERT(result != InvalidIndex);
return result;
}
@@ -350,8 +398,10 @@ public:
#if CPU(X86_64)
#if !OS(WINDOWS)
#define NUMBER_OF_ARGUMENT_REGISTERS 6u
+#define NUMBER_OF_CALLEE_SAVES_REGISTERS 5u
#else
#define NUMBER_OF_ARGUMENT_REGISTERS 4u
+#define NUMBER_OF_CALLEE_SAVES_REGISTERS 7u
#endif
class GPRInfo {
@@ -360,50 +410,78 @@ public:
static const unsigned numberOfRegisters = 11;
static const unsigned numberOfArgumentRegisters = NUMBER_OF_ARGUMENT_REGISTERS;
- // Note: regT3 is required to be callee-preserved.
-
// These registers match the baseline JIT.
- static const GPRReg cachedResultRegister = X86Registers::eax;
static const GPRReg callFrameRegister = X86Registers::ebp;
static const GPRReg tagTypeNumberRegister = X86Registers::r14;
static const GPRReg tagMaskRegister = X86Registers::r15;
+
// Temporary registers.
static const GPRReg regT0 = X86Registers::eax;
- static const GPRReg regT1 = X86Registers::edx;
- static const GPRReg regT2 = X86Registers::ecx;
- static const GPRReg regT3 = X86Registers::ebx;
- static const GPRReg regT4 = X86Registers::edi;
- static const GPRReg regT5 = X86Registers::esi;
- static const GPRReg regT6 = X86Registers::r8;
+#if !OS(WINDOWS)
+ static const GPRReg regT1 = X86Registers::esi;
+ static const GPRReg regT2 = X86Registers::edx;
+ static const GPRReg regT3 = X86Registers::ecx;
+ static const GPRReg regT4 = X86Registers::r8;
+ static const GPRReg regT5 = X86Registers::r10;
+ static const GPRReg regT6 = X86Registers::edi;
static const GPRReg regT7 = X86Registers::r9;
- static const GPRReg regT8 = X86Registers::r10;
- static const GPRReg regT9 = X86Registers::r12;
- static const GPRReg regT10 = X86Registers::r13;
+#else
+ static const GPRReg regT1 = X86Registers::edx;
+ static const GPRReg regT2 = X86Registers::r8;
+ static const GPRReg regT3 = X86Registers::r9;
+ static const GPRReg regT4 = X86Registers::r10;
+ static const GPRReg regT5 = X86Registers::ecx;
+#endif
+
+ static const GPRReg regCS0 = X86Registers::ebx;
+
+#if !OS(WINDOWS)
+ static const GPRReg regCS1 = X86Registers::r12;
+ static const GPRReg regCS2 = X86Registers::r13;
+ static const GPRReg regCS3 = X86Registers::r14;
+ static const GPRReg regCS4 = X86Registers::r15;
+#else
+ static const GPRReg regCS1 = X86Registers::esi;
+ static const GPRReg regCS2 = X86Registers::edi;
+ static const GPRReg regCS3 = X86Registers::r12;
+ static const GPRReg regCS4 = X86Registers::r13;
+ static const GPRReg regCS5 = X86Registers::r14;
+ static const GPRReg regCS6 = X86Registers::r15;
+#endif
+
// These constants provide the names for the general purpose argument & return value registers.
#if !OS(WINDOWS)
- static const GPRReg argumentGPR0 = X86Registers::edi; // regT4
- static const GPRReg argumentGPR1 = X86Registers::esi; // regT5
- static const GPRReg argumentGPR2 = X86Registers::edx; // regT1
- static const GPRReg argumentGPR3 = X86Registers::ecx; // regT2
- static const GPRReg argumentGPR4 = X86Registers::r8; // regT6
- static const GPRReg argumentGPR5 = X86Registers::r9; // regT7
+ static const GPRReg argumentGPR0 = X86Registers::edi; // regT6
+ static const GPRReg argumentGPR1 = X86Registers::esi; // regT1
+ static const GPRReg argumentGPR2 = X86Registers::edx; // regT2
+ static const GPRReg argumentGPR3 = X86Registers::ecx; // regT3
+ static const GPRReg argumentGPR4 = X86Registers::r8; // regT4
+ static const GPRReg argumentGPR5 = X86Registers::r9; // regT7
#else
- static const GPRReg argumentGPR0 = X86Registers::ecx;
- static const GPRReg argumentGPR1 = X86Registers::edx;
- static const GPRReg argumentGPR2 = X86Registers::r8; // regT6
- static const GPRReg argumentGPR3 = X86Registers::r9; // regT7
+ static const GPRReg argumentGPR0 = X86Registers::ecx; // regT5
+ static const GPRReg argumentGPR1 = X86Registers::edx; // regT1
+ static const GPRReg argumentGPR2 = X86Registers::r8; // regT2
+ static const GPRReg argumentGPR3 = X86Registers::r9; // regT3
#endif
- static const GPRReg nonArgGPR0 = X86Registers::r10; // regT8
- static const GPRReg nonArgGPR1 = X86Registers::ebx; // regT3
- static const GPRReg nonArgGPR2 = X86Registers::r12; // regT9
+ static const GPRReg nonArgGPR0 = X86Registers::r10; // regT5 (regT4 on Windows)
static const GPRReg returnValueGPR = X86Registers::eax; // regT0
- static const GPRReg returnValueGPR2 = X86Registers::edx; // regT1
- static const GPRReg nonPreservedNonReturnGPR = X86Registers::esi;
+ static const GPRReg returnValueGPR2 = X86Registers::edx; // regT1 or regT2
+ static const GPRReg nonPreservedNonReturnGPR = X86Registers::r10; // regT5 (regT4 on Windows)
+ static const GPRReg nonPreservedNonArgumentGPR = X86Registers::r10; // regT5 (regT4 on Windows)
+
+ // FIXME: I believe that all uses of this are dead in the sense that it just causes the scratch
+ // register allocator to select a different register and potentially spill things. It would be better
+ // if we instead had a more explicit way of saying that we don't have a scratch register.
+ static const GPRReg patchpointScratchRegister;
static GPRReg toRegister(unsigned index)
{
ASSERT(index < numberOfRegisters);
- static const GPRReg registerForIndex[numberOfRegisters] = { regT0, regT1, regT2, regT3, regT4, regT5, regT6, regT7, regT8, regT9, regT10 };
+#if !OS(WINDOWS)
+ static const GPRReg registerForIndex[numberOfRegisters] = { regT0, regT1, regT2, regT3, regT4, regT5, regT6, regT7, regCS0, regCS1, regCS2 };
+#else
+ static const GPRReg registerForIndex[numberOfRegisters] = { regT0, regT1, regT2, regT3, regT4, regT5, regCS0, regCS1, regCS2, regCS3, regCS4 };
+#endif
return registerForIndex[index];
}
@@ -422,7 +500,11 @@ public:
{
ASSERT(reg != InvalidGPRReg);
ASSERT(static_cast<int>(reg) < 16);
- static const unsigned indexForRegister[16] = { 0, 2, 1, 3, InvalidIndex, InvalidIndex, 5, 4, 6, 7, 8, InvalidIndex, 9, 10, InvalidIndex, InvalidIndex };
+#if !OS(WINDOWS)
+ static const unsigned indexForRegister[16] = { 0, 3, 2, 8, InvalidIndex, InvalidIndex, 1, 6, 4, 7, 5, InvalidIndex, 9, 10, InvalidIndex, InvalidIndex };
+#else
+ static const unsigned indexForRegister[16] = { 0, 5, 1, 6, InvalidIndex, InvalidIndex, 7, 8, 2, 3, 4, InvalidIndex, 9, 10, InvalidIndex, InvalidIndex };
+#endif
return indexForRegister[reg];
}
@@ -439,6 +521,16 @@ public:
return nameForRegister[reg];
}
+ static const std::array<GPRReg, 3>& reservedRegisters()
+ {
+ static const std::array<GPRReg, 3> reservedRegisters { {
+ MacroAssembler::s_scratchRegister,
+ tagTypeNumberRegister,
+ tagMaskRegister,
+ } };
+ return reservedRegisters;
+ }
+
static const unsigned InvalidIndex = 0xffffffff;
};
@@ -446,6 +538,7 @@ public:
#if CPU(ARM)
#define NUMBER_OF_ARGUMENT_REGISTERS 4u
+#define NUMBER_OF_CALLEE_SAVES_REGISTERS 0u
class GPRInfo {
public:
@@ -453,13 +546,11 @@ public:
static const unsigned numberOfRegisters = 9;
static const unsigned numberOfArgumentRegisters = NUMBER_OF_ARGUMENT_REGISTERS;
- // Note: regT3 is required to be callee-preserved.
-
// Temporary registers.
static const GPRReg regT0 = ARMRegisters::r0;
static const GPRReg regT1 = ARMRegisters::r1;
static const GPRReg regT2 = ARMRegisters::r2;
- static const GPRReg regT3 = ARMRegisters::r4;
+ static const GPRReg regT3 = ARMRegisters::r3;
static const GPRReg regT4 = ARMRegisters::r8;
static const GPRReg regT5 = ARMRegisters::r9;
static const GPRReg regT6 = ARMRegisters::r10;
@@ -468,22 +559,19 @@ public:
#else
static const GPRReg regT7 = ARMRegisters::r7;
#endif
- static const GPRReg regT8 = ARMRegisters::r3;
+ static const GPRReg regT8 = ARMRegisters::r4;
// These registers match the baseline JIT.
- static const GPRReg cachedResultRegister = regT0;
- static const GPRReg cachedResultRegister2 = regT1;
static const GPRReg callFrameRegister = ARMRegisters::fp;
// These constants provide the names for the general purpose argument & return value registers.
static const GPRReg argumentGPR0 = ARMRegisters::r0; // regT0
static const GPRReg argumentGPR1 = ARMRegisters::r1; // regT1
static const GPRReg argumentGPR2 = ARMRegisters::r2; // regT2
- static const GPRReg argumentGPR3 = ARMRegisters::r3; // regT8
- static const GPRReg nonArgGPR0 = ARMRegisters::r4; // regT3
+ static const GPRReg argumentGPR3 = ARMRegisters::r3; // regT3
+ static const GPRReg nonArgGPR0 = ARMRegisters::r4; // regT8
static const GPRReg nonArgGPR1 = ARMRegisters::r8; // regT4
- static const GPRReg nonArgGPR2 = ARMRegisters::r9; // regT5
static const GPRReg returnValueGPR = ARMRegisters::r0; // regT0
static const GPRReg returnValueGPR2 = ARMRegisters::r1; // regT1
- static const GPRReg nonPreservedNonReturnGPR = ARMRegisters::r5; // regT7
+ static const GPRReg nonPreservedNonReturnGPR = ARMRegisters::r5;
static GPRReg toRegister(unsigned index)
{
@@ -492,18 +580,24 @@ public:
return registerForIndex[index];
}
+ static GPRReg toArgumentRegister(unsigned index)
+ {
+ ASSERT(index < numberOfArgumentRegisters);
+ static const GPRReg registerForIndex[numberOfArgumentRegisters] = { argumentGPR0, argumentGPR1, argumentGPR2, argumentGPR3 };
+ return registerForIndex[index];
+ }
+
static unsigned toIndex(GPRReg reg)
{
ASSERT(reg != InvalidGPRReg);
ASSERT(static_cast<int>(reg) < 16);
static const unsigned indexForRegister[16] =
#if CPU(ARM_THUMB2)
- { 0, 1, 2, 8, 3, 9, InvalidIndex, InvalidIndex, 4, 5, 6, 7, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex };
+ { 0, 1, 2, 3, 8, InvalidIndex, InvalidIndex, InvalidIndex, 4, 5, 6, 7, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex };
#else
- { 0, 1, 2, 8, 3, 9, InvalidIndex, 7, 4, 5, 6, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex };
+ { 0, 1, 2, 3, 8, InvalidIndex, InvalidIndex, 7, 4, 5, 6, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex };
#endif
unsigned result = indexForRegister[reg];
- ASSERT(result != InvalidIndex);
return result;
}
@@ -527,26 +621,27 @@ public:
#if CPU(ARM64)
#define NUMBER_OF_ARGUMENT_REGISTERS 8u
+// Callee Saves includes x19..x28 and FP registers q8..q15
+#define NUMBER_OF_CALLEE_SAVES_REGISTERS 18u
class GPRInfo {
public:
typedef GPRReg RegisterType;
static const unsigned numberOfRegisters = 16;
-
- // Note: regT3 is required to be callee-preserved.
+ static const unsigned numberOfArgumentRegisters = 8;
// These registers match the baseline JIT.
- static const GPRReg cachedResultRegister = ARM64Registers::x0;
- static const GPRReg timeoutCheckRegister = ARM64Registers::x26;
static const GPRReg callFrameRegister = ARM64Registers::fp;
static const GPRReg tagTypeNumberRegister = ARM64Registers::x27;
static const GPRReg tagMaskRegister = ARM64Registers::x28;
+ static const GPRReg dataTempRegister = MacroAssembler::dataTempRegister;
+ static const GPRReg memoryTempRegister = MacroAssembler::memoryTempRegister;
// Temporary registers.
static const GPRReg regT0 = ARM64Registers::x0;
static const GPRReg regT1 = ARM64Registers::x1;
static const GPRReg regT2 = ARM64Registers::x2;
- static const GPRReg regT3 = ARM64Registers::x23;
- static const GPRReg regT4 = ARM64Registers::x24;
+ static const GPRReg regT3 = ARM64Registers::x3;
+ static const GPRReg regT4 = ARM64Registers::x4;
static const GPRReg regT5 = ARM64Registers::x5;
static const GPRReg regT6 = ARM64Registers::x6;
static const GPRReg regT7 = ARM64Registers::x7;
@@ -558,6 +653,16 @@ public:
static const GPRReg regT13 = ARM64Registers::x13;
static const GPRReg regT14 = ARM64Registers::x14;
static const GPRReg regT15 = ARM64Registers::x15;
+ static const GPRReg regCS0 = ARM64Registers::x19; // Used by FTL only
+ static const GPRReg regCS1 = ARM64Registers::x20; // Used by FTL only
+ static const GPRReg regCS2 = ARM64Registers::x21; // Used by FTL only
+ static const GPRReg regCS3 = ARM64Registers::x22; // Used by FTL only
+ static const GPRReg regCS4 = ARM64Registers::x23; // Used by FTL only
+ static const GPRReg regCS5 = ARM64Registers::x24; // Used by FTL only
+ static const GPRReg regCS6 = ARM64Registers::x25; // Used by FTL only
+ static const GPRReg regCS7 = ARM64Registers::x26;
+ static const GPRReg regCS8 = ARM64Registers::x27; // tagTypeNumber
+ static const GPRReg regCS9 = ARM64Registers::x28; // tagMask
// These constants provide the names for the general purpose argument & return value registers.
static const GPRReg argumentGPR0 = ARM64Registers::x0; // regT0
static const GPRReg argumentGPR1 = ARM64Registers::x1; // regT1
@@ -569,12 +674,13 @@ public:
static const GPRReg argumentGPR7 = ARM64Registers::x7; // regT7
static const GPRReg nonArgGPR0 = ARM64Registers::x8; // regT8
static const GPRReg nonArgGPR1 = ARM64Registers::x9; // regT9
- static const GPRReg nonArgGPR2 = ARM64Registers::x10; // regT10
static const GPRReg returnValueGPR = ARM64Registers::x0; // regT0
static const GPRReg returnValueGPR2 = ARM64Registers::x1; // regT1
static const GPRReg nonPreservedNonReturnGPR = ARM64Registers::x2;
+ static const GPRReg nonPreservedNonArgumentGPR = ARM64Registers::x8;
+ static const GPRReg patchpointScratchRegister;
- // GPRReg mapping is direct, the machine regsiter numbers can
+ // GPRReg mapping is direct, the machine register numbers can
// be used directly as indices into the GPR RegisterBank.
COMPILE_ASSERT(ARM64Registers::q0 == 0, q0_is_0);
COMPILE_ASSERT(ARM64Registers::q1 == 1, q1_is_1);
@@ -598,12 +704,20 @@ public:
}
static unsigned toIndex(GPRReg reg)
{
+ if (reg > regT15)
+ return InvalidIndex;
return (unsigned)reg;
}
+ static GPRReg toArgumentRegister(unsigned index)
+ {
+ ASSERT(index < numberOfArgumentRegisters);
+ return toRegister(index);
+ }
+
static const char* debugName(GPRReg reg)
{
- ASSERT(static_cast<unsigned>(reg) != InvalidGPRReg);
+ ASSERT(reg != InvalidGPRReg);
ASSERT(static_cast<unsigned>(reg) < 32);
static const char* nameForRegister[32] = {
"r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7",
@@ -614,6 +728,17 @@ public:
return nameForRegister[reg];
}
+ static const std::array<GPRReg, 4>& reservedRegisters()
+ {
+ static const std::array<GPRReg, 4> reservedRegisters { {
+ dataTempRegister,
+ memoryTempRegister,
+ tagTypeNumberRegister,
+ tagMaskRegister,
+ } };
+ return reservedRegisters;
+ }
+
static const unsigned InvalidIndex = 0xffffffff;
};
@@ -621,6 +746,7 @@ public:
#if CPU(MIPS)
#define NUMBER_OF_ARGUMENT_REGISTERS 4u
+#define NUMBER_OF_CALLEE_SAVES_REGISTERS 0u
class GPRInfo {
public:
@@ -630,31 +756,26 @@ public:
// regT0 must be v0 for returning a 32-bit value.
// regT1 must be v1 for returning a pair of 32-bit value.
- // regT3 must be saved in the callee, so use an S register.
// Temporary registers.
static const GPRReg regT0 = MIPSRegisters::v0;
static const GPRReg regT1 = MIPSRegisters::v1;
- static const GPRReg regT2 = MIPSRegisters::t4;
- static const GPRReg regT3 = MIPSRegisters::s2;
- static const GPRReg regT4 = MIPSRegisters::t5;
- static const GPRReg regT5 = MIPSRegisters::t6;
- static const GPRReg regT6 = MIPSRegisters::s0;
+ static const GPRReg regT2 = MIPSRegisters::t2;
+ static const GPRReg regT3 = MIPSRegisters::t3;
+ static const GPRReg regT4 = MIPSRegisters::t4;
+ static const GPRReg regT5 = MIPSRegisters::t5;
+ static const GPRReg regT6 = MIPSRegisters::t6;
// These registers match the baseline JIT.
- static const GPRReg cachedResultRegister = regT0;
- static const GPRReg cachedResultRegister2 = regT1;
static const GPRReg callFrameRegister = MIPSRegisters::fp;
// These constants provide the names for the general purpose argument & return value registers.
static const GPRReg argumentGPR0 = MIPSRegisters::a0;
static const GPRReg argumentGPR1 = MIPSRegisters::a1;
static const GPRReg argumentGPR2 = MIPSRegisters::a2;
static const GPRReg argumentGPR3 = MIPSRegisters::a3;
- static const GPRReg nonArgGPR0 = regT2;
- static const GPRReg nonArgGPR1 = regT3;
- static const GPRReg nonArgGPR2 = regT4;
+ static const GPRReg nonArgGPR0 = regT4;
static const GPRReg returnValueGPR = regT0;
static const GPRReg returnValueGPR2 = regT1;
- static const GPRReg nonPreservedNonReturnGPR = regT5;
+ static const GPRReg nonPreservedNonReturnGPR = regT2;
static GPRReg toRegister(unsigned index)
{
@@ -663,17 +784,24 @@ public:
return registerForIndex[index];
}
+ static GPRReg toArgumentRegister(unsigned index)
+ {
+ ASSERT(index < numberOfArgumentRegisters);
+ static const GPRReg registerForIndex[numberOfArgumentRegisters] = { argumentGPR0, argumentGPR1, argumentGPR2, argumentGPR3 };
+ return registerForIndex[index];
+ }
+
static unsigned toIndex(GPRReg reg)
{
ASSERT(reg != InvalidGPRReg);
- ASSERT(reg < 24);
- static const unsigned indexForRegister[24] = {
+ ASSERT(reg < 32);
+ static const unsigned indexForRegister[32] = {
InvalidIndex, InvalidIndex, 0, 1, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex,
- InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex, 2, 4, 5, InvalidIndex,
- 6, InvalidIndex, 3, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex
+ InvalidIndex, InvalidIndex, 2, 3, 4, 5, 6, InvalidIndex,
+ InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex,
+ InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex, InvalidIndex
};
unsigned result = indexForRegister[reg];
- ASSERT(result != InvalidIndex);
return result;
}
@@ -695,76 +823,6 @@ public:
#endif // CPU(MIPS)
-#if CPU(SH4)
-#define NUMBER_OF_ARGUMENT_REGISTERS 4u
-
-class GPRInfo {
-public:
- typedef GPRReg RegisterType;
- static const unsigned numberOfRegisters = 10;
-
- // Note: regT3 is required to be callee-preserved.
-
- // Temporary registers.
- static const GPRReg regT0 = SH4Registers::r0;
- static const GPRReg regT1 = SH4Registers::r1;
- static const GPRReg regT2 = SH4Registers::r2;
- static const GPRReg regT3 = SH4Registers::r10;
- static const GPRReg regT4 = SH4Registers::r4;
- static const GPRReg regT5 = SH4Registers::r5;
- static const GPRReg regT6 = SH4Registers::r6;
- static const GPRReg regT7 = SH4Registers::r7;
- static const GPRReg regT8 = SH4Registers::r8;
- static const GPRReg regT9 = SH4Registers::r9;
- // These registers match the baseline JIT.
- static const GPRReg cachedResultRegister = regT0;
- static const GPRReg cachedResultRegister2 = regT1;
- static const GPRReg callFrameRegister = SH4Registers::fp;
- // These constants provide the names for the general purpose argument & return value registers.
- static const GPRReg argumentGPR0 = regT4;
- static const GPRReg argumentGPR1 = regT5;
- static const GPRReg argumentGPR2 = regT6;
- static const GPRReg argumentGPR3 = regT7;
- static const GPRReg nonArgGPR0 = regT3;
- static const GPRReg nonArgGPR1 = regT8;
- static const GPRReg nonArgGPR2 = regT9;
- static const GPRReg returnValueGPR = regT0;
- static const GPRReg returnValueGPR2 = regT1;
- static const GPRReg nonPreservedNonReturnGPR = regT2;
-
- static GPRReg toRegister(unsigned index)
- {
- ASSERT(index < numberOfRegisters);
- static const GPRReg registerForIndex[numberOfRegisters] = { regT0, regT1, regT2, regT3, regT4, regT5, regT6, regT7, regT8, regT9 };
- return registerForIndex[index];
- }
-
- static unsigned toIndex(GPRReg reg)
- {
- ASSERT(reg != InvalidGPRReg);
- ASSERT(reg < 14);
- static const unsigned indexForRegister[14] = { 0, 1, 2, InvalidIndex, 4, 5, 6, 7, 8, 9, 3, InvalidIndex, InvalidIndex, InvalidIndex };
- unsigned result = indexForRegister[reg];
- ASSERT(result != InvalidIndex);
- return result;
- }
-
- static const char* debugName(GPRReg reg)
- {
- ASSERT(reg != InvalidGPRReg);
- ASSERT(reg < 16);
- static const char* nameForRegister[16] = {
- "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7",
- "r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15"
- };
- return nameForRegister[reg];
- }
-
- static const unsigned InvalidIndex = 0xffffffff;
-};
-
-#endif // CPU(SH4)
-
// The baseline JIT uses "accumulator" style execution with regT0 (for 64-bit)
// and regT0 + regT1 (for 32-bit) serving as the accumulator register(s) for
// passing results of one opcode to the next. Hence:
@@ -773,6 +831,14 @@ COMPILE_ASSERT(GPRInfo::regT0 == GPRInfo::returnValueGPR, regT0_must_equal_retur
COMPILE_ASSERT(GPRInfo::regT1 == GPRInfo::returnValueGPR2, regT1_must_equal_returnValueGPR2);
#endif
+inline GPRReg extractResult(GPRReg result) { return result; }
+#if USE(JSVALUE64)
+inline GPRReg extractResult(JSValueRegs result) { return result.gpr(); }
+#else
+inline JSValueRegs extractResult(JSValueRegs result) { return result; }
+#endif
+inline NoResultTag extractResult(NoResultTag) { return NoResult; }
+
#endif // ENABLE(JIT)
} // namespace JSC
@@ -789,5 +855,3 @@ inline void printInternal(PrintStream& out, JSC::GPRReg reg)
}
} // namespace WTF
-
-#endif