28 #ifndef __ARCH_ARM_AAPCS64_HH__
29 #define __ARCH_ARM_AAPCS64_HH__
33 #include <type_traits>
64 nsaa(tc->readIntReg(ArmISA::INTREG_SPX))
81 template <
typename T,
typename Enabled=
void>
84 template <
typename E,
size_t N>
86 typename
std::enable_if_t<
87 (std::is_integral<E>::value || std::is_floating_point<E>::value) &&
88 (sizeof(E) * N == 8 || sizeof(E) * N == 16)>> :
96 template <
typename T,
typename Enabled=
void>
101 (std::is_array<T>::value ||
102 std::is_class<T>::value ||
103 std::is_union<T>::value) &&
105 !IsVarArgs<T>::value &&
107 !IsAapcs64ShortVector<T>::value
108 >> :
public std::true_type
120 template <
typename T,
typename Enabled=
void>
123 template <
typename E,
size_t N>
125 typename
std::enable_if_t<std::is_floating_point<E>::value && N <= 4>> :
126 public std::true_type
133 template <
typename T,
typename Enabled=
void>
134 struct IsAapcs64Hva :
public std::false_type {};
136 template <
typename E,
size_t N>
137 struct IsAapcs64Hva<
E[N],
138 typename
std::enable_if_t<IsAapcs64ShortVector<E>::value && N <= 4>> :
139 public std::true_type
143 template <
typename T,
typename Enabled=
void>
144 struct IsAapcs64Hxa :
public std::false_type {};
146 template <
typename T>
147 struct IsAapcs64Hxa<T, typename
std::enable_if_t<
148 IsAapcs64Hfa<T>::value || IsAapcs64Hva<T>::value>> :
149 public std::true_type
152 struct Aapcs64ArgumentBase
154 template <
typename T>
156 loadFromStack(ThreadContext *tc, Aapcs64::State &state)
159 size_t align = std::max<size_t>(8,
alignof(T));
161 size_t size =
roundUp(
sizeof(T), 8);
167 ConstVPtr<T>
val(state.nsaa, tc);
182 template <
typename Float>
183 struct Argument<Aapcs64, Float, typename
std::enable_if_t<
184 std::is_floating_point<Float>::value ||
185 IsAapcs64ShortVector<Float>::value>> :
186 public Aapcs64ArgumentBase
189 get(ThreadContext *tc, Aapcs64::State &state)
191 if (state.nsrn <= state.MAX_SRN) {
193 return tc->readVecReg(
id).as<Float>()[0];
196 return loadFromStack<Float>(tc, state);
200 template <
typename Float>
201 struct Result<Aapcs64, Float, typename
std::enable_if_t<
202 std::is_floating_point<Float>::value ||
203 IsAapcs64ShortVector<Float>::value>>
206 store(ThreadContext *tc,
const Float &
f)
209 auto reg = tc->readVecReg(
id);
210 reg.as<Float>()[0] =
f;
211 tc->setVecReg(
id,
reg);
221 template <
typename Integer>
223 std::is_integral<Integer>::value && (sizeof(Integer) <= 8)>> :
224 public Aapcs64ArgumentBase
235 return loadFromStack<Integer>(tc, state);
239 template <
typename Integer>
241 std::is_integral<Integer>::value && (sizeof(Integer) > 8)>> :
242 public Aapcs64ArgumentBase
247 if (
alignof(Integer) == 16 && (state.
ngrn % 2))
250 if (
sizeof(Integer) == 16 && state.
ngrn + 1 <= state.
MAX_GRN) {
260 return loadFromStack<Integer>(tc, state);
264 template <
typename Integer>
266 std::is_integral<Integer>::value && (sizeof(Integer) <= 8)>>
275 template <
typename Integer>
277 std::is_integral<Integer>::value && (sizeof(Integer) > 8)>>
293 template <
typename T>
296 template <
typename E,
size_t N>
299 template <
typename HA>
301 IsAapcs64Hxa<HA>::value>> :
public Aapcs64ArgumentBase
307 constexpr
size_t Count =
sizeof(HA) /
sizeof(Elem);
311 for (
int i = 0;
i < Count;
i++)
319 return loadFromStack<HA>(tc, state);
323 template <
typename HA>
325 typename
std::enable_if_t<IsAapcs64Hxa<HA>::value>>
331 constexpr
size_t Count =
sizeof(HA) /
sizeof(Elem);
333 for (
int i = 0;
i < Count;
i++)
343 template <
typename Composite>
345 IsAapcs64Composite<Composite>::value && !IsAapcs64Hxa<Composite>::value>> :
346 public Aapcs64ArgumentBase
351 if (
sizeof(Composite) > 16) {
362 size_t bytes =
sizeof(Composite);
363 using Chunk = uint64_t;
365 const int chunk_size =
sizeof(Chunk);
366 const int regs = (bytes + chunk_size - 1) / chunk_size;
370 alignas(
alignof(Composite)) uint8_t buf[bytes];
371 for (
int i = 0;
i < regs;
i++) {
374 size_t to_copy = std::min<size_t>(bytes, chunk_size);
375 memcpy(buf +
i * chunk_size, &
val, to_copy);
384 return loadFromStack<Composite>(tc, state);
388 template <
typename Composite>
390 IsAapcs64Composite<Composite>::value && !IsAapcs64Hxa<Composite>::value>>
395 if (
sizeof(Composite) > 16) {
404 size_t bytes =
sizeof(Composite);
405 using Chunk = uint64_t;
407 const int chunk_size =
sizeof(Chunk);
408 const int regs = (bytes + chunk_size - 1) / chunk_size;
411 uint8_t *buf = (uint8_t *)&cp;
412 for (
int i = 0;
i < regs;
i++) {
413 size_t to_copy = std::min<size_t>(bytes, chunk_size);
416 memcpy(&
val, buf, to_copy);
430 #endif // __ARCH_ARM_AAPCS64_HH__