OLD | NEW |
1 //===- subzero/src/IceAssemblerARM32.cpp - Assembler for ARM32 --*- C++ -*-===// | 1 //===- subzero/src/IceAssemblerARM32.cpp - Assembler for ARM32 --*- C++ -*-===// |
2 // | 2 // |
3 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file | 3 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file |
4 // for details. All rights reserved. Use of this source code is governed by a | 4 // for details. All rights reserved. Use of this source code is governed by a |
5 // BSD-style license that can be found in the LICENSE file. | 5 // BSD-style license that can be found in the LICENSE file. |
6 // | 6 // |
7 // Modified by the Subzero authors. | 7 // Modified by the Subzero authors. |
8 // | 8 // |
9 //===----------------------------------------------------------------------===// | 9 //===----------------------------------------------------------------------===// |
10 // | 10 // |
(...skipping 97 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
108 static constexpr IValueT kDivRmShift = 8; | 108 static constexpr IValueT kDivRmShift = 8; |
109 static constexpr IValueT kDivRnShift = 0; | 109 static constexpr IValueT kDivRnShift = 0; |
110 | 110 |
111 // Type of instruction encoding (bits 25-27). See ARM section A5.1 | 111 // Type of instruction encoding (bits 25-27). See ARM section A5.1 |
112 static constexpr IValueT kInstTypeDataRegister = 0; // i.e. 000 | 112 static constexpr IValueT kInstTypeDataRegister = 0; // i.e. 000 |
113 static constexpr IValueT kInstTypeDataRegShift = 0; // i.e. 000 | 113 static constexpr IValueT kInstTypeDataRegShift = 0; // i.e. 000 |
114 static constexpr IValueT kInstTypeDataImmediate = 1; // i.e. 001 | 114 static constexpr IValueT kInstTypeDataImmediate = 1; // i.e. 001 |
115 static constexpr IValueT kInstTypeMemImmediate = 2; // i.e. 010 | 115 static constexpr IValueT kInstTypeMemImmediate = 2; // i.e. 010 |
116 static constexpr IValueT kInstTypeRegisterShift = 3; // i.e. 011 | 116 static constexpr IValueT kInstTypeRegisterShift = 3; // i.e. 011 |
117 | 117 |
| 118 // Limit on number of registers in a vpush/vpop. |
| 119 static constexpr SizeT VpushVpopMaxConsecRegs = 16; |
| 120 |
118 // Offset modifier to current PC for next instruction. The offset is off by 8 | 121 // Offset modifier to current PC for next instruction. The offset is off by 8 |
119 // due to the way the ARM CPUs read PC. | 122 // due to the way the ARM CPUs read PC. |
120 static constexpr IOffsetT kPCReadOffset = 8; | 123 static constexpr IOffsetT kPCReadOffset = 8; |
121 | 124 |
122 // Mask to pull out PC offset from branch (b) instruction. | 125 // Mask to pull out PC offset from branch (b) instruction. |
123 static constexpr int kBranchOffsetBits = 24; | 126 static constexpr int kBranchOffsetBits = 24; |
124 static constexpr IOffsetT kBranchOffsetMask = 0x00ffffff; | 127 static constexpr IOffsetT kBranchOffsetMask = 0x00ffffff; |
125 | 128 |
126 IValueT encodeBool(bool B) { return B ? 1 : 0; } | 129 IValueT encodeBool(bool B) { return B ? 1 : 0; } |
127 | 130 |
(...skipping 64 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
192 // actually expect a register. | 195 // actually expect a register. |
193 OpEncodingMemEx | 196 OpEncodingMemEx |
194 }; | 197 }; |
195 | 198 |
196 IValueT getEncodedGPRegNum(const Variable *Var) { | 199 IValueT getEncodedGPRegNum(const Variable *Var) { |
197 int32_t Reg = Var->getRegNum(); | 200 int32_t Reg = Var->getRegNum(); |
198 return llvm::isa<Variable64On32>(Var) ? RegARM32::getI64PairFirstGPRNum(Reg) | 201 return llvm::isa<Variable64On32>(Var) ? RegARM32::getI64PairFirstGPRNum(Reg) |
199 : RegARM32::getEncodedGPR(Reg); | 202 : RegARM32::getEncodedGPR(Reg); |
200 } | 203 } |
201 | 204 |
| 205 IValueT getEncodedSRegNum(const Variable *Var) { |
| 206 assert(Var->hasReg()); |
| 207 assert(RegARM32::isEncodedSReg(Var->getRegNum())); |
| 208 return RegARM32::getEncodedSReg(Var->getRegNum()); |
| 209 } |
| 210 |
202 // The way an operand is encoded into a sequence of bits in functions | 211 // The way an operand is encoded into a sequence of bits in functions |
203 // encodeOperand and encodeAddress below. | 212 // encodeOperand and encodeAddress below. |
204 enum EncodedOperand { | 213 enum EncodedOperand { |
205 // Unable to encode, value left undefined. | 214 // Unable to encode, value left undefined. |
206 CantEncode = 0, | 215 CantEncode = 0, |
207 // Value is register found. | 216 // Value is register found. |
208 EncodedAsRegister, | 217 EncodedAsRegister, |
209 // Value=rrrriiiiiiii where rrrr is the rotation, and iiiiiiii is the imm8 | 218 // Value=rrrriiiiiiii where rrrr is the rotation, and iiiiiiii is the imm8 |
210 // value. | 219 // value. |
211 EncodedAsRotatedImm8, | 220 EncodedAsRotatedImm8, |
(...skipping 1778 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1990 emitMulOp(Cond, UmullOpcode, RdLo, RdHi, Rn, Rm, SetFlags, UmullName); | 1999 emitMulOp(Cond, UmullOpcode, RdLo, RdHi, Rn, Rm, SetFlags, UmullName); |
1991 } | 2000 } |
1992 | 2001 |
1993 void AssemblerARM32::uxt(const Operand *OpRd, const Operand *OpSrc0, | 2002 void AssemblerARM32::uxt(const Operand *OpRd, const Operand *OpSrc0, |
1994 CondARM32::Cond Cond) { | 2003 CondARM32::Cond Cond) { |
1995 constexpr const char *UxtName = "uxt"; | 2004 constexpr const char *UxtName = "uxt"; |
1996 constexpr IValueT UxtOpcode = B26 | B25 | B23 | B22 | B21; | 2005 constexpr IValueT UxtOpcode = B26 | B25 | B23 | B22 | B21; |
1997 emitSignExtend(Cond, UxtOpcode, OpRd, OpSrc0, UxtName); | 2006 emitSignExtend(Cond, UxtOpcode, OpRd, OpSrc0, UxtName); |
1998 } | 2007 } |
1999 | 2008 |
| 2009 void AssemblerARM32::emitVStackOp(CondARM32::Cond Cond, IValueT Opcode, |
| 2010 const Variable *OpBaseReg, |
| 2011 SizeT NumConsecRegs, const char *InstName) { |
| 2012 |
| 2013 const IValueT BaseReg = getEncodedSRegNum(OpBaseReg); |
| 2014 const IValueT DLastBit = mask(BaseReg, 0, 1); // Last bit of base register. |
| 2015 const IValueT Rd = mask(BaseReg, 1, 4); // Top 4 bits of base register. |
| 2016 assert(0 < NumConsecRegs); |
| 2017 assert(NumConsecRegs <= VpushVpopMaxConsecRegs); |
| 2018 assert((BaseReg + NumConsecRegs) <= RegARM32::getNumSRegs()); |
| 2019 verifyCondDefined(Cond, InstName); |
| 2020 AssemblerBuffer::EnsureCapacity ensured(&Buffer); |
| 2021 const IValueT Encoding = Opcode | (Cond << kConditionShift) | DLastBit | |
| 2022 (Rd << kRdShift) | NumConsecRegs; |
| 2023 emitInst(Encoding); |
| 2024 } |
| 2025 |
| 2026 void AssemblerARM32::vpop(const Variable *OpBaseReg, SizeT NumConsecRegs, |
| 2027 CondARM32::Cond Cond) { |
| 2028 // Note: Current implementation assumes that OpBaseReg is defined using S |
| 2029 // registers. It doesn't implement the D register form. |
| 2030 // |
| 2031 // VPOP - ARM section A8.8.367, encoding A2: |
| 2032 // vpop<c> <RegList> |
| 2033 // |
| 2034 // cccc11001D111101dddd1010iiiiiiii where cccc=Cond, ddddD=BaseReg, and |
| 2035 // iiiiiiii=NumConsecRegs. |
| 2036 constexpr const char *VpopName = "vpop"; |
| 2037 constexpr IValueT VpopOpcode = |
| 2038 B27 | B26 | B23 | B21 | B20 | B19 | B18 | B16 | B11 | B9; |
| 2039 emitVStackOp(Cond, VpopOpcode, OpBaseReg, NumConsecRegs, VpopName); |
| 2040 } |
| 2041 |
| 2042 void AssemblerARM32::vpush(const Variable *OpBaseReg, SizeT NumConsecRegs, |
| 2043 CondARM32::Cond Cond) { |
| 2044 // Note: Current implementation assumes that OpBaseReg is defined using S |
| 2045 // registers. It doesn't implement the D register form. |
| 2046 // |
| 2047 // VPUSH - ARM section A8.8.368, encoding A2: |
| 2048 // vpush<c> <RegList> |
| 2049 // |
| 2050 // cccc11010D101101dddd1010iiiiiiii where cccc=Cond, ddddD=BaseReg, and |
| 2051 // iiiiiiii=NumConsecRegs. |
| 2052 constexpr const char *VpushName = "vpush"; |
| 2053 constexpr IValueT VpushOpcode = |
| 2054 B27 | B26 | B24 | B21 | B19 | B18 | B16 | B11 | B9; |
| 2055 emitVStackOp(Cond, VpushOpcode, OpBaseReg, NumConsecRegs, VpushName); |
| 2056 } |
| 2057 |
2000 } // end of namespace ARM32 | 2058 } // end of namespace ARM32 |
2001 } // end of namespace Ice | 2059 } // end of namespace Ice |
OLD | NEW |