Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(72)

Side by Side Diff: src/a64/macro-assembler-a64.h

Issue 164793003: A64: Use a scope utility to allocate scratch registers. (Closed) Base URL: https://v8.googlecode.com/svn/branches/bleeding_edge
Patch Set: Created 6 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
1 // Copyright 2013 the V8 project authors. All rights reserved. 1 // Copyright 2013 the V8 project authors. All rights reserved.
2 // Redistribution and use in source and binary forms, with or without 2 // Redistribution and use in source and binary forms, with or without
3 // modification, are permitted provided that the following conditions are 3 // modification, are permitted provided that the following conditions are
4 // met: 4 // met:
5 // 5 //
6 // * Redistributions of source code must retain the above copyright 6 // * Redistributions of source code must retain the above copyright
7 // notice, this list of conditions and the following disclaimer. 7 // notice, this list of conditions and the following disclaimer.
8 // * Redistributions in binary form must reproduce the above 8 // * Redistributions in binary form must reproduce the above
9 // copyright notice, this list of conditions and the following 9 // copyright notice, this list of conditions and the following
10 // disclaimer in the documentation and/or other materials provided 10 // disclaimer in the documentation and/or other materials provided
(...skipping 502 matching lines...) Expand 10 before | Expand all | Expand 10 after
513 // must be aligned to 16 bytes on entry and the total size of the specified 513 // must be aligned to 16 bytes on entry and the total size of the specified
514 // registers must also be a multiple of 16 bytes. 514 // registers must also be a multiple of 16 bytes.
515 // 515 //
516 // Even if the current stack pointer is not the system stack pointer (csp), 516 // Even if the current stack pointer is not the system stack pointer (csp),
517 // Push (and derived methods) will still modify the system stack pointer in 517 // Push (and derived methods) will still modify the system stack pointer in
518 // order to comply with ABI rules about accessing memory below the system 518 // order to comply with ABI rules about accessing memory below the system
519 // stack pointer. 519 // stack pointer.
520 // 520 //
521 // Other than the registers passed into Pop, the stack pointer and (possibly) 521 // Other than the registers passed into Pop, the stack pointer and (possibly)
522 // the system stack pointer, these methods do not modify any other registers. 522 // the system stack pointer, these methods do not modify any other registers.
523 // Scratch registers such as Tmp0() and Tmp1() are preserved.
524 void Push(const CPURegister& src0, const CPURegister& src1 = NoReg, 523 void Push(const CPURegister& src0, const CPURegister& src1 = NoReg,
525 const CPURegister& src2 = NoReg, const CPURegister& src3 = NoReg); 524 const CPURegister& src2 = NoReg, const CPURegister& src3 = NoReg);
526 void Pop(const CPURegister& dst0, const CPURegister& dst1 = NoReg, 525 void Pop(const CPURegister& dst0, const CPURegister& dst1 = NoReg,
527 const CPURegister& dst2 = NoReg, const CPURegister& dst3 = NoReg); 526 const CPURegister& dst2 = NoReg, const CPURegister& dst3 = NoReg);
528 527
529 // Alternative forms of Push and Pop, taking a RegList or CPURegList that 528 // Alternative forms of Push and Pop, taking a RegList or CPURegList that
530 // specifies the registers that are to be pushed or popped. Higher-numbered 529 // specifies the registers that are to be pushed or popped. Higher-numbered
531 // registers are associated with higher memory addresses (as in the A32 push 530 // registers are associated with higher memory addresses (as in the A32 push
532 // and pop instructions). 531 // and pop instructions).
533 // 532 //
(...skipping 205 matching lines...) Expand 10 before | Expand all | Expand 10 after
739 // Floating-point registers are popped after general-purpose registers, and 738 // Floating-point registers are popped after general-purpose registers, and
740 // thus come from higher addresses. 739 // thus come from higher addresses.
741 // 740 //
742 // This method must not be called unless the current stack pointer (as set by 741 // This method must not be called unless the current stack pointer (as set by
743 // SetStackPointer) is the system stack pointer (csp), and is aligned to 742 // SetStackPointer) is the system stack pointer (csp), and is aligned to
744 // ActivationFrameAlignment(). 743 // ActivationFrameAlignment().
745 void PopCalleeSavedRegisters(); 744 void PopCalleeSavedRegisters();
746 745
747 // Set the current stack pointer, but don't generate any code. 746 // Set the current stack pointer, but don't generate any code.
748 inline void SetStackPointer(const Register& stack_pointer) { 747 inline void SetStackPointer(const Register& stack_pointer) {
749 ASSERT(!AreAliased(stack_pointer, Tmp0(), Tmp1())); 748 ASSERT(!TmpList()->IncludesAliasOf(stack_pointer));
750 sp_ = stack_pointer; 749 sp_ = stack_pointer;
751 } 750 }
752 751
753 // Return the current stack pointer, as set by SetStackPointer. 752 // Return the current stack pointer, as set by SetStackPointer.
754 inline const Register& StackPointer() const { 753 inline const Register& StackPointer() const {
755 return sp_; 754 return sp_;
756 } 755 }
757 756
758 // Align csp for a frame, as per ActivationFrameAlignment, and make it the 757 // Align csp for a frame, as per ActivationFrameAlignment, and make it the
759 // current stack pointer. 758 // current stack pointer.
(...skipping 173 matching lines...) Expand 10 before | Expand all | Expand 10 after
933 Label* on_failed_conversion = NULL) { 932 Label* on_failed_conversion = NULL) {
934 ASSERT(as_int.Is64Bits()); 933 ASSERT(as_int.Is64Bits());
935 TryConvertDoubleToInt(as_int, value, scratch_d, on_successful_conversion, 934 TryConvertDoubleToInt(as_int, value, scratch_d, on_successful_conversion,
936 on_failed_conversion); 935 on_failed_conversion);
937 } 936 }
938 937
939 // ---- Object Utilities ---- 938 // ---- Object Utilities ----
940 939
941 // Copy fields from 'src' to 'dst', where both are tagged objects. 940 // Copy fields from 'src' to 'dst', where both are tagged objects.
942 // The 'temps' list is a list of X registers which can be used for scratch 941 // The 'temps' list is a list of X registers which can be used for scratch
943 // values. The temps list must include at least one register, and it must not 942 // values. The temps list must include at least one register.
944 // contain Tmp0() or Tmp1().
945 // 943 //
946 // Currently, CopyFields cannot make use of more than three registers from 944 // Currently, CopyFields cannot make use of more than three registers from
947 // the 'temps' list. 945 // the 'temps' list.
948 // 946 //
949 // As with several MacroAssembler methods, Tmp0() and Tmp1() will be used. 947 // CopyFields expects to be able to take at least two registers from
948 // MacroAssembler::TmpList().
950 void CopyFields(Register dst, Register src, CPURegList temps, unsigned count); 949 void CopyFields(Register dst, Register src, CPURegList temps, unsigned count);
951 950
952 // Copies a number of bytes from src to dst. All passed registers are 951 // Copies a number of bytes from src to dst. All passed registers are
953 // clobbered. On exit src and dst will point to the place just after where the 952 // clobbered. On exit src and dst will point to the place just after where the
954 // last byte was read or written and length will be zero. Hint may be used to 953 // last byte was read or written and length will be zero. Hint may be used to
955 // determine which is the most efficient algorithm to use for copying. 954 // determine which is the most efficient algorithm to use for copying.
956 void CopyBytes(Register dst, 955 void CopyBytes(Register dst,
957 Register src, 956 Register src,
958 Register length, 957 Register length,
959 Register scratch, 958 Register scratch,
(...skipping 480 matching lines...) Expand 10 before | Expand all | Expand 10 after
1440 1439
1441 // Test the bitfield of the heap object map with mask and set the condition 1440 // Test the bitfield of the heap object map with mask and set the condition
1442 // flags. The object register is preserved. 1441 // flags. The object register is preserved.
1443 void TestMapBitfield(Register object, uint64_t mask); 1442 void TestMapBitfield(Register object, uint64_t mask);
1444 1443
1445 // Load the elements kind field of an object, and return it in the result 1444 // Load the elements kind field of an object, and return it in the result
1446 // register. 1445 // register.
1447 void LoadElementsKind(Register result, Register object); 1446 void LoadElementsKind(Register result, Register object);
1448 1447
1449 // Compare the object in a register to a value from the root list. 1448 // Compare the object in a register to a value from the root list.
1450 // Uses the Tmp0() register as scratch.
1451 void CompareRoot(const Register& obj, Heap::RootListIndex index); 1449 void CompareRoot(const Register& obj, Heap::RootListIndex index);
1452 1450
1453 // Compare the object in a register to a value and jump if they are equal. 1451 // Compare the object in a register to a value and jump if they are equal.
1454 void JumpIfRoot(const Register& obj, 1452 void JumpIfRoot(const Register& obj,
1455 Heap::RootListIndex index, 1453 Heap::RootListIndex index,
1456 Label* if_equal); 1454 Label* if_equal);
1457 1455
1458 // Compare the object in a register to a value and jump if they are not equal. 1456 // Compare the object in a register to a value and jump if they are not equal.
1459 void JumpIfNotRoot(const Register& obj, 1457 void JumpIfNotRoot(const Register& obj,
1460 Heap::RootListIndex index, 1458 Heap::RootListIndex index,
(...skipping 86 matching lines...) Expand 10 before | Expand all | Expand 10 after
1547 void EmitSeqStringSetCharCheck(Register string, 1545 void EmitSeqStringSetCharCheck(Register string,
1548 Register index, 1546 Register index,
1549 SeqStringSetCharCheckIndexType index_type, 1547 SeqStringSetCharCheckIndexType index_type,
1550 Register scratch, 1548 Register scratch,
1551 uint32_t encoding_mask); 1549 uint32_t encoding_mask);
1552 1550
1553 // Generate code for checking access rights - used for security checks 1551 // Generate code for checking access rights - used for security checks
1554 // on access to global objects across environments. The holder register 1552 // on access to global objects across environments. The holder register
1555 // is left untouched, whereas both scratch registers are clobbered. 1553 // is left untouched, whereas both scratch registers are clobbered.
1556 void CheckAccessGlobalProxy(Register holder_reg, 1554 void CheckAccessGlobalProxy(Register holder_reg,
1557 Register scratch, 1555 Register scratch1,
1556 Register scratch2,
1558 Label* miss); 1557 Label* miss);
1559 1558
1560 // Hash the interger value in 'key' register. 1559 // Hash the interger value in 'key' register.
1561 // It uses the same algorithm as ComputeIntegerHash in utils.h. 1560 // It uses the same algorithm as ComputeIntegerHash in utils.h.
1562 void GetNumberHash(Register key, Register scratch); 1561 void GetNumberHash(Register key, Register scratch);
1563 1562
1564 // Load value from the dictionary. 1563 // Load value from the dictionary.
1565 // 1564 //
1566 // elements - holds the slow-case elements of the receiver on entry. 1565 // elements - holds the slow-case elements of the receiver on entry.
1567 // Unchanged unless 'result' is the same register. 1566 // Unchanged unless 'result' is the same register.
(...skipping 11 matching lines...) Expand all
1579 Register result, 1578 Register result,
1580 Register scratch0, 1579 Register scratch0,
1581 Register scratch1, 1580 Register scratch1,
1582 Register scratch2, 1581 Register scratch2,
1583 Register scratch3); 1582 Register scratch3);
1584 1583
1585 // --------------------------------------------------------------------------- 1584 // ---------------------------------------------------------------------------
1586 // Frames. 1585 // Frames.
1587 1586
1588 // Activation support. 1587 // Activation support.
1589 // Note that Tmp0() and Tmp1() are used as a scratch registers. This is safe
1590 // because these methods are not used in Crankshaft.
1591 void EnterFrame(StackFrame::Type type); 1588 void EnterFrame(StackFrame::Type type);
1592 void LeaveFrame(StackFrame::Type type); 1589 void LeaveFrame(StackFrame::Type type);
1593 1590
1594 // Returns map with validated enum cache in object register. 1591 // Returns map with validated enum cache in object register.
1595 void CheckEnumCache(Register object, 1592 void CheckEnumCache(Register object,
1596 Register null_value, 1593 Register null_value,
1597 Register scratch0, 1594 Register scratch0,
1598 Register scratch1, 1595 Register scratch1,
1599 Register scratch2, 1596 Register scratch2,
1600 Register scratch3, 1597 Register scratch3,
(...skipping 90 matching lines...) Expand 10 before | Expand all | Expand 10 after
1691 enum RememberedSetFinalAction { 1688 enum RememberedSetFinalAction {
1692 kReturnAtEnd, 1689 kReturnAtEnd,
1693 kFallThroughAtEnd 1690 kFallThroughAtEnd
1694 }; 1691 };
1695 1692
1696 // Record in the remembered set the fact that we have a pointer to new space 1693 // Record in the remembered set the fact that we have a pointer to new space
1697 // at the address pointed to by the addr register. Only works if addr is not 1694 // at the address pointed to by the addr register. Only works if addr is not
1698 // in new space. 1695 // in new space.
1699 void RememberedSetHelper(Register object, // Used for debug code. 1696 void RememberedSetHelper(Register object, // Used for debug code.
1700 Register addr, 1697 Register addr,
1701 Register scratch, 1698 Register scratch1,
1699 Register scratch2,
1702 SaveFPRegsMode save_fp, 1700 SaveFPRegsMode save_fp,
1703 RememberedSetFinalAction and_then); 1701 RememberedSetFinalAction and_then);
1704 1702
1705 // Push and pop the registers that can hold pointers, as defined by the 1703 // Push and pop the registers that can hold pointers, as defined by the
1706 // RegList constant kSafepointSavedRegisters. 1704 // RegList constant kSafepointSavedRegisters.
1707 void PushSafepointRegisters(); 1705 void PushSafepointRegisters();
1708 void PopSafepointRegisters(); 1706 void PopSafepointRegisters();
1709 1707
1710 void PushSafepointFPRegisters(); 1708 void PushSafepointFPRegisters();
1711 void PopSafepointFPRegisters(); 1709 void PopSafepointFPRegisters();
(...skipping 164 matching lines...) Expand 10 before | Expand all | Expand 10 after
1876 void Abort(BailoutReason reason); 1874 void Abort(BailoutReason reason);
1877 1875
1878 // Conditionally load the cached Array transitioned map of type 1876 // Conditionally load the cached Array transitioned map of type
1879 // transitioned_kind from the native context if the map in register 1877 // transitioned_kind from the native context if the map in register
1880 // map_in_out is the cached Array map in the native context of 1878 // map_in_out is the cached Array map in the native context of
1881 // expected_kind. 1879 // expected_kind.
1882 void LoadTransitionedArrayMapConditional( 1880 void LoadTransitionedArrayMapConditional(
1883 ElementsKind expected_kind, 1881 ElementsKind expected_kind,
1884 ElementsKind transitioned_kind, 1882 ElementsKind transitioned_kind,
1885 Register map_in_out, 1883 Register map_in_out,
1886 Register scratch, 1884 Register scratch1,
1885 Register scratch2,
1887 Label* no_map_match); 1886 Label* no_map_match);
1888 1887
1889 // Load the initial map for new Arrays from a JSFunction. 1888 // Load the initial map for new Arrays from a JSFunction.
1890 void LoadInitialArrayMap(Register function_in, 1889 void LoadInitialArrayMap(Register function_in,
1891 Register scratch, 1890 Register scratch1,
1891 Register scratch2,
1892 Register map_out, 1892 Register map_out,
1893 ArrayHasHoles holes); 1893 ArrayHasHoles holes);
1894 1894
1895 void LoadArrayFunction(Register function); 1895 void LoadArrayFunction(Register function);
1896 void LoadGlobalFunction(int index, Register function); 1896 void LoadGlobalFunction(int index, Register function);
1897 1897
1898 // Load the initial map from the global function. The registers function and 1898 // Load the initial map from the global function. The registers function and
1899 // map can be the same, function is then overwritten. 1899 // map can be the same, function is then overwritten.
1900 void LoadGlobalFunctionInitialMap(Register function, 1900 void LoadGlobalFunctionInitialMap(Register function,
1901 Register map, 1901 Register map,
1902 Register scratch); 1902 Register scratch);
1903 1903
1904 // -------------------------------------------------------------------------- 1904 CPURegList* TmpList() { return &tmp_list_; }
1905 // Set the registers used internally by the MacroAssembler as scratch 1905 CPURegList* FPTmpList() { return &fptmp_list_; }
1906 // registers. These registers are used to implement behaviours which are not
1907 // directly supported by A64, and where an intermediate result is required.
1908 //
1909 // Both tmp0 and tmp1 may be set to any X register except for xzr, sp,
1910 // and StackPointer(). Also, they must not be the same register (though they
1911 // may both be NoReg).
1912 //
1913 // It is valid to set either or both of these registers to NoReg if you don't
1914 // want the MacroAssembler to use any scratch registers. In a debug build, the
1915 // Assembler will assert that any registers it uses are valid. Be aware that
1916 // this check is not present in release builds. If this is a problem, use the
1917 // Assembler directly.
1918 void SetScratchRegisters(const Register& tmp0, const Register& tmp1) {
1919 // V8 assumes the macro assembler uses ip0 and ip1 as temp registers.
1920 ASSERT(tmp0.IsNone() || tmp0.Is(ip0));
1921 ASSERT(tmp1.IsNone() || tmp1.Is(ip1));
1922
1923 ASSERT(!AreAliased(xzr, csp, tmp0, tmp1));
1924 ASSERT(!AreAliased(StackPointer(), tmp0, tmp1));
1925 tmp0_ = tmp0;
1926 tmp1_ = tmp1;
1927 }
1928
1929 const Register& Tmp0() const {
1930 return tmp0_;
1931 }
1932
1933 const Register& Tmp1() const {
1934 return tmp1_;
1935 }
1936
1937 const Register WTmp0() const {
1938 return Register::Create(tmp0_.code(), kWRegSize);
1939 }
1940
1941 const Register WTmp1() const {
1942 return Register::Create(tmp1_.code(), kWRegSize);
1943 }
1944
1945 void SetFPScratchRegister(const FPRegister& fptmp0) {
1946 fptmp0_ = fptmp0;
1947 }
1948
1949 const FPRegister& FPTmp0() const {
1950 return fptmp0_;
1951 }
1952
1953 const Register AppropriateTempFor(
1954 const Register& target,
1955 const CPURegister& forbidden = NoCPUReg) const {
1956 Register candidate = forbidden.Is(Tmp0()) ? Tmp1() : Tmp0();
1957 ASSERT(!candidate.Is(target));
1958 return Register::Create(candidate.code(), target.SizeInBits());
1959 }
1960
1961 const FPRegister AppropriateTempFor(
1962 const FPRegister& target,
1963 const CPURegister& forbidden = NoCPUReg) const {
1964 USE(forbidden);
1965 FPRegister candidate = FPTmp0();
1966 ASSERT(!candidate.Is(forbidden));
1967 ASSERT(!candidate.Is(target));
1968 return FPRegister::Create(candidate.code(), target.SizeInBits());
1969 }
1970 1906
1971 // Like printf, but print at run-time from generated code. 1907 // Like printf, but print at run-time from generated code.
1972 // 1908 //
1973 // The caller must ensure that arguments for floating-point placeholders 1909 // The caller must ensure that arguments for floating-point placeholders
1974 // (such as %e, %f or %g) are FPRegisters, and that arguments for integer 1910 // (such as %e, %f or %g) are FPRegisters, and that arguments for integer
1975 // placeholders are Registers. 1911 // placeholders are Registers.
1976 // 1912 //
1977 // A maximum of four arguments may be given to any single Printf call. The 1913 // A maximum of four arguments may be given to any single Printf call. The
1978 // arguments must be of the same type, but they do not need to have the same 1914 // arguments must be of the same type, but they do not need to have the same
1979 // size. 1915 // size.
1980 // 1916 //
1981 // The following registers cannot be printed: 1917 // The following registers cannot be printed:
1982 // Tmp0(), Tmp1(), StackPointer(), csp. 1918 // StackPointer(), csp.
1983 // 1919 //
1984 // This function automatically preserves caller-saved registers so that 1920 // This function automatically preserves caller-saved registers so that
1985 // calling code can use Printf at any point without having to worry about 1921 // calling code can use Printf at any point without having to worry about
1986 // corruption. The preservation mechanism generates a lot of code. If this is 1922 // corruption. The preservation mechanism generates a lot of code. If this is
1987 // a problem, preserve the important registers manually and then call 1923 // a problem, preserve the important registers manually and then call
1988 // PrintfNoPreserve. Callee-saved registers are not used by Printf, and are 1924 // PrintfNoPreserve. Callee-saved registers are not used by Printf, and are
1989 // implicitly preserved. 1925 // implicitly preserved.
1990 // 1926 //
1991 // Unlike many MacroAssembler functions, x8 and x9 are guaranteed to be 1927 // Unlike many MacroAssembler functions, x8 and x9 are guaranteed to be
1992 // preserved, and can be printed. This allows Printf to be used during debug 1928 // preserved, and can be printed. This allows Printf to be used during debug
(...skipping 64 matching lines...) Expand 10 before | Expand all | Expand 10 after
2057 1993
2058 // Jumps to found label if a prototype map has dictionary elements. 1994 // Jumps to found label if a prototype map has dictionary elements.
2059 void JumpIfDictionaryInPrototypeChain(Register object, Register scratch0, 1995 void JumpIfDictionaryInPrototypeChain(Register object, Register scratch0,
2060 Register scratch1, Label* found); 1996 Register scratch1, Label* found);
2061 1997
2062 private: 1998 private:
2063 // Helpers for CopyFields. 1999 // Helpers for CopyFields.
2064 // These each implement CopyFields in a different way. 2000 // These each implement CopyFields in a different way.
2065 void CopyFieldsLoopPairsHelper(Register dst, Register src, unsigned count, 2001 void CopyFieldsLoopPairsHelper(Register dst, Register src, unsigned count,
2066 Register scratch1, Register scratch2, 2002 Register scratch1, Register scratch2,
2067 Register scratch3); 2003 Register scratch3, Register scratch4,
2004 Register scratch5);
2068 void CopyFieldsUnrolledPairsHelper(Register dst, Register src, unsigned count, 2005 void CopyFieldsUnrolledPairsHelper(Register dst, Register src, unsigned count,
2069 Register scratch1, Register scratch2); 2006 Register scratch1, Register scratch2,
2007 Register scratch3, Register scratch4);
2070 void CopyFieldsUnrolledHelper(Register dst, Register src, unsigned count, 2008 void CopyFieldsUnrolledHelper(Register dst, Register src, unsigned count,
2071 Register scratch1); 2009 Register scratch1, Register scratch2,
2010 Register scratch3);
2072 2011
2073 // The actual Push and Pop implementations. These don't generate any code 2012 // The actual Push and Pop implementations. These don't generate any code
2074 // other than that required for the push or pop. This allows 2013 // other than that required for the push or pop. This allows
2075 // (Push|Pop)CPURegList to bundle together run-time assertions for a large 2014 // (Push|Pop)CPURegList to bundle together run-time assertions for a large
2076 // block of registers. 2015 // block of registers.
2077 // 2016 //
2078 // Note that size is per register, and is specified in bytes. 2017 // Note that size is per register, and is specified in bytes.
2079 void PushHelper(int count, int size, 2018 void PushHelper(int count, int size,
2080 const CPURegister& src0, const CPURegister& src1, 2019 const CPURegister& src0, const CPURegister& src1,
2081 const CPURegister& src2, const CPURegister& src3); 2020 const CPURegister& src2, const CPURegister& src3);
(...skipping 60 matching lines...) Expand 10 before | Expand all | Expand 10 after
2142 // The purpose of this is to allow Aborts to be compiled whilst CEntryStub is 2081 // The purpose of this is to allow Aborts to be compiled whilst CEntryStub is
2143 // being generated. 2082 // being generated.
2144 bool use_real_aborts_; 2083 bool use_real_aborts_;
2145 2084
2146 // This handle will be patched with the code object on installation. 2085 // This handle will be patched with the code object on installation.
2147 Handle<Object> code_object_; 2086 Handle<Object> code_object_;
2148 2087
2149 // The register to use as a stack pointer for stack operations. 2088 // The register to use as a stack pointer for stack operations.
2150 Register sp_; 2089 Register sp_;
2151 2090
2152 // Scratch registers used internally by the MacroAssembler. 2091 // Scratch registers available for use by the MacroAssembler.
2153 Register tmp0_; 2092 CPURegList tmp_list_;
2154 Register tmp1_; 2093 CPURegList fptmp_list_;
2155 FPRegister fptmp0_;
2156 2094
2157 void InitializeNewString(Register string, 2095 void InitializeNewString(Register string,
2158 Register length, 2096 Register length,
2159 Heap::RootListIndex map_index, 2097 Heap::RootListIndex map_index,
2160 Register scratch1, 2098 Register scratch1,
2161 Register scratch2); 2099 Register scratch2);
2162 2100
2163 public: 2101 public:
2164 // Far branches resolving. 2102 // Far branches resolving.
2165 // 2103 //
(...skipping 87 matching lines...) Expand 10 before | Expand all | Expand 10 after
2253 private: 2191 private:
2254 MacroAssembler* masm_; 2192 MacroAssembler* masm_;
2255 #ifdef DEBUG 2193 #ifdef DEBUG
2256 size_t size_; 2194 size_t size_;
2257 Label start_; 2195 Label start_;
2258 bool previous_allow_macro_instructions_; 2196 bool previous_allow_macro_instructions_;
2259 #endif 2197 #endif
2260 }; 2198 };
2261 2199
2262 2200
2201 // This scope utility allows scratch registers to be managed safely. The
2202 // MacroAssembler's TmpList() (and FPTmpList()) is used as a pool of scratch
2203 // registers. These registers can be allocated on demand, and will be returned
2204 // at the end of the scope.
2205 //
2206 // When the scope ends, the MacroAssembler's lists will be restored to their
2207 // original state, even if the lists were modified by some other means.
2208 class UseScratchRegisterScope {
2209 public:
2210 explicit UseScratchRegisterScope(MacroAssembler* masm)
2211 : available_(masm->TmpList()),
2212 availablefp_(masm->FPTmpList()),
2213 old_available_(available_->list()),
2214 old_availablefp_(availablefp_->list()) {
2215 ASSERT(available_->type() == CPURegister::kRegister);
2216 ASSERT(availablefp_->type() == CPURegister::kFPRegister);
2217 }
2218
2219
2220 ~UseScratchRegisterScope();
2221
2222
2223 // Take a register from the appropriate temps list. It will be returned
2224 // automatically when the scope ends.
2225 Register AcquireW() { return AcquireNextAvailable(available_).W(); }
2226 Register AcquireX() { return AcquireNextAvailable(available_).X(); }
2227 FPRegister AcquireS() { return AcquireNextAvailable(availablefp_).S(); }
2228 FPRegister AcquireD() { return AcquireNextAvailable(availablefp_).D(); }
2229
2230
2231 Register AcquireSameSizeAs(const Register& reg);
2232 FPRegister AcquireSameSizeAs(const FPRegister& reg);
2233
2234
2235 // Explicitly release an acquired (or excluded) register, putting it back in
2236 // the appropriate temps list.
2237 void Release(const CPURegister& reg);
2238
2239
2240 // Make the specified registers available as scratch registers for the
2241 // duration of this scope.
2242 void Include(const CPURegList& regs);
2243 void Include(const Register& reg1,
2244 const Register& reg2 = NoReg,
2245 const Register& reg3 = NoReg,
2246 const Register& reg4 = NoReg);
2247 void Include(const FPRegister& reg1,
2248 const FPRegister& reg2 = NoFPReg,
2249 const FPRegister& reg3 = NoFPReg,
2250 const FPRegister& reg4 = NoFPReg);
2251
2252
2253 // Make sure that the specified registers are not available in this scope.
2254 // This can be used to prevent helper functions from using sensitive
2255 // registers, for example.
2256 void Exclude(const Register& reg1,
2257 const Register& reg2 = NoReg,
2258 const Register& reg3 = NoReg,
2259 const Register& reg4 = NoReg);
2260 void Exclude(const FPRegister& reg1,
2261 const FPRegister& reg2 = NoFPReg,
2262 const FPRegister& reg3 = NoFPReg,
2263 const FPRegister& reg4 = NoFPReg);
2264 void Exclude(const CPURegister& reg1,
2265 const CPURegister& reg2 = NoCPUReg,
2266 const CPURegister& reg3 = NoCPUReg,
2267 const CPURegister& reg4 = NoCPUReg);
2268
2269
2270 // Prevent any scratch registers from being used in this scope.
2271 void ExcludeAll();
2272
2273
2274 private:
2275 static CPURegister AcquireNextAvailable(CPURegList* available);
2276
2277 static void ReleaseByCode(CPURegList* available, int code);
2278
2279 static void ReleaseByRegList(CPURegList* available,
2280 RegList regs);
2281
2282 static void IncludeByRegList(CPURegList* available,
2283 RegList exclude);
2284
2285 static void ExcludeByRegList(CPURegList* available,
2286 RegList exclude);
2287
2288 // Available scratch registers.
2289 CPURegList* available_; // kRegister
2290 CPURegList* availablefp_; // kFPRegister
2291
2292 // The state of the available lists at the start of this scope.
2293 RegList old_available_; // kRegister
2294 RegList old_availablefp_; // kFPRegister
2295 };
2296
2297
2263 inline MemOperand ContextMemOperand(Register context, int index) { 2298 inline MemOperand ContextMemOperand(Register context, int index) {
2264 return MemOperand(context, Context::SlotOffset(index)); 2299 return MemOperand(context, Context::SlotOffset(index));
2265 } 2300 }
2266 2301
2267 inline MemOperand GlobalObjectMemOperand() { 2302 inline MemOperand GlobalObjectMemOperand() {
2268 return ContextMemOperand(cp, Context::GLOBAL_OBJECT_INDEX); 2303 return ContextMemOperand(cp, Context::GLOBAL_OBJECT_INDEX);
2269 } 2304 }
2270 2305
2271 2306
2272 // Encode and decode information about patchable inline SMI checks. 2307 // Encode and decode information about patchable inline SMI checks.
(...skipping 50 matching lines...) Expand 10 before | Expand all | Expand 10 after
2323 #error "Unsupported option" 2358 #error "Unsupported option"
2324 #define CODE_COVERAGE_STRINGIFY(x) #x 2359 #define CODE_COVERAGE_STRINGIFY(x) #x
2325 #define CODE_COVERAGE_TOSTRING(x) CODE_COVERAGE_STRINGIFY(x) 2360 #define CODE_COVERAGE_TOSTRING(x) CODE_COVERAGE_STRINGIFY(x)
2326 #define __FILE_LINE__ __FILE__ ":" CODE_COVERAGE_TOSTRING(__LINE__) 2361 #define __FILE_LINE__ __FILE__ ":" CODE_COVERAGE_TOSTRING(__LINE__)
2327 #define ACCESS_MASM(masm) masm->stop(__FILE_LINE__); masm-> 2362 #define ACCESS_MASM(masm) masm->stop(__FILE_LINE__); masm->
2328 #else 2363 #else
2329 #define ACCESS_MASM(masm) masm-> 2364 #define ACCESS_MASM(masm) masm->
2330 #endif 2365 #endif
2331 2366
2332 #endif // V8_A64_MACRO_ASSEMBLER_A64_H_ 2367 #endif // V8_A64_MACRO_ASSEMBLER_A64_H_
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698