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_v<E> || std::is_floating_point_v<E>) &&
88 (sizeof(E) * N == 8 || sizeof(E) * N == 16)>> :
99 template <
typename T,
typename Enabled=
void>
102 template <
typename T>
104 (std::is_array_v<T> || std::is_class_v<T> || std::is_union_v<T>) &&
108 !IsAapcs64ShortVectorV<T>
109 >> :
public std::true_type
112 template <
typename T>
124 template <
typename T,
typename Enabled=
void>
127 template <
typename E,
size_t N>
129 typename
std::enable_if_t<std::is_floating_point_v<E> && N <= 4>> :
130 public std::true_type
133 template <
typename T>
134 constexpr
bool IsAapcs64HfaV = IsAapcs64Hfa<T>::value;
140 template <
typename T,
typename Enabled=
void>
141 struct IsAapcs64Hva :
public std::false_type {};
143 template <
typename E,
size_t N>
144 struct IsAapcs64Hva<
E[N],
145 typename
std::enable_if_t<IsAapcs64ShortVectorV<E> && N <= 4>> :
146 public std::true_type
149 template <
typename T>
150 constexpr
bool IsAapcs64HvaV = IsAapcs64Hva<T>::value;
153 template <
typename T,
typename Enabled=
void>
154 struct IsAapcs64Hxa :
public std::false_type {};
156 template <
typename T>
157 struct IsAapcs64Hxa<T, typename
std::enable_if_t<
158 IsAapcs64HfaV<T> || IsAapcs64HvaV<T>>> :
159 public std::true_type
162 template <
typename T>
163 constexpr
bool IsAapcs64HxaV = IsAapcs64Hxa<T>::value;
165 struct Aapcs64ArgumentBase
167 template <
typename T>
169 loadFromStack(ThreadContext *tc, Aapcs64::State &state)
172 size_t align = std::max<size_t>(8,
alignof(T));
174 size_t size =
roundUp(
sizeof(T), 8);
180 ConstVPtr<T>
val(state.nsaa, tc);
195 template <
typename Float>
196 struct Argument<Aapcs64, Float, typename
std::enable_if_t<
197 std::is_floating_point_v<Float> || IsAapcs64ShortVectorV<Float>>> :
198 public Aapcs64ArgumentBase
201 get(ThreadContext *tc, Aapcs64::State &state)
203 if (state.nsrn <= state.MAX_SRN) {
205 return tc->readVecReg(
id).as<Float>()[0];
208 return loadFromStack<Float>(tc, state);
212 template <
typename Float>
213 struct Result<Aapcs64, Float, typename
std::enable_if_t<
214 std::is_floating_point_v<Float> || IsAapcs64ShortVectorV<Float>>>
217 store(ThreadContext *tc,
const Float &
f)
220 auto reg = tc->readVecReg(
id);
221 reg.as<Float>()[0] =
f;
222 tc->setVecReg(
id,
reg);
232 template <
typename Integer>
234 std::is_integral_v<Integer> && (sizeof(Integer) <= 8)>> :
235 public Aapcs64ArgumentBase
246 return loadFromStack<Integer>(tc, state);
250 template <
typename Integer>
252 std::is_integral_v<Integer> && (sizeof(Integer) > 8)>> :
253 public Aapcs64ArgumentBase
258 if (
alignof(Integer) == 16 && (state.
ngrn % 2))
261 if (
sizeof(Integer) == 16 && state.
ngrn + 1 <= state.
MAX_GRN) {
271 return loadFromStack<Integer>(tc, state);
275 template <
typename Integer>
277 std::is_integral_v<Integer> && (sizeof(Integer) <= 8)>>
286 template <
typename Integer>
288 std::is_integral_v<Integer> && (sizeof(Integer) > 8)>>
304 template <
typename T>
307 template <
typename E,
size_t N>
310 template <
typename HA>
312 IsAapcs64HxaV<HA>>> :
public Aapcs64ArgumentBase
318 constexpr
size_t Count =
sizeof(HA) /
sizeof(Elem);
322 for (
int i = 0;
i < Count;
i++)
330 return loadFromStack<HA>(tc, state);
334 template <
typename HA>
341 constexpr
size_t Count =
sizeof(HA) /
sizeof(Elem);
343 for (
int i = 0;
i < Count;
i++)
353 template <
typename Composite>
355 IsAapcs64CompositeV<Composite> && !IsAapcs64HxaV<Composite>>> :
356 public Aapcs64ArgumentBase
361 if (
sizeof(Composite) > 16) {
372 size_t bytes =
sizeof(Composite);
373 using Chunk = uint64_t;
375 const int chunk_size =
sizeof(Chunk);
376 const int regs = (bytes + chunk_size - 1) / chunk_size;
380 alignas(
alignof(Composite)) uint8_t buf[bytes];
381 for (
int i = 0;
i < regs;
i++) {
384 size_t to_copy = std::min<size_t>(bytes, chunk_size);
385 memcpy(buf +
i * chunk_size, &
val, to_copy);
394 return loadFromStack<Composite>(tc, state);
398 template <
typename Composite>
400 IsAapcs64CompositeV<Composite> && !IsAapcs64HxaV<Composite>>>
405 if (
sizeof(Composite) > 16) {
414 size_t bytes =
sizeof(Composite);
415 using Chunk = uint64_t;
417 const int chunk_size =
sizeof(Chunk);
418 const int regs = (bytes + chunk_size - 1) / chunk_size;
421 uint8_t *buf = (uint8_t *)&cp;
422 for (
int i = 0;
i < regs;
i++) {
423 size_t to_copy = std::min<size_t>(bytes, chunk_size);
426 memcpy(&
val, buf, to_copy);
440 #endif // __ARCH_ARM_AAPCS64_HH__