1 /*
2  * Copyright (C) 2014 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #ifndef ART_COMPILER_UTILS_X86_64_MANAGED_REGISTER_X86_64_H_
18 #define ART_COMPILER_UTILS_X86_64_MANAGED_REGISTER_X86_64_H_
19 
20 #include "base/macros.h"
21 #include "constants_x86_64.h"
22 #include "utils/managed_register.h"
23 
24 namespace art HIDDEN {
25 namespace x86_64 {
26 
27 // Values for register pairs.
28 // The registers in kReservedCpuRegistersArray in x86.cc are not used in pairs.
29 // The table kRegisterPairs in x86.cc must be kept in sync with this enum.
30 enum RegisterPair {
31   RAX_RDX = 0,
32   RAX_RCX = 1,
33   RAX_RBX = 2,
34   RAX_RDI = 3,
35   RDX_RCX = 4,
36   RDX_RBX = 5,
37   RDX_RDI = 6,
38   RCX_RBX = 7,
39   RCX_RDI = 8,
40   RBX_RDI = 9,
41   kNumberOfRegisterPairs = 10,
42   kNoRegisterPair = -1,
43 };
44 
45 std::ostream& operator<<(std::ostream& os, const RegisterPair& reg);
46 
47 const int kNumberOfCpuRegIds = kNumberOfCpuRegisters;
48 const int kNumberOfCpuAllocIds = kNumberOfCpuRegisters;
49 
50 const int kNumberOfXmmRegIds = kNumberOfFloatRegisters;
51 const int kNumberOfXmmAllocIds = kNumberOfFloatRegisters;
52 
53 const int kNumberOfX87RegIds = kNumberOfX87Registers;
54 const int kNumberOfX87AllocIds = kNumberOfX87Registers;
55 
56 const int kNumberOfPairRegIds = kNumberOfRegisterPairs;
57 
58 const int kNumberOfRegIds = kNumberOfCpuRegIds + kNumberOfXmmRegIds +
59     kNumberOfX87RegIds + kNumberOfPairRegIds;
60 const int kNumberOfAllocIds = kNumberOfCpuAllocIds + kNumberOfXmmAllocIds +
61     kNumberOfX87RegIds;
62 
63 // Register ids map:
64 //   [0..R[  cpu registers (enum Register)
65 //   [R..X[  xmm registers (enum XmmRegister)
66 //   [X..S[  x87 registers (enum X87Register)
67 //   [S..P[  register pairs (enum RegisterPair)
68 // where
69 //   R = kNumberOfCpuRegIds
70 //   X = R + kNumberOfXmmRegIds
71 //   S = X + kNumberOfX87RegIds
72 //   P = X + kNumberOfRegisterPairs
73 
74 // Allocation ids map:
75 //   [0..R[  cpu registers (enum Register)
76 //   [R..X[  xmm registers (enum XmmRegister)
77 //   [X..S[  x87 registers (enum X87Register)
78 // where
79 //   R = kNumberOfCpuRegIds
80 //   X = R + kNumberOfXmmRegIds
81 //   S = X + kNumberOfX87RegIds
82 
83 
84 // An instance of class 'ManagedRegister' represents a single cpu register (enum
85 // Register), an xmm register (enum XmmRegister), or a pair of cpu registers
86 // (enum RegisterPair).
87 // 'ManagedRegister::NoRegister()' provides an invalid register.
88 // There is a one-to-one mapping between ManagedRegister and register id.
89 class X86_64ManagedRegister : public ManagedRegister {
90  public:
AsCpuRegister()91   constexpr CpuRegister AsCpuRegister() const {
92     CHECK(IsCpuRegister());
93     return CpuRegister(static_cast<Register>(id_));
94   }
95 
AsXmmRegister()96   constexpr XmmRegister AsXmmRegister() const {
97     CHECK(IsXmmRegister());
98     return XmmRegister(static_cast<FloatRegister>(id_ - kNumberOfCpuRegIds));
99   }
100 
AsX87Register()101   constexpr X87Register AsX87Register() const {
102     CHECK(IsX87Register());
103     return static_cast<X87Register>(id_ -
104                                     (kNumberOfCpuRegIds + kNumberOfXmmRegIds));
105   }
106 
AsRegisterPairLow()107   constexpr CpuRegister AsRegisterPairLow() const {
108     CHECK(IsRegisterPair());
109     // Appropriate mapping of register ids allows to use AllocIdLow().
110     return FromRegId(AllocIdLow()).AsCpuRegister();
111   }
112 
AsRegisterPairHigh()113   constexpr CpuRegister AsRegisterPairHigh() const {
114     CHECK(IsRegisterPair());
115     // Appropriate mapping of register ids allows to use AllocIdHigh().
116     return FromRegId(AllocIdHigh()).AsCpuRegister();
117   }
118 
IsCpuRegister()119   constexpr bool IsCpuRegister() const {
120     CHECK(IsValidManagedRegister());
121     return (0 <= id_) && (id_ < kNumberOfCpuRegIds);
122   }
123 
IsXmmRegister()124   constexpr bool IsXmmRegister() const {
125     CHECK(IsValidManagedRegister());
126     const int test = id_ - kNumberOfCpuRegIds;
127     return (0 <= test) && (test < kNumberOfXmmRegIds);
128   }
129 
IsX87Register()130   constexpr bool IsX87Register() const {
131     CHECK(IsValidManagedRegister());
132     const int test = id_ - (kNumberOfCpuRegIds + kNumberOfXmmRegIds);
133     return (0 <= test) && (test < kNumberOfX87RegIds);
134   }
135 
IsRegisterPair()136   constexpr bool IsRegisterPair() const {
137     CHECK(IsValidManagedRegister());
138     const int test = id_ -
139         (kNumberOfCpuRegIds + kNumberOfXmmRegIds + kNumberOfX87RegIds);
140     return (0 <= test) && (test < kNumberOfPairRegIds);
141   }
142 
143   void Print(std::ostream& os) const;
144 
145   // Returns true if the two managed-registers ('this' and 'other') overlap.
146   // Either managed-register may be the NoRegister. If both are the NoRegister
147   // then false is returned.
148   bool Overlaps(const X86_64ManagedRegister& other) const;
149 
FromCpuRegister(Register r)150   static constexpr X86_64ManagedRegister FromCpuRegister(Register r) {
151     CHECK_NE(r, kNoRegister);
152     return FromRegId(r);
153   }
154 
FromXmmRegister(FloatRegister r)155   static constexpr X86_64ManagedRegister FromXmmRegister(FloatRegister r) {
156     return FromRegId(r + kNumberOfCpuRegIds);
157   }
158 
FromX87Register(X87Register r)159   static constexpr X86_64ManagedRegister FromX87Register(X87Register r) {
160     CHECK_NE(r, kNoX87Register);
161     return FromRegId(r + kNumberOfCpuRegIds + kNumberOfXmmRegIds);
162   }
163 
FromRegisterPair(RegisterPair r)164   static constexpr X86_64ManagedRegister FromRegisterPair(RegisterPair r) {
165     CHECK_NE(r, kNoRegisterPair);
166     return FromRegId(r + (kNumberOfCpuRegIds + kNumberOfXmmRegIds +
167                           kNumberOfX87RegIds));
168   }
169 
170  private:
IsValidManagedRegister()171   constexpr bool IsValidManagedRegister() const {
172     return (0 <= id_) && (id_ < kNumberOfRegIds);
173   }
174 
RegId()175   constexpr int RegId() const {
176     CHECK(!IsNoRegister());
177     return id_;
178   }
179 
AllocId()180   int AllocId() const {
181     CHECK(IsValidManagedRegister() && !IsRegisterPair());
182     CHECK_LT(id_, kNumberOfAllocIds);
183     return id_;
184   }
185 
186   int AllocIdLow() const;
187   int AllocIdHigh() const;
188 
189   friend class ManagedRegister;
190 
X86_64ManagedRegister(int reg_id)191   explicit constexpr X86_64ManagedRegister(int reg_id) : ManagedRegister(reg_id) {}
192 
FromRegId(int reg_id)193   static constexpr X86_64ManagedRegister FromRegId(int reg_id) {
194     X86_64ManagedRegister reg(reg_id);
195     CHECK(reg.IsValidManagedRegister());
196     return reg;
197   }
198 };
199 
200 std::ostream& operator<<(std::ostream& os, const X86_64ManagedRegister& reg);
201 
202 }  // namespace x86_64
203 
AsX86_64()204 constexpr x86_64::X86_64ManagedRegister ManagedRegister::AsX86_64() const {
205   x86_64::X86_64ManagedRegister reg(id_);
206   CHECK(reg.IsNoRegister() || reg.IsValidManagedRegister());
207   return reg;
208 }
209 
210 }  // namespace art
211 
212 #endif  // ART_COMPILER_UTILS_X86_64_MANAGED_REGISTER_X86_64_H_
213