gem5  v21.0.0.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
aapcs64.hh
Go to the documentation of this file.
1 /*
2  * Copyright 2019 Google Inc.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are
6  * met: redistributions of source code must retain the above copyright
7  * notice, this list of conditions and the following disclaimer;
8  * redistributions in binary form must reproduce the above copyright
9  * notice, this list of conditions and the following disclaimer in the
10  * documentation and/or other materials provided with the distribution;
11  * neither the name of the copyright holders nor the names of its
12  * contributors may be used to endorse or promote products derived from
13  * this software without specific prior written permission.
14  *
15  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
16  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
17  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
18  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
19  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
21  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
25  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26  */
27 
28 #ifndef __ARCH_ARM_AAPCS64_HH__
29 #define __ARCH_ARM_AAPCS64_HH__
30 
31 #include <algorithm>
32 #include <array>
33 #include <type_traits>
34 #include <utility>
35 
36 #include "arch/arm/intregs.hh"
37 #include "arch/arm/utility.hh"
38 #include "base/intmath.hh"
39 #include "cpu/thread_context.hh"
40 #include "sim/guest_abi.hh"
41 #include "sim/proxy_ptr.hh"
42 
43 class ThreadContext;
44 
45 struct Aapcs64
46 {
47  using UintPtr = uint64_t;
48 
49  struct State
50  {
51  int ngrn=0; // Next general purpose register number.
52  int nsrn=0; // Next SIMD and floating point register number.
53  Addr nsaa; // Next stacked argument address.
54 
55  // The maximum allowed general purpose register number.
56  static const int MAX_GRN = 7;
57  // The maximum allowed SIMD and floating point register number.
58  static const int MAX_SRN = 7;
59 
60  explicit State(const ThreadContext *tc) :
61  nsaa(tc->readIntReg(ArmISA::INTREG_SPX))
62  {}
63  };
64 };
65 
66 namespace GuestABI
67 {
68 
69 /*
70  * Short Vectors
71  */
72 
73 // A short vector is a machine type that is composed of repeated instances of
74 // one fundamental integral or floating- point type. It may be 8 or 16 bytes
75 // in total size.
76 
77 template <typename T, typename Enabled=void>
78 struct IsAapcs64ShortVector : public std::false_type {};
79 
80 template <typename E, size_t N>
82  typename std::enable_if_t<
83  (std::is_integral<E>::value || std::is_floating_point<E>::value) &&
84  (sizeof(E) * N == 8 || sizeof(E) * N == 16)>> :
85  public std::true_type
86 {};
87 
88 /*
89  * Composite Types
90  */
91 
92 template <typename T, typename Enabled=void>
93 struct IsAapcs64Composite : public std::false_type {};
94 
95 template <typename T>
96 struct IsAapcs64Composite<T, typename std::enable_if_t<
97  (std::is_array<T>::value ||
98  std::is_class<T>::value ||
99  std::is_union<T>::value) &&
100  // VarArgs is technically a composite type, but it's not a normal argument.
101  !IsVarArgs<T>::value &&
102  // Short vectors are also composite types, but don't treat them as one.
103  !IsAapcs64ShortVector<T>::value
104  >> : public std::true_type
105 {};
106 
107 // Homogeneous Aggregates
108 // These *should* be any aggregate type which has only one type of member, but
109 // we can't actually detect that or manipulate that with templates. Instead,
110 // we approximate that by detecting only arrays with that property.
111 
112 // An Homogeneous Floating-Point Aggregate (HFA) is an Homogeneous Aggregate
113 // with a Fundemental Data Type that is a Floating-Point type and at most four
114 // uniquely addressable members.
115 
116 template <typename T, typename Enabled=void>
117 struct IsAapcs64Hfa : public std::false_type {};
118 
119 template <typename E, size_t N>
120 struct IsAapcs64Hfa<E[N],
121  typename std::enable_if_t<std::is_floating_point<E>::value && N <= 4>> :
122  public std::true_type
123 {};
124 
125 // An Homogeneous Short-Vector Aggregate (HVA) is an Homogeneous Aggregate with
126 // a Fundamental Data Type that is a Short-Vector type and at most four
127 // uniquely addressable members.
128 
129 template <typename T, typename Enabled=void>
130 struct IsAapcs64Hva : public std::false_type {};
131 
132 template <typename E, size_t N>
133 struct IsAapcs64Hva<E[N],
134  typename std::enable_if_t<IsAapcs64ShortVector<E>::value && N <= 4>> :
135  public std::true_type
136 {};
137 
138 // A shorthand to test if a type is an HVA or an HFA.
139 template <typename T, typename Enabled=void>
140 struct IsAapcs64Hxa : public std::false_type {};
141 
142 template <typename T>
143 struct IsAapcs64Hxa<T, typename std::enable_if_t<
144  IsAapcs64Hfa<T>::value || IsAapcs64Hva<T>::value>> :
145  public std::true_type
146 {};
147 
148 struct Aapcs64ArgumentBase
149 {
150  template <typename T>
151  static T
152  loadFromStack(ThreadContext *tc, Aapcs64::State &state)
153  {
154  // The alignment is the larger of 8 or the natural alignment of T.
155  size_t align = std::max<size_t>(8, alignof(T));
156  // Increase the size to the next multiple of 8.
157  size_t size = roundUp(sizeof(T), 8);
158 
159  // Align the stack.
160  state.nsaa = roundUp(state.nsaa, align);
161 
162  // Extract the value from it.
163  ConstVPtr<T> val(state.nsaa, tc);
164 
165  // Move the nsaa past this argument.
166  state.nsaa += size;
167 
168  // Return the value we extracted.
169  return gtoh(*val, ArmISA::byteOrder(tc));
170  }
171 };
172 
173 
174 /*
175  * Floating point and Short-Vector arguments and return values.
176  */
177 
178 template <typename Float>
179 struct Argument<Aapcs64, Float, typename std::enable_if_t<
180  std::is_floating_point<Float>::value ||
181  IsAapcs64ShortVector<Float>::value>> :
182  public Aapcs64ArgumentBase
183 {
184  static Float
185  get(ThreadContext *tc, Aapcs64::State &state)
186  {
187  if (state.nsrn <= state.MAX_SRN) {
188  RegId id(VecRegClass, state.nsrn++);
189  return tc->readVecReg(id).laneView<Float, 0>();
190  }
191 
192  return loadFromStack<Float>(tc, state);
193  }
194 };
195 
196 template <typename Float>
197 struct Result<Aapcs64, Float, typename std::enable_if_t<
198  std::is_floating_point<Float>::value ||
199  IsAapcs64ShortVector<Float>::value>>
200 {
201  static void
202  store(ThreadContext *tc, const Float &f)
203  {
204  RegId id(VecRegClass, 0);
205  auto reg = tc->readVecReg(id);
206  reg.laneView<Float, 0>() = f;
207  tc->setVecReg(id, reg);
208  }
209 };
210 
211 
212 /*
213  * Integer arguments and return values.
214  */
215 
216 // This will pick up Addr as well, which should be used for guest pointers.
217 template <typename Integer>
218 struct Argument<Aapcs64, Integer, typename std::enable_if_t<
219  std::is_integral<Integer>::value && (sizeof(Integer) <= 8)>> :
220  public Aapcs64ArgumentBase
221 {
222  static Integer
224  {
225  if (state.ngrn <= state.MAX_GRN)
226  return tc->readIntReg(state.ngrn++);
227 
228  // Max out ngrn since we've effectively saturated it.
229  state.ngrn = state.MAX_GRN + 1;
230 
231  return loadFromStack<Integer>(tc, state);
232  }
233 };
234 
235 template <typename Integer>
236 struct Argument<Aapcs64, Integer, typename std::enable_if_t<
237  std::is_integral<Integer>::value && (sizeof(Integer) > 8)>> :
238  public Aapcs64ArgumentBase
239 {
240  static Integer
241  get(ThreadContext *tc, Aapcs64::State &state)
242  {
243  if (alignof(Integer) == 16 && (state.ngrn % 2))
244  state.ngrn++;
245 
246  if (sizeof(Integer) == 16 && state.ngrn + 1 <= state.MAX_GRN) {
247  Integer low = tc->readIntReg(state.ngrn++);
248  Integer high = tc->readIntReg(state.ngrn++);
249  high = high << 64;
250  return high | low;
251  }
252 
253  // Max out ngrn since we've effectively saturated it.
254  state.ngrn = state.MAX_GRN + 1;
255 
256  return loadFromStack<Integer>(tc, state);
257  }
258 };
259 
260 template <typename Integer>
261 struct Result<Aapcs64, Integer, typename std::enable_if_t<
262  std::is_integral<Integer>::value && (sizeof(Integer) <= 8)>>
263 {
264  static void
265  store(ThreadContext *tc, const Integer &i)
266  {
267  tc->setIntReg(0, i);
268  }
269 };
270 
271 template <typename Integer>
272 struct Result<Aapcs64, Integer, typename std::enable_if_t<
273  std::is_integral<Integer>::value && (sizeof(Integer) > 8)>>
274 {
275  static void
276  store(ThreadContext *tc, const Integer &i)
277  {
278  tc->setIntReg(0, (uint64_t)i);
279  tc->setIntReg(1, (uint64_t)(i >> 64));
280  }
281 };
282 
283 
284 /*
285  * Homogeneous Floating-Point and Short-Vector Aggregates (HFAs and HVAs)
286  * argument and return values.
287  */
288 
289 template <typename T>
290 struct Aapcs64ArrayType { using Type = void; };
291 
292 template <typename E, size_t N>
293 struct Aapcs64ArrayType<E[N]> { using Type = E; };
294 
295 template <typename HA>
296 struct Argument<Aapcs64, HA, typename std::enable_if_t<
297  IsAapcs64Hxa<HA>::value>> : public Aapcs64ArgumentBase
298 {
299  static HA
301  {
302  using Elem = typename Aapcs64ArrayType<HA>::Type;
303  constexpr size_t Count = sizeof(HA) / sizeof(Elem);
304 
305  if (state.nsrn + Count - 1 <= state.MAX_SRN) {
306  HA ha;
307  for (int i = 0; i < Count; i++)
308  ha[i] = Argument<Aapcs64, Elem>::get(tc, state);
309  return ha;
310  }
311 
312  // Max out the nsrn since we effectively exhausted it.
313  state.nsrn = state.MAX_SRN + 1;
314 
315  return loadFromStack<HA>(tc, state);
316  }
317 };
318 
319 template <typename HA>
320 struct Result<Aapcs64, HA,
321  typename std::enable_if_t<IsAapcs64Hxa<HA>::value>>
322 {
323  static HA
324  store(ThreadContext *tc, const HA &ha)
325  {
326  using Elem = typename Aapcs64ArrayType<HA>::Type;
327  constexpr size_t Count = sizeof(HA) / sizeof(Elem);
328 
329  for (int i = 0; i < Count; i++)
331  }
332 };
333 
334 
335 /*
336  * Composite arguments and return values which are not HVAs or HFAs.
337  */
338 
339 template <typename Composite>
340 struct Argument<Aapcs64, Composite, typename std::enable_if_t<
341  IsAapcs64Composite<Composite>::value && !IsAapcs64Hxa<Composite>::value>> :
342  public Aapcs64ArgumentBase
343 {
344  static Composite
346  {
347  if (sizeof(Composite) > 16) {
348  // Composite values larger than 16 which aren't HFAs or HVAs are
349  // kept in a buffer, and the argument is actually a pointer to that
350  // buffer.
352  ConstVPtr<Composite> composite(addr, tc);
353  return gtoh(*composite, ArmISA::byteOrder(tc));
354  }
355 
356  // The size of Composite must be 16 bytes or less after this point.
357 
358  size_t bytes = sizeof(Composite);
359  using Chunk = uint64_t;
360 
361  const int chunk_size = sizeof(Chunk);
362  const int regs = (bytes + chunk_size - 1) / chunk_size;
363 
364  // Can it fit in GPRs?
365  if (state.ngrn + regs - 1 <= state.MAX_GRN) {
366  alignas(alignof(Composite)) uint8_t buf[bytes];
367  for (int i = 0; i < regs; i++) {
368  Chunk val = tc->readIntReg(state.ngrn++);
369  val = htog(val, ArmISA::byteOrder(tc));
370  size_t to_copy = std::min<size_t>(bytes, chunk_size);
371  memcpy(buf + i * chunk_size, &val, to_copy);
372  bytes -= to_copy;
373  }
374  return gtoh(*(Composite *)buf, ArmISA::byteOrder(tc));
375  }
376 
377  // Max out the ngrn since we effectively exhausted it.
378  state.ngrn = state.MAX_GRN;
379 
380  return loadFromStack<Composite>(tc, state);
381  }
382 };
383 
384 template <typename Composite>
385 struct Result<Aapcs64, Composite, typename std::enable_if_t<
386  IsAapcs64Composite<Composite>::value && !IsAapcs64Hxa<Composite>::value>>
387 {
388  static void
389  store(ThreadContext *tc, const Composite &c)
390  {
391  if (sizeof(Composite) > 16) {
393  VPtr<Composite> composite(addr, tc);
394  *composite = htog(c, ArmISA::byteOrder(tc));
395  return;
396  }
397 
398  // The size of Composite must be 16 bytes or less after this point.
399 
400  size_t bytes = sizeof(Composite);
401  using Chunk = uint64_t;
402 
403  const int chunk_size = sizeof(Chunk);
404  const int regs = (bytes + chunk_size - 1) / chunk_size;
405 
406  Composite cp = htog(c, ArmISA::byteOrder(tc));
407  uint8_t *buf = (uint8_t *)&cp;
408  for (int i = 0; i < regs; i++) {
409  size_t to_copy = std::min<size_t>(bytes, chunk_size);
410 
411  Chunk val;
412  memcpy(&val, buf, to_copy);
413  val = gtoh(val, ArmISA::byteOrder(tc));
414 
415  tc->setIntReg(i, val);
416 
417  bytes -= to_copy;
418  buf += to_copy;
419  }
420  }
421 };
422 
423 } // namespace GuestABI
424 
425 #endif // __ARCH_ARM_AAPCS64_HH__
GuestABI::IsAapcs64Composite
Definition: aapcs64.hh:93
ArmISA::byteOrder
ByteOrder byteOrder(const ThreadContext *tc)
Definition: utility.hh:430
ArmISA::i
Bitfield< 7 > i
Definition: miscregs_types.hh:63
ConstProxyPtr
Definition: proxy_ptr.hh:106
ProxyPtr
Definition: proxy_ptr.hh:235
ThreadContext::setIntReg
virtual void setIntReg(RegIndex reg_idx, RegVal val)=0
X86ISA::E
Bitfield< 31, 0 > E
Definition: int.hh:51
GuestABI::Aapcs64ArrayType
Definition: aapcs64.hh:290
proxy_ptr.hh
ThreadContext::readVecReg
virtual const TheISA::VecRegContainer & readVecReg(const RegId &reg) const =0
GuestABI::Aapcs64ArrayType::Type
void Type
Definition: aapcs64.hh:290
GuestABI::enable_if_t< std::is_integral< Integer >::value &&(sizeof(Integer)<=8)> >::get
static Integer get(ThreadContext *tc, Aapcs64::State &state)
Definition: aapcs64.hh:223
ArmISA
Definition: ccregs.hh:41
GuestABI::enable_if_t< std::is_integral< Integer >::value &&(sizeof(Integer)<=8)> >::store
static void store(ThreadContext *tc, const Integer &i)
Definition: aapcs64.hh:265
X86ISA::reg
Bitfield< 5, 3 > reg
Definition: types.hh:88
GuestABI::IsAapcs64ShortVector
Definition: aapcs64.hh:78
sc_dt::align
void align(const scfx_rep &lhs, const scfx_rep &rhs, int &new_wp, int &len_mant, scfx_mant_ref &lhs_mant, scfx_mant_ref &rhs_mant)
Definition: scfx_rep.cc:2083
RegId
Register ID: describe an architectural register with its class and index.
Definition: reg_class.hh:75
ArmISA::INTREG_SPX
@ INTREG_SPX
Definition: intregs.hh:160
cp
Definition: cprintf.cc:37
ThreadContext
ThreadContext is the external interface to all thread state for anything outside of the CPU.
Definition: thread_context.hh:88
GuestABI::Argument< Aapcs64, HA, typename std::enable_if_t< IsAapcs64Hxa< HA >::value > >::get
static HA get(ThreadContext *tc, Aapcs64::State &state)
Definition: aapcs64.hh:300
GuestABI
Definition: aapcs32.hh:66
Aapcs64
Definition: aapcs64.hh:45
htog
T htog(T value, ByteOrder guest_byte_order)
Definition: byteswap.hh:156
GuestABI::Argument
Definition: definition.hh:93
intregs.hh
Aapcs64::State::nsrn
int nsrn
Definition: aapcs64.hh:52
GuestABI::Result< Aapcs64, Composite, typename std::enable_if_t< IsAapcs64Composite< Composite >::value &&!IsAapcs64Hxa< Composite >::value > >::store
static void store(ThreadContext *tc, const Composite &c)
Definition: aapcs64.hh:389
ThreadContext::setVecReg
virtual void setVecReg(const RegId &reg, const TheISA::VecRegContainer &val)=0
Aapcs64::UintPtr
uint64_t UintPtr
Definition: aapcs64.hh:47
X86ISA::val
Bitfield< 63 > val
Definition: misc.hh:769
Addr
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:148
Aapcs64::State::State
State(const ThreadContext *tc)
Definition: aapcs64.hh:60
Aapcs64::State::MAX_SRN
static const int MAX_SRN
Definition: aapcs64.hh:58
utility.hh
Stats::Result
double Result
All results are doubles.
Definition: types.hh:50
X86ISA::addr
Bitfield< 3 > addr
Definition: types.hh:80
Aapcs64::State::ngrn
int ngrn
Definition: aapcs64.hh:51
VecRegClass
@ VecRegClass
Vector Register.
Definition: reg_class.hh:56
ArmISA::ha
Bitfield< 39 > ha
Definition: miscregs_types.hh:538
GuestABI::Result
Definition: definition.hh:58
Aapcs64::State::nsaa
Addr nsaa
Definition: aapcs64.hh:53
std
Overload hash function for BasicBlockRange type.
Definition: vec_reg.hh:587
roundUp
T roundUp(const T &val, const U &align)
This function is used to align addresses in memory.
Definition: intmath.hh:131
guest_abi.hh
GuestABI::Result< Aapcs64, HA, typename std::enable_if_t< IsAapcs64Hxa< HA >::value > >::store
static HA store(ThreadContext *tc, const HA &ha)
Definition: aapcs64.hh:324
ArmISA::INTREG_X8
@ INTREG_X8
Definition: intregs.hh:135
Aapcs64::State::MAX_GRN
static const int MAX_GRN
Definition: aapcs64.hh:56
ArmISA::c
Bitfield< 29 > c
Definition: miscregs_types.hh:50
gtoh
T gtoh(T value, ByteOrder guest_byte_order)
Definition: byteswap.hh:163
Aapcs64::State
Definition: aapcs64.hh:49
intmath.hh
ThreadContext::readIntReg
virtual RegVal readIntReg(RegIndex reg_idx) const =0
GuestABI::IsAapcs64Hfa
Definition: aapcs64.hh:117
thread_context.hh
GuestABI::Argument< Aapcs64, Composite, typename std::enable_if_t< IsAapcs64Composite< Composite >::value &&!IsAapcs64Hxa< Composite >::value > >::get
static Composite get(ThreadContext *tc, Aapcs64::State &state)
Definition: aapcs64.hh:345
GuestABI::Aapcs64ArrayType< E[N]>::Type
E Type
Definition: aapcs64.hh:293
ArmISA::id
Bitfield< 33 > id
Definition: miscregs_types.hh:247
ArmISA::f
Bitfield< 6 > f
Definition: miscregs_types.hh:64

Generated on Tue Mar 23 2021 19:41:18 for gem5 by doxygen 1.8.17