28 #ifndef __ARCH_ARM_AAPCS64_HH__ 29 #define __ARCH_ARM_AAPCS64_HH__ 33 #include <type_traits> 75 template <
typename T,
typename Enabled=
void>
78 template <
typename E,
size_t N>
80 typename
std::enable_if<
81 (std::is_integral<E>::value || std::is_floating_point<E>::value) &&
82 (sizeof(E) * N == 8 || sizeof(E) * N == 16)>
::type> :
90 template <
typename T,
typename Enabled=
void>
95 (std::is_array<T>::value ||
96 std::is_class<T>::value ||
97 std::is_union<T>::value) &&
99 !IsVarArgs<T>::value &&
101 !IsAapcs64ShortVector<T>::value
102 >
::type> :
public std::true_type
114 template <
typename T,
typename Enabled=
void>
117 template <
typename E,
size_t N>
119 typename std::enable_if<std::is_floating_point<E>::value &&
120 N <= 4>::type> :
public std::true_type
127 template <
typename T,
typename Enabled=
void>
128 struct IsAapcs64Hva :
public std::false_type {};
130 template <
typename E,
size_t N>
131 struct IsAapcs64Hva<E[N],
132 typename std::enable_if<IsAapcs64ShortVector<E>::value &&
133 N <= 4>::type> :
public std::true_type
137 template <
typename T,
typename Enabled=
void>
138 struct IsAapcs64Hxa :
public std::false_type {};
140 template <
typename T>
141 struct IsAapcs64Hxa<T, typename std::enable_if<
142 IsAapcs64Hfa<T>::value || IsAapcs64Hva<T>::value>
::type> :
143 public std::true_type
146 struct Aapcs64ArgumentBase
148 template <
typename T>
153 size_t align = std::max<size_t>(8,
alignof(T));
155 size_t size =
roundUp(
sizeof(T), 8);
177 template <
typename Float>
178 struct Argument<Aapcs64, Float, typename std::enable_if<
179 std::is_floating_point<Float>::value ||
180 IsAapcs64ShortVector<Float>::value>
::type> :
181 public Aapcs64ArgumentBase
186 if (state.nsrn <= state.MAX_SRN) {
188 return tc->readVecReg(
id).laneView<Float, 0>();
191 return loadFromStack<Float>(tc, state);
195 template <
typename Float>
196 struct Result<Aapcs64, Float, typename std::enable_if<
197 std::is_floating_point<Float>::value ||
198 IsAapcs64ShortVector<Float>::value>
::type>
205 reg.laneView<Float, 0>() = f;
216 template <
typename Integer>
217 struct Argument<Aapcs64, Integer, typename std::enable_if<
218 std::is_integral<Integer>::value && (sizeof(Integer) <= 8)
219 >::type> :
public Aapcs64ArgumentBase
224 if (state.ngrn <= state.MAX_GRN)
228 state.ngrn = state.MAX_GRN + 1;
230 return loadFromStack<Integer>(tc, state);
234 template <
typename Integer>
235 struct Argument<Aapcs64, Integer, typename std::enable_if<
236 std::is_integral<Integer>::value && (sizeof(Integer) > 8)
237 >
::type> :
public Aapcs64ArgumentBase
242 if (
alignof(Integer) == 16 && (state.ngrn % 2))
245 if (
sizeof(Integer) == 16 && state.ngrn + 1 <= state.MAX_GRN) {
253 state.ngrn = state.MAX_GRN + 1;
255 return loadFromStack<Integer>(tc, state);
259 template <
typename Integer>
260 struct Result<Aapcs64, Integer, typename std::enable_if<
261 std::is_integral<Integer>::value && (sizeof(Integer) <= 8)
271 template <
typename Integer>
272 struct Result<Aapcs64, Integer, typename std::enable_if<
273 std::is_integral<Integer>::value && (sizeof(Integer) > 8)
290 template <
typename T>
293 template <
typename E,
size_t N>
296 template <
typename HA>
297 struct Argument<Aapcs64, HA, typename std::enable_if<
298 IsAapcs64Hxa<HA>::value>
::type> :
public Aapcs64ArgumentBase
304 constexpr
size_t Count =
sizeof(HA) /
sizeof(Elem);
306 if (state.nsrn + Count - 1 <= state.MAX_SRN) {
308 for (
int i = 0;
i < Count;
i++)
314 state.nsrn = state.MAX_SRN + 1;
316 return loadFromStack<HA>(tc, state);
320 template <
typename HA>
322 typename std::enable_if<IsAapcs64Hxa<HA>::value>
::type>
328 constexpr
size_t Count =
sizeof(HA) /
sizeof(Elem);
330 for (
int i = 0;
i < Count;
i++)
340 template <
typename Composite>
341 struct Argument<Aapcs64, Composite, typename std::enable_if<
342 IsAapcs64Composite<Composite>::value && !IsAapcs64Hxa<Composite>::value
343 >
::type> :
public Aapcs64ArgumentBase
348 if (
sizeof(Composite) > 16) {
360 size_t bytes =
sizeof(Composite);
361 using Chunk = uint64_t;
363 const int chunk_size =
sizeof(Chunk);
364 const int regs = (bytes + chunk_size - 1) / chunk_size;
367 if (state.ngrn + regs - 1 <= state.MAX_GRN) {
368 alignas(
alignof(Composite)) uint8_t buf[bytes];
369 for (
int i = 0;
i < regs;
i++) {
372 size_t to_copy = std::min<size_t>(bytes, chunk_size);
373 memcpy(buf +
i * chunk_size, &val, to_copy);
380 state.ngrn = state.MAX_GRN;
382 return loadFromStack<Composite>(tc, state);
386 template <
typename Composite>
387 struct Result<Aapcs64, Composite, typename std::enable_if<
388 IsAapcs64Composite<Composite>::value && !IsAapcs64Hxa<Composite>::value
394 if (
sizeof(Composite) > 16) {
403 size_t bytes =
sizeof(Composite);
404 using Chunk = uint64_t;
406 const int chunk_size =
sizeof(Chunk);
407 const int regs = (bytes + chunk_size - 1) / chunk_size;
410 uint8_t *buf = (uint8_t *)&cp;
411 for (
int i = 0;
i < regs;
i++) {
412 size_t to_copy = std::min<size_t>(bytes, chunk_size);
415 memcpy(&val, buf, to_copy);
428 #endif // __ARCH_ARM_AAPCS64_HH__
This file defines buffer classes used to handle pointer arguments in emulated syscalls.
T gtoh(T value, ByteOrder guest_byte_order)
static void store(ThreadContext *tc, const Integer &i)
virtual void setVecReg(const RegId ®, const VecRegContainer &val)=0
virtual RegVal readIntReg(RegIndex reg_idx) const =0
bool copyIn(PortProxy &memproxy)
copy data into simulator space (read from target memory)
virtual PortProxy & getVirtProxy()=0
Overload hash function for BasicBlockRange type.
T roundUp(const T &val, const U &align)
This function is used to align addresses in memory.
TypedBufferArg is a class template; instances of this template represent typed buffers in target user...
ThreadContext is the external interface to all thread state for anything outside of the CPU...
virtual const VecRegContainer & readVecReg(const RegId ®) const =0
State(const ThreadContext *tc)
T htog(T value, ByteOrder guest_byte_order)
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)
static HA store(ThreadContext *tc, const HA &ha)
virtual void setIntReg(RegIndex reg_idx, RegVal val)=0
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
static void store(ThreadContext *tc, const Composite &c)
Register ID: describe an architectural register with its class and index.
static void store(ThreadContext *tc, const Integer &i)
ByteOrder byteOrder(const ThreadContext *tc)