// Copyright 2017, VIXL authors // All rights reserved. // // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are met: // // * Redistributions of source code must retain the above copyright notice, // this list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above copyright notice, // this list of conditions and the following disclaimer in the documentation // and/or other materials provided with the distribution. // * Neither the name of ARM Limited nor the names of its contributors may be // used to endorse or promote products derived from this software without // specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS CONTRIBUTORS "AS IS" AND // ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED // WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE // DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE // FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR // SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER // CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, // OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. #ifndef VIXL_AARCH32_INSTRUCTIONS_AARCH32_H_ #define VIXL_AARCH32_INSTRUCTIONS_AARCH32_H_ extern "C" { #include <stdint.h> } #include <algorithm> #include <ostream> #include "code-buffer-vixl.h" #include "utils-vixl.h" #include "aarch32/constants-aarch32.h" #ifdef __arm__ #define HARDFLOAT __attribute__((noinline, pcs("aapcs-vfp"))) #else #define HARDFLOAT __attribute__((noinline)) #endif namespace vixl { namespace aarch32 { class Operand; class SOperand; class DOperand; class QOperand; class MemOperand; class AlignedMemOperand; enum AddrMode { Offset = 0, PreIndex = 1, PostIndex = 2 }; class CPURegister { public: enum RegisterType { kNoRegister = 0, kRRegister = 1, kSRegister = 2, kDRegister = 3, kQRegister = 4 }; private: static const int kCodeBits = 5; static const int kTypeBits = 4; static const int kSizeBits = 8; static const int kCodeShift = 0; static const int kTypeShift = kCodeShift + kCodeBits; static const int kSizeShift = kTypeShift + kTypeBits; static const uint32_t kCodeMask = ((1 << kCodeBits) - 1) << kCodeShift; static const uint32_t kTypeMask = ((1 << kTypeBits) - 1) << kTypeShift; static const uint32_t kSizeMask = ((1 << kSizeBits) - 1) << kSizeShift; uint32_t value_; public: CPURegister(RegisterType type, uint32_t code, int size) : value_((type << kTypeShift) | (code << kCodeShift) | (size << kSizeShift)) { #ifdef VIXL_DEBUG switch (type) { case kNoRegister: break; case kRRegister: VIXL_ASSERT(code < kNumberOfRegisters); VIXL_ASSERT(size == kRegSizeInBits); break; case kSRegister: VIXL_ASSERT(code < kNumberOfSRegisters); VIXL_ASSERT(size == kSRegSizeInBits); break; case kDRegister: VIXL_ASSERT(code < kMaxNumberOfDRegisters); VIXL_ASSERT(size == kDRegSizeInBits); break; case kQRegister: VIXL_ASSERT(code < kNumberOfQRegisters); VIXL_ASSERT(size == kQRegSizeInBits); break; default: VIXL_UNREACHABLE(); break; } #endif } RegisterType GetType() const { return static_cast<RegisterType>((value_ & kTypeMask) >> kTypeShift); } bool IsRegister() const { return GetType() == kRRegister; } bool IsS() const { return GetType() == kSRegister; } bool IsD() const { return GetType() == kDRegister; } bool IsQ() const { return GetType() == kQRegister; } bool IsVRegister() const { return IsS() || IsD() || IsQ(); } bool IsFPRegister() const { return IsS() || IsD(); } uint32_t GetCode() const { return (value_ & kCodeMask) >> kCodeShift; } uint32_t GetReg() const { return value_; } int GetSizeInBits() const { return (value_ & kSizeMask) >> kSizeShift; } int GetRegSizeInBytes() const { return (GetType() == kNoRegister) ? 0 : (GetSizeInBits() / 8); } bool Is64Bits() const { return GetSizeInBits() == 64; } bool Is128Bits() const { return GetSizeInBits() == 128; } bool IsSameFormat(CPURegister reg) { return (value_ & ~kCodeMask) == (reg.value_ & ~kCodeMask); } bool Is(CPURegister ref) const { return GetReg() == ref.GetReg(); } bool IsValid() const { return GetType() != kNoRegister; } }; class Register : public CPURegister { public: Register() : CPURegister(kNoRegister, 0, kRegSizeInBits) {} explicit Register(uint32_t code) : CPURegister(kRRegister, code % kNumberOfRegisters, kRegSizeInBits) { VIXL_ASSERT(GetCode() < kNumberOfRegisters); } bool Is(Register ref) const { return GetCode() == ref.GetCode(); } bool IsLow() const { return GetCode() < kNumberOfT32LowRegisters; } bool IsLR() const { return GetCode() == kLrCode; } bool IsPC() const { return GetCode() == kPcCode; } bool IsSP() const { return GetCode() == kSpCode; } }; std::ostream& operator<<(std::ostream& os, const Register reg); class RegisterOrAPSR_nzcv { uint32_t code_; public: explicit RegisterOrAPSR_nzcv(uint32_t code) : code_(code) { VIXL_ASSERT(code_ < kNumberOfRegisters); } bool IsAPSR_nzcv() const { return code_ == kPcCode; } uint32_t GetCode() const { return code_; } Register AsRegister() const { VIXL_ASSERT(!IsAPSR_nzcv()); return Register(code_); } }; const RegisterOrAPSR_nzcv APSR_nzcv(kPcCode); inline std::ostream& operator<<(std::ostream& os, const RegisterOrAPSR_nzcv reg) { if (reg.IsAPSR_nzcv()) return os << "APSR_nzcv"; return os << reg.AsRegister(); } class SRegister; class DRegister; class QRegister; class VRegister : public CPURegister { public: VRegister() : CPURegister(kNoRegister, 0, 0) {} VRegister(RegisterType type, uint32_t code, int size) : CPURegister(type, code, size) {} SRegister S() const; DRegister D() const; QRegister Q() const; }; class SRegister : public VRegister { public: SRegister() : VRegister(kNoRegister, 0, kSRegSizeInBits) {} explicit SRegister(uint32_t code) : VRegister(kSRegister, code, kSRegSizeInBits) {} uint32_t Encode(int single_bit_field, int four_bit_field_lowest_bit) const { if (four_bit_field_lowest_bit == 0) { return ((GetCode() & 0x1) << single_bit_field) | ((GetCode() & 0x1e) >> 1); } return ((GetCode() & 0x1) << single_bit_field) | ((GetCode() & 0x1e) << (four_bit_field_lowest_bit - 1)); } }; inline unsigned ExtractSRegister(uint32_t instr, int single_bit_field, int four_bit_field_lowest_bit) { VIXL_ASSERT(single_bit_field > 0); if (four_bit_field_lowest_bit == 0) { return ((instr << 1) & 0x1e) | ((instr >> single_bit_field) & 0x1); } return ((instr >> (four_bit_field_lowest_bit - 1)) & 0x1e) | ((instr >> single_bit_field) & 0x1); } inline std::ostream& operator<<(std::ostream& os, const SRegister reg) { return os << "s" << reg.GetCode(); } class DRegister : public VRegister { public: DRegister() : VRegister(kNoRegister, 0, kDRegSizeInBits) {} explicit DRegister(uint32_t code) : VRegister(kDRegister, code, kDRegSizeInBits) {} SRegister GetLane(uint32_t lane) const { uint32_t lane_count = kDRegSizeInBits / kSRegSizeInBits; VIXL_ASSERT(lane < lane_count); VIXL_ASSERT(GetCode() * lane_count < kNumberOfSRegisters); return SRegister(GetCode() * lane_count + lane); } uint32_t Encode(int single_bit_field, int four_bit_field_lowest_bit) const { VIXL_ASSERT(single_bit_field >= 4); return ((GetCode() & 0x10) << (single_bit_field - 4)) | ((GetCode() & 0xf) << four_bit_field_lowest_bit); } }; inline unsigned ExtractDRegister(uint32_t instr, int single_bit_field, int four_bit_field_lowest_bit) { VIXL_ASSERT(single_bit_field >= 4); return ((instr >> (single_bit_field - 4)) & 0x10) | ((instr >> four_bit_field_lowest_bit) & 0xf); } inline std::ostream& operator<<(std::ostream& os, const DRegister reg) { return os << "d" << reg.GetCode(); } enum DataTypeType { kDataTypeS = 0x100, kDataTypeU = 0x200, kDataTypeF = 0x300, kDataTypeI = 0x400, kDataTypeP = 0x500, kDataTypeUntyped = 0x600 }; const int kDataTypeSizeMask = 0x0ff; const int kDataTypeTypeMask = 0x100; enum DataTypeValue { kDataTypeValueInvalid = 0x000, kDataTypeValueNone = 0x001, // value used when dt is ignored. S8 = kDataTypeS | 8, S16 = kDataTypeS | 16, S32 = kDataTypeS | 32, S64 = kDataTypeS | 64, U8 = kDataTypeU | 8, U16 = kDataTypeU | 16, U32 = kDataTypeU | 32, U64 = kDataTypeU | 64, F16 = kDataTypeF | 16, F32 = kDataTypeF | 32, F64 = kDataTypeF | 64, I8 = kDataTypeI | 8, I16 = kDataTypeI | 16, I32 = kDataTypeI | 32, I64 = kDataTypeI | 64, P8 = kDataTypeP | 8, P64 = kDataTypeP | 64, Untyped8 = kDataTypeUntyped | 8, Untyped16 = kDataTypeUntyped | 16, Untyped32 = kDataTypeUntyped | 32, Untyped64 = kDataTypeUntyped | 64 }; class DataType { DataTypeValue value_; public: explicit DataType(uint32_t size) : value_(static_cast<DataTypeValue>(kDataTypeUntyped | size)) { VIXL_ASSERT((size == 8) || (size == 16) || (size == 32) || (size == 64)); } // Users should be able to use "S8", "S6" and so forth to instantiate this // class. DataType(DataTypeValue value) : value_(value) {} // NOLINT(runtime/explicit) DataTypeValue GetValue() const { return value_; } DataTypeType GetType() const { return static_cast<DataTypeType>(value_ & kDataTypeTypeMask); } uint32_t GetSize() const { return value_ & kDataTypeSizeMask; } bool IsSize(uint32_t size) const { return (value_ & kDataTypeSizeMask) == size; } const char* GetName() const; bool Is(DataType type) const { return value_ == type.value_; } bool Is(DataTypeValue value) const { return value_ == value; } bool Is(DataTypeType type) const { return GetType() == type; } bool IsNoneOr(DataTypeValue value) const { return (value_ == value) || (value_ == kDataTypeValueNone); } bool Is(DataTypeType type, uint32_t size) const { return value_ == static_cast<DataTypeValue>(type | size); } bool IsNoneOr(DataTypeType type, uint32_t size) const { return Is(type, size) || Is(kDataTypeValueNone); } }; inline std::ostream& operator<<(std::ostream& os, DataType dt) { return os << dt.GetName(); } class DRegisterLane : public DRegister { uint32_t lane_; public: DRegisterLane(DRegister reg, uint32_t lane) : DRegister(reg.GetCode()), lane_(lane) {} DRegisterLane(uint32_t code, uint32_t lane) : DRegister(code), lane_(lane) {} uint32_t GetLane() const { return lane_; } uint32_t EncodeX(DataType dt, int single_bit_field, int four_bit_field_lowest_bit) const { VIXL_ASSERT(single_bit_field >= 4); uint32_t value = lane_ << ((dt.GetSize() == 16) ? 3 : 4) | GetCode(); return ((value & 0x10) << (single_bit_field - 4)) | ((value & 0xf) << four_bit_field_lowest_bit); } }; inline unsigned ExtractDRegisterAndLane(uint32_t instr, DataType dt, int single_bit_field, int four_bit_field_lowest_bit, int* lane) { VIXL_ASSERT(single_bit_field >= 4); uint32_t value = ((instr >> (single_bit_field - 4)) & 0x10) | ((instr >> four_bit_field_lowest_bit) & 0xf); if (dt.GetSize() == 16) { *lane = value >> 3; return value & 0x7; } *lane = value >> 4; return value & 0xf; } inline std::ostream& operator<<(std::ostream& os, const DRegisterLane lane) { os << "d" << lane.GetCode() << "["; if (lane.GetLane() == static_cast<uint32_t>(-1)) return os << "??]"; return os << lane.GetLane() << "]"; } class QRegister : public VRegister { public: QRegister() : VRegister(kNoRegister, 0, kQRegSizeInBits) {} explicit QRegister(uint32_t code) : VRegister(kQRegister, code, kQRegSizeInBits) {} uint32_t Encode(int offset) { return GetCode() << offset; } DRegister GetDLane(uint32_t lane) const { uint32_t lane_count = kQRegSizeInBits / kDRegSizeInBits; VIXL_ASSERT(lane < lane_count); return DRegister(GetCode() * lane_count + lane); } DRegister GetLowDRegister() const { return DRegister(GetCode() * 2); } DRegister GetHighDRegister() const { return DRegister(1 + GetCode() * 2); } SRegister GetSLane(uint32_t lane) const { uint32_t lane_count = kQRegSizeInBits / kSRegSizeInBits; VIXL_ASSERT(lane < lane_count); VIXL_ASSERT(GetCode() * lane_count < kNumberOfSRegisters); return SRegister(GetCode() * lane_count + lane); } uint32_t Encode(int single_bit_field, int four_bit_field_lowest_bit) { // Encode "code * 2". VIXL_ASSERT(single_bit_field >= 3); return ((GetCode() & 0x8) << (single_bit_field - 3)) | ((GetCode() & 0x7) << (four_bit_field_lowest_bit + 1)); } }; inline unsigned ExtractQRegister(uint32_t instr, int single_bit_field, int four_bit_field_lowest_bit) { VIXL_ASSERT(single_bit_field >= 3); return ((instr >> (single_bit_field - 3)) & 0x8) | ((instr >> (four_bit_field_lowest_bit + 1)) & 0x7); } inline std::ostream& operator<<(std::ostream& os, const QRegister reg) { return os << "q" << reg.GetCode(); } // clang-format off #define AARCH32_REGISTER_CODE_LIST(R) \ R(0) R(1) R(2) R(3) R(4) R(5) R(6) R(7) \ R(8) R(9) R(10) R(11) R(12) R(13) R(14) R(15) // clang-format on #define DEFINE_REGISTER(N) const Register r##N(N); AARCH32_REGISTER_CODE_LIST(DEFINE_REGISTER) #undef DEFINE_REGISTER #undef AARCH32_REGISTER_CODE_LIST enum RegNum { kIPRegNum = 12, kSPRegNum = 13, kLRRegNum = 14, kPCRegNum = 15 }; const Register ip(kIPRegNum); const Register sp(kSPRegNum); const Register pc(kPCRegNum); const Register lr(kLRRegNum); const Register NoReg; const VRegister NoVReg; // clang-format off #define SREGISTER_CODE_LIST(R) \ R(0) R(1) R(2) R(3) R(4) R(5) R(6) R(7) \ R(8) R(9) R(10) R(11) R(12) R(13) R(14) R(15) \ R(16) R(17) R(18) R(19) R(20) R(21) R(22) R(23) \ R(24) R(25) R(26) R(27) R(28) R(29) R(30) R(31) // clang-format on #define DEFINE_REGISTER(N) const SRegister s##N(N); SREGISTER_CODE_LIST(DEFINE_REGISTER) #undef DEFINE_REGISTER #undef SREGISTER_CODE_LIST const SRegister NoSReg; // clang-format off #define DREGISTER_CODE_LIST(R) \ R(0) R(1) R(2) R(3) R(4) R(5) R(6) R(7) \ R(8) R(9) R(10) R(11) R(12) R(13) R(14) R(15) \ R(16) R(17) R(18) R(19) R(20) R(21) R(22) R(23) \ R(24) R(25) R(26) R(27) R(28) R(29) R(30) R(31) // clang-format on #define DEFINE_REGISTER(N) const DRegister d##N(N); DREGISTER_CODE_LIST(DEFINE_REGISTER) #undef DEFINE_REGISTER #undef DREGISTER_CODE_LIST const DRegister NoDReg; // clang-format off #define QREGISTER_CODE_LIST(R) \ R(0) R(1) R(2) R(3) R(4) R(5) R(6) R(7) \ R(8) R(9) R(10) R(11) R(12) R(13) R(14) R(15) // clang-format on #define DEFINE_REGISTER(N) const QRegister q##N(N); QREGISTER_CODE_LIST(DEFINE_REGISTER) #undef DEFINE_REGISTER #undef QREGISTER_CODE_LIST const QRegister NoQReg; class RegisterList { public: RegisterList() : list_(0) {} RegisterList(Register reg) // NOLINT(runtime/explicit) : list_(RegisterToList(reg)) {} RegisterList(Register reg1, Register reg2) : list_(RegisterToList(reg1) | RegisterToList(reg2)) {} RegisterList(Register reg1, Register reg2, Register reg3) : list_(RegisterToList(reg1) | RegisterToList(reg2) | RegisterToList(reg3)) {} RegisterList(Register reg1, Register reg2, Register reg3, Register reg4) : list_(RegisterToList(reg1) | RegisterToList(reg2) | RegisterToList(reg3) | RegisterToList(reg4)) {} explicit RegisterList(uint32_t list) : list_(list) {} uint32_t GetList() const { return list_; } void SetList(uint32_t list) { list_ = list; } bool Includes(const Register& reg) const { return (list_ & RegisterToList(reg)) != 0; } void Combine(const RegisterList& other) { list_ |= other.GetList(); } void Combine(const Register& reg) { list_ |= RegisterToList(reg); } void Remove(const RegisterList& other) { list_ &= ~other.GetList(); } void Remove(const Register& reg) { list_ &= ~RegisterToList(reg); } bool Overlaps(const RegisterList& other) const { return (list_ & other.list_) != 0; } bool IsR0toR7orPC() const { // True if all the registers from the list are not from r8-r14. return (list_ & 0x7f00) == 0; } bool IsR0toR7orLR() const { // True if all the registers from the list are not from r8-r13 nor from r15. return (list_ & 0xbf00) == 0; } Register GetFirstAvailableRegister() const; bool IsEmpty() const { return list_ == 0; } static RegisterList Union(const RegisterList& list_1, const RegisterList& list_2) { return RegisterList(list_1.list_ | list_2.list_); } static RegisterList Union(const RegisterList& list_1, const RegisterList& list_2, const RegisterList& list_3) { return Union(list_1, Union(list_2, list_3)); } static RegisterList Union(const RegisterList& list_1, const RegisterList& list_2, const RegisterList& list_3, const RegisterList& list_4) { return Union(Union(list_1, list_2), Union(list_3, list_4)); } static RegisterList Intersection(const RegisterList& list_1, const RegisterList& list_2) { return RegisterList(list_1.list_ & list_2.list_); } static RegisterList Intersection(const RegisterList& list_1, const RegisterList& list_2, const RegisterList& list_3) { return Intersection(list_1, Intersection(list_2, list_3)); } static RegisterList Intersection(const RegisterList& list_1, const RegisterList& list_2, const RegisterList& list_3, const RegisterList& list_4) { return Intersection(Intersection(list_1, list_2), Intersection(list_3, list_4)); } private: static uint32_t RegisterToList(Register reg) { if (reg.GetType() == CPURegister::kNoRegister) { return 0; } else { return UINT32_C(1) << reg.GetCode(); } } // Bitfield representation of all registers in the list // (1 for r0, 2 for r1, 4 for r2, ...). uint32_t list_; }; inline uint32_t GetRegisterListEncoding(const RegisterList& registers, int first, int count) { return (registers.GetList() >> first) & ((1 << count) - 1); } std::ostream& operator<<(std::ostream& os, RegisterList registers); class VRegisterList { public: VRegisterList() : list_(0) {} explicit VRegisterList(VRegister reg) : list_(RegisterToList(reg)) {} VRegisterList(VRegister reg1, VRegister reg2) : list_(RegisterToList(reg1) | RegisterToList(reg2)) {} VRegisterList(VRegister reg1, VRegister reg2, VRegister reg3) : list_(RegisterToList(reg1) | RegisterToList(reg2) | RegisterToList(reg3)) {} VRegisterList(VRegister reg1, VRegister reg2, VRegister reg3, VRegister reg4) : list_(RegisterToList(reg1) | RegisterToList(reg2) | RegisterToList(reg3) | RegisterToList(reg4)) {} explicit VRegisterList(uint64_t list) : list_(list) {} uint64_t GetList() const { return list_; } void SetList(uint64_t list) { list_ = list; } // Because differently-sized V registers overlap with one another, there is no // way to implement a single 'Includes' function in a way that is unsurprising // for all existing uses. bool IncludesAllOf(const VRegister& reg) const { return (list_ & RegisterToList(reg)) == RegisterToList(reg); } bool IncludesAliasOf(const VRegister& reg) const { return (list_ & RegisterToList(reg)) != 0; } void Combine(const VRegisterList& other) { list_ |= other.GetList(); } void Combine(const VRegister& reg) { list_ |= RegisterToList(reg); } void Remove(const VRegisterList& other) { list_ &= ~other.GetList(); } void Remove(const VRegister& reg) { list_ &= ~RegisterToList(reg); } bool Overlaps(const VRegisterList& other) const { return (list_ & other.list_) != 0; } QRegister GetFirstAvailableQRegister() const; DRegister GetFirstAvailableDRegister() const; SRegister GetFirstAvailableSRegister() const; bool IsEmpty() const { return list_ == 0; } static VRegisterList Union(const VRegisterList& list_1, const VRegisterList& list_2) { return VRegisterList(list_1.list_ | list_2.list_); } static VRegisterList Union(const VRegisterList& list_1, const VRegisterList& list_2, const VRegisterList& list_3) { return Union(list_1, Union(list_2, list_3)); } static VRegisterList Union(const VRegisterList& list_1, const VRegisterList& list_2, const VRegisterList& list_3, const VRegisterList& list_4) { return Union(Union(list_1, list_2), Union(list_3, list_4)); } static VRegisterList Intersection(const VRegisterList& list_1, const VRegisterList& list_2) { return VRegisterList(list_1.list_ & list_2.list_); } static VRegisterList Intersection(const VRegisterList& list_1, const VRegisterList& list_2, const VRegisterList& list_3) { return Intersection(list_1, Intersection(list_2, list_3)); } static VRegisterList Intersection(const VRegisterList& list_1, const VRegisterList& list_2, const VRegisterList& list_3, const VRegisterList& list_4) { return Intersection(Intersection(list_1, list_2), Intersection(list_3, list_4)); } private: static uint64_t RegisterToList(VRegister reg) { if (reg.GetType() == CPURegister::kNoRegister) { return 0; } else { switch (reg.GetSizeInBits()) { case kQRegSizeInBits: return UINT64_C(0xf) << (reg.GetCode() * 4); case kDRegSizeInBits: return UINT64_C(0x3) << (reg.GetCode() * 2); case kSRegSizeInBits: return UINT64_C(0x1) << reg.GetCode(); default: VIXL_UNREACHABLE(); return 0; } } } // Bitfield representation of all registers in the list. // (0x3 for d0, 0xc0 for d1, 0x30 for d2, ...). We have one, two or four bits // per register according to their size. This way we can make sure that we // account for overlapping registers. // A register is wholly included in this list only if all of its bits are set. // A register is aliased by the list if at least one of its bits are set. // The IncludesAllOf and IncludesAliasOf helpers are provided to make this // distinction clear. uint64_t list_; }; class SRegisterList { SRegister first_; int length_; public: explicit SRegisterList(SRegister reg) : first_(reg.GetCode()), length_(1) {} SRegisterList(SRegister first, int length) : first_(first.GetCode()), length_(length) { VIXL_ASSERT(length >= 0); } SRegister GetSRegister(int n) const { VIXL_ASSERT(n >= 0); VIXL_ASSERT(n < length_); return SRegister((first_.GetCode() + n) % kNumberOfSRegisters); } const SRegister& GetFirstSRegister() const { return first_; } SRegister GetLastSRegister() const { return GetSRegister(length_ - 1); } int GetLength() const { return length_; } }; std::ostream& operator<<(std::ostream& os, SRegisterList registers); class DRegisterList { DRegister first_; int length_; public: explicit DRegisterList(DRegister reg) : first_(reg.GetCode()), length_(1) {} DRegisterList(DRegister first, int length) : first_(first.GetCode()), length_(length) { VIXL_ASSERT(length >= 0); } DRegister GetDRegister(int n) const { VIXL_ASSERT(n >= 0); VIXL_ASSERT(n < length_); return DRegister((first_.GetCode() + n) % kMaxNumberOfDRegisters); } const DRegister& GetFirstDRegister() const { return first_; } DRegister GetLastDRegister() const { return GetDRegister(length_ - 1); } int GetLength() const { return length_; } }; std::ostream& operator<<(std::ostream& os, DRegisterList registers); enum SpacingType { kSingle, kDouble }; enum TransferType { kMultipleLanes, kOneLane, kAllLanes }; class NeonRegisterList { DRegister first_; SpacingType spacing_; TransferType type_; int lane_; int length_; public: NeonRegisterList(DRegister reg, TransferType type) : first_(reg.GetCode()), spacing_(kSingle), type_(type), lane_(-1), length_(1) { VIXL_ASSERT(type_ != kOneLane); } NeonRegisterList(DRegister reg, int lane) : first_(reg.GetCode()), spacing_(kSingle), type_(kOneLane), lane_(lane), length_(1) { VIXL_ASSERT((lane_ >= 0) && (lane_ < 8)); } NeonRegisterList(DRegister first, DRegister last, SpacingType spacing, TransferType type) : first_(first.GetCode()), spacing_(spacing), type_(type), lane_(-1) { VIXL_ASSERT(type != kOneLane); VIXL_ASSERT(first.GetCode() <= last.GetCode()); int range = last.GetCode() - first.GetCode(); VIXL_ASSERT(IsSingleSpaced() || IsMultiple(range, 2)); length_ = (IsDoubleSpaced() ? (range / 2) : range) + 1; VIXL_ASSERT(length_ <= 4); } NeonRegisterList(DRegister first, DRegister last, SpacingType spacing, int lane) : first_(first.GetCode()), spacing_(spacing), type_(kOneLane), lane_(lane) { VIXL_ASSERT((lane >= 0) && (lane < 8)); VIXL_ASSERT(first.GetCode() <= last.GetCode()); int range = last.GetCode() - first.GetCode(); VIXL_ASSERT(IsSingleSpaced() || IsMultiple(range, 2)); length_ = (IsDoubleSpaced() ? (range / 2) : range) + 1; VIXL_ASSERT(length_ <= 4); } DRegister GetDRegister(int n) const { VIXL_ASSERT(n >= 0); VIXL_ASSERT(n < length_); unsigned code = first_.GetCode() + (IsDoubleSpaced() ? (2 * n) : n); VIXL_ASSERT(code < kMaxNumberOfDRegisters); return DRegister(code); } const DRegister& GetFirstDRegister() const { return first_; } DRegister GetLastDRegister() const { return GetDRegister(length_ - 1); } int GetLength() const { return length_; } bool IsSingleSpaced() const { return spacing_ == kSingle; } bool IsDoubleSpaced() const { return spacing_ == kDouble; } bool IsTransferAllLanes() const { return type_ == kAllLanes; } bool IsTransferOneLane() const { return type_ == kOneLane; } bool IsTransferMultipleLanes() const { return type_ == kMultipleLanes; } int GetTransferLane() const { return lane_; } }; std::ostream& operator<<(std::ostream& os, NeonRegisterList registers); enum SpecialRegisterType { APSR = 0, CPSR = 0, SPSR = 1 }; class SpecialRegister { uint32_t reg_; public: explicit SpecialRegister(uint32_t reg) : reg_(reg) {} SpecialRegister(SpecialRegisterType reg) // NOLINT(runtime/explicit) : reg_(reg) {} uint32_t GetReg() const { return reg_; } const char* GetName() const; bool Is(SpecialRegister value) const { return reg_ == value.reg_; } bool Is(uint32_t value) const { return reg_ == value; } bool IsNot(uint32_t value) const { return reg_ != value; } }; inline std::ostream& operator<<(std::ostream& os, SpecialRegister reg) { return os << reg.GetName(); } enum BankedRegisterType { R8_usr = 0x00, R9_usr = 0x01, R10_usr = 0x02, R11_usr = 0x03, R12_usr = 0x04, SP_usr = 0x05, LR_usr = 0x06, R8_fiq = 0x08, R9_fiq = 0x09, R10_fiq = 0x0a, R11_fiq = 0x0b, R12_fiq = 0x0c, SP_fiq = 0x0d, LR_fiq = 0x0e, LR_irq = 0x10, SP_irq = 0x11, LR_svc = 0x12, SP_svc = 0x13, LR_abt = 0x14, SP_abt = 0x15, LR_und = 0x16, SP_und = 0x17, LR_mon = 0x1c, SP_mon = 0x1d, ELR_hyp = 0x1e, SP_hyp = 0x1f, SPSR_fiq = 0x2e, SPSR_irq = 0x30, SPSR_svc = 0x32, SPSR_abt = 0x34, SPSR_und = 0x36, SPSR_mon = 0x3c, SPSR_hyp = 0x3e }; class BankedRegister { uint32_t reg_; public: explicit BankedRegister(unsigned reg) : reg_(reg) {} BankedRegister(BankedRegisterType reg) // NOLINT(runtime/explicit) : reg_(reg) {} uint32_t GetCode() const { return reg_; } const char* GetName() const; }; inline std::ostream& operator<<(std::ostream& os, BankedRegister reg) { return os << reg.GetName(); } enum MaskedSpecialRegisterType { APSR_nzcvq = 0x08, APSR_g = 0x04, APSR_nzcvqg = 0x0c, CPSR_c = 0x01, CPSR_x = 0x02, CPSR_xc = 0x03, CPSR_s = APSR_g, CPSR_sc = 0x05, CPSR_sx = 0x06, CPSR_sxc = 0x07, CPSR_f = APSR_nzcvq, CPSR_fc = 0x09, CPSR_fx = 0x0a, CPSR_fxc = 0x0b, CPSR_fs = APSR_nzcvqg, CPSR_fsc = 0x0d, CPSR_fsx = 0x0e, CPSR_fsxc = 0x0f, SPSR_c = 0x11, SPSR_x = 0x12, SPSR_xc = 0x13, SPSR_s = 0x14, SPSR_sc = 0x15, SPSR_sx = 0x16, SPSR_sxc = 0x17, SPSR_f = 0x18, SPSR_fc = 0x19, SPSR_fx = 0x1a, SPSR_fxc = 0x1b, SPSR_fs = 0x1c, SPSR_fsc = 0x1d, SPSR_fsx = 0x1e, SPSR_fsxc = 0x1f }; class MaskedSpecialRegister { uint32_t reg_; public: explicit MaskedSpecialRegister(uint32_t reg) : reg_(reg) { VIXL_ASSERT(reg <= SPSR_fsxc); } MaskedSpecialRegister( MaskedSpecialRegisterType reg) // NOLINT(runtime/explicit) : reg_(reg) {} uint32_t GetReg() const { return reg_; } const char* GetName() const; bool Is(MaskedSpecialRegister value) const { return reg_ == value.reg_; } bool Is(uint32_t value) const { return reg_ == value; } bool IsNot(uint32_t value) const { return reg_ != value; } }; inline std::ostream& operator<<(std::ostream& os, MaskedSpecialRegister reg) { return os << reg.GetName(); } enum SpecialFPRegisterType { FPSID = 0x0, FPSCR = 0x1, MVFR2 = 0x5, MVFR1 = 0x6, MVFR0 = 0x7, FPEXC = 0x8 }; class SpecialFPRegister { uint32_t reg_; public: explicit SpecialFPRegister(uint32_t reg) : reg_(reg) { #ifdef VIXL_DEBUG switch (reg) { case FPSID: case FPSCR: case MVFR2: case MVFR1: case MVFR0: case FPEXC: break; default: VIXL_UNREACHABLE(); } #endif } SpecialFPRegister(SpecialFPRegisterType reg) // NOLINT(runtime/explicit) : reg_(reg) {} uint32_t GetReg() const { return reg_; } const char* GetName() const; bool Is(SpecialFPRegister value) const { return reg_ == value.reg_; } bool Is(uint32_t value) const { return reg_ == value; } bool IsNot(uint32_t value) const { return reg_ != value; } }; inline std::ostream& operator<<(std::ostream& os, SpecialFPRegister reg) { return os << reg.GetName(); } class CRegister { uint32_t code_; public: explicit CRegister(uint32_t code) : code_(code) { VIXL_ASSERT(code < kNumberOfRegisters); } uint32_t GetCode() const { return code_; } bool Is(CRegister value) const { return code_ == value.code_; } }; inline std::ostream& operator<<(std::ostream& os, const CRegister reg) { return os << "c" << reg.GetCode(); } // clang-format off #define CREGISTER_CODE_LIST(R) \ R(0) R(1) R(2) R(3) R(4) R(5) R(6) R(7) \ R(8) R(9) R(10) R(11) R(12) R(13) R(14) R(15) // clang-format on #define DEFINE_CREGISTER(N) const CRegister c##N(N); CREGISTER_CODE_LIST(DEFINE_CREGISTER) enum CoprocessorName { p10 = 10, p11 = 11, p14 = 14, p15 = 15 }; class Coprocessor { uint32_t coproc_; public: explicit Coprocessor(uint32_t coproc) : coproc_(coproc) {} Coprocessor(CoprocessorName coproc) // NOLINT(runtime/explicit) : coproc_(static_cast<uint32_t>(coproc)) {} bool Is(Coprocessor coproc) const { return coproc_ == coproc.coproc_; } bool Is(CoprocessorName coproc) const { return coproc_ == coproc; } uint32_t GetCoprocessor() const { return coproc_; } }; inline std::ostream& operator<<(std::ostream& os, Coprocessor coproc) { return os << "p" << coproc.GetCoprocessor(); } enum ConditionType { eq = 0, ne = 1, cs = 2, cc = 3, mi = 4, pl = 5, vs = 6, vc = 7, hi = 8, ls = 9, ge = 10, lt = 11, gt = 12, le = 13, al = 14, hs = cs, lo = cc }; class Condition { uint32_t condition_; static const uint32_t kNever = 15; static const uint32_t kMask = 0xf; static const uint32_t kNone = 0x10 | al; public: static const Condition None() { return Condition(kNone); } static const Condition Never() { return Condition(kNever); } explicit Condition(uint32_t condition) : condition_(condition) { VIXL_ASSERT(condition <= kNone); } // Users should be able to use "eq", "ne" and so forth to instantiate this // class. Condition(ConditionType condition) // NOLINT(runtime/explicit) : condition_(condition) {} uint32_t GetCondition() const { return condition_ & kMask; } bool IsNone() const { return condition_ == kNone; } const char* GetName() const; bool Is(Condition value) const { return condition_ == value.condition_; } bool Is(uint32_t value) const { return condition_ == value; } bool IsNot(uint32_t value) const { return condition_ != value; } bool IsNever() const { return condition_ == kNever; } bool IsNotNever() const { return condition_ != kNever; } Condition Negate() const { VIXL_ASSERT(IsNot(al) && IsNot(kNever)); return Condition(condition_ ^ 1); } }; inline std::ostream& operator<<(std::ostream& os, Condition condition) { return os << condition.GetName(); } enum SignType { plus, minus }; class Sign { public: Sign() : sign_(plus) {} Sign(SignType sign) : sign_(sign) {} // NOLINT(runtime/explicit) const char* GetName() const { return (IsPlus() ? "" : "-"); } bool IsPlus() const { return sign_ == plus; } bool IsMinus() const { return sign_ == minus; } int32_t ApplyTo(uint32_t value) { return IsPlus() ? value : -value; } private: SignType sign_; }; inline std::ostream& operator<<(std::ostream& os, Sign sign) { return os << sign.GetName(); } enum ShiftType { LSL = 0x0, LSR = 0x1, ASR = 0x2, ROR = 0x3, RRX = 0x4 }; class Shift { public: Shift() : shift_(LSL) {} Shift(ShiftType shift) : shift_(shift) {} // NOLINT(runtime/explicit) explicit Shift(uint32_t shift) : shift_(static_cast<ShiftType>(shift)) {} const Shift& GetShift() const { return *this; } ShiftType GetType() const { return shift_; } uint32_t GetValue() const { return shift_; } const char* GetName() const; bool IsLSL() const { return shift_ == LSL; } bool IsLSR() const { return shift_ == LSR; } bool IsASR() const { return shift_ == ASR; } bool IsROR() const { return shift_ == ROR; } bool IsRRX() const { return shift_ == RRX; } bool Is(Shift value) const { return shift_ == value.shift_; } bool IsNot(Shift value) const { return shift_ != value.shift_; } bool IsValidAmount(uint32_t amount) const; static const Shift NoShift; protected: void SetType(ShiftType s) { shift_ = s; } private: ShiftType shift_; }; inline std::ostream& operator<<(std::ostream& os, Shift shift) { return os << shift.GetName(); } class ImmediateShiftOperand : public Shift { public: // Constructor used for assembly. ImmediateShiftOperand(Shift shift, uint32_t amount) : Shift(shift), amount_(amount) { #ifdef VIXL_DEBUG switch (shift.GetType()) { case LSL: VIXL_ASSERT(amount <= 31); break; case ROR: VIXL_ASSERT(amount > 0); VIXL_ASSERT(amount <= 31); break; case LSR: case ASR: VIXL_ASSERT(amount > 0); VIXL_ASSERT(amount <= 32); break; case RRX: VIXL_ASSERT(amount == 0); break; default: VIXL_UNREACHABLE(); break; } #endif } // Constructor used for disassembly. ImmediateShiftOperand(int shift, int amount); uint32_t GetAmount() const { return amount_; } bool Is(const ImmediateShiftOperand& rhs) const { return amount_ == (rhs.amount_) && Shift::Is(*this); } private: uint32_t amount_; }; inline std::ostream& operator<<(std::ostream& os, ImmediateShiftOperand const& shift_operand) { if (shift_operand.IsLSL() && shift_operand.GetAmount() == 0) return os; if (shift_operand.IsRRX()) return os << ", rrx"; return os << ", " << shift_operand.GetName() << " #" << shift_operand.GetAmount(); } class RegisterShiftOperand : public Shift { public: RegisterShiftOperand(ShiftType shift, Register shift_register) : Shift(shift), shift_register_(shift_register) { VIXL_ASSERT(!IsRRX() && shift_register_.IsValid()); } const Register GetShiftRegister() const { return shift_register_; } bool Is(const RegisterShiftOperand& rhs) const { return shift_register_.Is(rhs.shift_register_) && Shift::Is(*this); } private: Register shift_register_; }; inline std::ostream& operator<<(std::ostream& s, const RegisterShiftOperand& shift_operand) { return s << shift_operand.GetName() << " " << shift_operand.GetShiftRegister(); } enum EncodingSizeType { Best, Narrow, Wide }; class EncodingSize { uint32_t size_; public: explicit EncodingSize(uint32_t size) : size_(size) {} EncodingSize(EncodingSizeType size) // NOLINT(runtime/explicit) : size_(size) {} uint32_t GetSize() const { return size_; } const char* GetName() const; bool IsBest() const { return size_ == Best; } bool IsNarrow() const { return size_ == Narrow; } bool IsWide() const { return size_ == Wide; } }; inline std::ostream& operator<<(std::ostream& os, EncodingSize size) { return os << size.GetName(); } enum WriteBackValue { NO_WRITE_BACK, WRITE_BACK }; class WriteBack { WriteBackValue value_; public: WriteBack(WriteBackValue value) // NOLINT(runtime/explicit) : value_(value) {} explicit WriteBack(int value) : value_((value == 0) ? NO_WRITE_BACK : WRITE_BACK) {} uint32_t GetWriteBackUint32() const { return (value_ == WRITE_BACK) ? 1 : 0; } bool DoesWriteBack() const { return value_ == WRITE_BACK; } }; inline std::ostream& operator<<(std::ostream& os, WriteBack write_back) { if (write_back.DoesWriteBack()) return os << "!"; return os; } class EncodingValue { bool valid_; uint32_t encoding_value_; public: EncodingValue() { valid_ = false; encoding_value_ = 0; } bool IsValid() const { return valid_; } uint32_t GetEncodingValue() const { return encoding_value_; } void SetEncodingValue(uint32_t encoding_value) { valid_ = true; encoding_value_ = encoding_value; } }; class EncodingValueAndImmediate : public EncodingValue { uint32_t encoded_immediate_; public: EncodingValueAndImmediate() { encoded_immediate_ = 0; } uint32_t GetEncodedImmediate() const { return encoded_immediate_; } void SetEncodedImmediate(uint32_t encoded_immediate) { encoded_immediate_ = encoded_immediate; } }; class ImmediateT32 : public EncodingValue { public: explicit ImmediateT32(uint32_t imm); static bool IsImmediateT32(uint32_t imm); static uint32_t Decode(uint32_t value); }; class ImmediateA32 : public EncodingValue { public: explicit ImmediateA32(uint32_t imm); static bool IsImmediateA32(uint32_t imm); static uint32_t Decode(uint32_t value); }; // Return the encoding value of a shift type. uint32_t TypeEncodingValue(Shift shift); // Return the encoding value for a shift amount depending on the shift type. uint32_t AmountEncodingValue(Shift shift, uint32_t amount); enum MemoryBarrierType { OSHLD = 0x1, OSHST = 0x2, OSH = 0x3, NSHLD = 0x5, NSHST = 0x6, NSH = 0x7, ISHLD = 0x9, ISHST = 0xa, ISH = 0xb, LD = 0xd, ST = 0xe, SY = 0xf }; class MemoryBarrier { MemoryBarrierType type_; public: MemoryBarrier(MemoryBarrierType type) // NOLINT(runtime/explicit) : type_(type) {} MemoryBarrier(uint32_t type) // NOLINT(runtime/explicit) : type_(static_cast<MemoryBarrierType>(type)) { VIXL_ASSERT((type & 0x3) != 0); } MemoryBarrierType GetType() const { return type_; } const char* GetName() const; }; inline std::ostream& operator<<(std::ostream& os, MemoryBarrier option) { return os << option.GetName(); } enum InterruptFlagsType { F = 0x1, I = 0x2, IF = 0x3, A = 0x4, AF = 0x5, AI = 0x6, AIF = 0x7 }; class InterruptFlags { InterruptFlagsType type_; public: InterruptFlags(InterruptFlagsType type) // NOLINT(runtime/explicit) : type_(type) {} InterruptFlags(uint32_t type) // NOLINT(runtime/explicit) : type_(static_cast<InterruptFlagsType>(type)) { VIXL_ASSERT(type <= 7); } InterruptFlagsType GetType() const { return type_; } const char* GetName() const; }; inline std::ostream& operator<<(std::ostream& os, InterruptFlags option) { return os << option.GetName(); } enum EndiannessType { LE = 0, BE = 1 }; class Endianness { EndiannessType type_; public: Endianness(EndiannessType type) : type_(type) {} // NOLINT(runtime/explicit) Endianness(uint32_t type) // NOLINT(runtime/explicit) : type_(static_cast<EndiannessType>(type)) { VIXL_ASSERT(type <= 1); } EndiannessType GetType() const { return type_; } const char* GetName() const; }; inline std::ostream& operator<<(std::ostream& os, Endianness endian_specifier) { return os << endian_specifier.GetName(); } enum AlignmentType { k16BitAlign = 0, k32BitAlign = 1, k64BitAlign = 2, k128BitAlign = 3, k256BitAlign = 4, kNoAlignment = 5, kBadAlignment = 6 }; class Alignment { AlignmentType align_; public: Alignment(AlignmentType align) // NOLINT(runtime/explicit) : align_(align) {} Alignment(uint32_t align) // NOLINT(runtime/explicit) : align_(static_cast<AlignmentType>(align)) { VIXL_ASSERT(align <= static_cast<uint32_t>(k256BitAlign)); } AlignmentType GetType() const { return align_; } bool Is(AlignmentType type) { return align_ == type; } }; inline std::ostream& operator<<(std::ostream& os, Alignment align) { if (align.GetType() == kBadAlignment) return os << " :??"; if (align.GetType() == kNoAlignment) return os; return os << " :" << (0x10 << static_cast<uint32_t>(align.GetType())); } // Structure containing information on forward references. struct ReferenceInfo { int size; int min_offset; int max_offset; int alignment; // As a power of two. enum { kAlignPc, kDontAlignPc } pc_needs_aligning; }; } // namespace aarch32 } // namespace vixl #endif // VIXL_AARCH32_INSTRUCTIONS_AARCH32_H_