28 #ifndef __ARCH_ARM_AAPCS32_HH__ 29 #define __ARCH_ARM_AAPCS32_HH__ 33 #include <type_traits> 72 template <
typename T,
typename Enabled=
void>
77 (std::is_array<T>::value ||
78 std::is_class<T>::value ||
79 std::is_union<T>::value) &&
82 >
::type> :
public std::true_type
90 template <
typename T, std::
size_t count,
typename Enabled=
void>
96 template <
typename E,
size_t N>
101 template <
typename T>
108 size_t align = std::max<size_t>(4,
alignof(T));
110 size_t size =
roundUp(
sizeof(T), 4);
132 template <
typename Integer>
134 std::is_integral<Integer>::value && (sizeof(Integer) < sizeof(uint32_t))
140 uint32_t
val = std::is_signed<Integer>::value ?
141 sext<sizeof(Integer) * 8>(
i) : i;
146 template <
typename Integer>
148 std::is_integral<Integer>::value && (sizeof(Integer) == sizeof(uint32_t))
158 template <
typename Integer>
160 std::is_integral<Integer>::value && (sizeof(Integer) == sizeof(uint64_t))
166 if (std::is_same<Integer, Addr>::value) {
178 template <
typename Integer>
180 std::is_integral<Integer>::value && (sizeof(Integer) <= sizeof(uint32_t))
186 if (state.ncrn <= state.MAX_CRN) {
191 state.ncrn = state.MAX_CRN + 1;
193 return loadFromStack<Integer>(tc, state);
197 template <
typename Integer>
199 std::is_integral<Integer>::value && (sizeof(Integer) > sizeof(uint32_t))
205 if (std::is_same<Integer, Addr>::value &&
206 state.ncrn <= state.MAX_CRN) {
210 if (
alignof(Integer) == 8 && (state.ncrn % 2))
213 if (
sizeof(Integer) ==
sizeof(uint64_t) &&
214 state.ncrn + 1 <= state.MAX_CRN) {
223 return low | (high << 32);
227 state.ncrn = state.MAX_CRN + 1;
229 return loadFromStack<Integer>(tc, state);
238 template <
typename Float>
240 std::is_floating_point<Float>::value>
::type>
246 storeResult<Aapcs32, decltype(i)>(tc,
i, state);
250 template <
typename Float>
257 if (
sizeof(Float) ==
sizeof(uint32_t)) {
259 getArgument<Aapcs32, uint32_t>(tc, state));
262 getArgument<Aapcs32, uint64_t>(tc, state));
272 template <
typename Composite>
274 IsAapcs32Composite<Composite>::value>
::type>
280 if (
sizeof(Composite) <=
sizeof(uint32_t)) {
283 memcpy((
void *)&val, (
void *)&cp,
sizeof(Composite));
296 if (
sizeof(Composite) >
sizeof(uint32_t))
301 template <
typename Composite>
303 IsAapcs32Composite<Composite>::value>
::type> :
309 size_t bytes =
sizeof(Composite);
310 using Chunk = uint32_t;
312 const int chunk_size =
sizeof(Chunk);
313 const int regs = (bytes + chunk_size - 1) / chunk_size;
315 if (bytes <= chunk_size) {
316 if (state.ncrn++ <= state.MAX_CRN) {
317 alignas(
alignof(Composite)) uint32_t
val =
324 if (
alignof(Composite) == 8 && (state.ncrn % 2))
327 if (state.ncrn + regs - 1 <= state.MAX_CRN) {
328 alignas(
alignof(Composite)) uint8_t buf[bytes];
329 for (
int i = 0;
i < regs;
i++) {
332 size_t to_copy = std::min<size_t>(bytes, chunk_size);
333 memcpy(buf +
i * chunk_size, &val, to_copy);
339 if (!state.stackUsed && state.ncrn <= state.MAX_CRN) {
340 alignas(
alignof(Composite)) uint8_t buf[bytes];
343 while (state.ncrn <= state.MAX_CRN) {
346 size_t to_copy = std::min<size_t>(bytes, chunk_size);
347 memcpy(buf + offset, &val, to_copy);
355 state.stackUsed =
true;
356 state.nsaa +=
roundUp(bytes, 4);
357 state.ncrn = state.MAX_CRN + 1;
363 state.ncrn = state.MAX_CRN + 1;
365 return loadFromStack<Composite>(tc, state);
384 std::array<bool, 16>
s;
385 std::array<bool, 8>
d;
397 for (
int i = 0;
i <= s.size() -
count;
i++) {
402 if (i - last + 1 == count) {
405 d[(last +
j) / 2] =
true;
419 for (
int i = 0;
i <= d.size() -
count;
i++) {
424 if (i - last + 1 == count) {
427 s[(last +
j) * 2] =
true;
428 s[(last +
j) * 2 + 1] =
true;
447 template <
typename Integer>
449 std::is_integral<Integer>::value>
::type> :
public Result<Aapcs32, Integer>
452 template <
typename Integer>
454 std::is_integral<Integer>::value>
::type> :
463 template <
typename Float>
465 std::is_floating_point<Float>::value>
::type>
471 storeResult<Aapcs32, Float>(tc,
f, state);
477 reg.laneView<Float, 0>() = f;
482 template <
typename Float>
490 return getArgument<Aapcs32, Float>(tc, state);
492 const int index = state.allocate(Float{}, 1);
495 constexpr
int lane_per_reg = 16 /
sizeof(Float);
496 const int reg = index / lane_per_reg;
497 const int lane = index % lane_per_reg;
500 auto val = tc->readVecReg(
id);
501 return val.laneView<Float>(lane);
504 return loadFromStack<Float>(tc, state);
513 template <
typename Composite>
515 IsAapcs32Composite<Composite>::value &&
516 !IsAapcs32HomogeneousAggregate<Composite>::value>
::type> :
517 public Result<Aapcs32, Composite>
520 template <
typename Composite>
522 IsAapcs32Composite<Composite>::value &&
523 !IsAapcs32HomogeneousAggregate<Composite>::value>
::type> :
532 template <
typename T>
535 template <
typename E,
size_t N>
538 template <
typename HA>
540 IsAapcs32HomogeneousAggregate<HA>::value>
::type> :
547 constexpr
size_t Count =
sizeof(HA) /
sizeof(Elem);
548 return state.
variadic || !std::is_floating_point<Elem>::value ||
556 constexpr
size_t Count =
sizeof(HA) /
sizeof(Elem);
558 if (useBaseABI(state))
559 return getArgument<Aapcs32, HA>(tc, state);
561 const int base = state.allocate(Elem{}, Count);
563 constexpr
int lane_per_reg = 16 /
sizeof(Elem);
565 for (
int i = 0;
i < Count;
i++) {
566 const int index = base +
i;
567 const int reg = index / lane_per_reg;
568 const int lane = index % lane_per_reg;
572 ha[
i] =
val.laneView<Elem>(lane);
577 return loadFromStack<HA>(tc, state);
583 if (useBaseABI(state))
588 template <
typename HA>
590 typename std::enable_if<IsAapcs32HomogeneousAggregate<HA>::value>
::type>
596 constexpr
size_t Count =
sizeof(HA) /
sizeof(Elem);
597 return state.
variadic || !std::is_floating_point<Elem>::value ||
605 constexpr
size_t Count =
sizeof(HA) /
sizeof(Elem);
607 if (useBaseABI(state)) {
608 storeResult<Aapcs32, HA>(tc,
ha, state);
612 constexpr
int lane_per_reg = 16 /
sizeof(Elem);
613 for (
int i = 0;
i < Count;
i++) {
614 const int reg =
i / lane_per_reg;
615 const int lane =
i % lane_per_reg;
619 val.laneView<Elem>(lane) = ha[
i];
627 if (useBaseABI(state))
637 template <
typename ...Types>
643 state.variadic =
true;
650 #endif // __ARCH_ARM_AAPCS32_HH__
This file defines buffer classes used to handle pointer arguments in emulated syscalls.
T gtoh(T value, ByteOrder guest_byte_order)
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)
static void store(ThreadContext *tc, const Integer &i)
virtual PortProxy & getVirtProxy()=0
static bool useBaseABI(Aapcs32Vfp::State &state)
Overload hash function for BasicBlockRange type.
static float bitsToFloat32(uint32_t val)
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)
static double bitsToFloat64(uint64_t val)
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)
uint64_t getArgument(ThreadContext *tc, int &number, uint16_t size, bool fp)
static void store(ThreadContext *tc, const Float &f, Aapcs32::State &state)
State(const ThreadContext *tc)
static void store(ThreadContext *tc, const Integer &i)
T[count] Aapcs32HomogeneousAggregate
static void store(ThreadContext *tc, const Float &f, Aapcs32Vfp::State &state)
int allocate(float, int count)
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 bool useBaseABI(Aapcs32Vfp::State &state)
void readBlob(Addr addr, void *p, int size) const
Higher level interfaces based on the above.
static void store(ThreadContext *tc, const Integer &i)
static void prepare(ThreadContext *tc, Aapcs32Vfp::State &state)
bool copyOut(PortProxy &memproxy)
copy data out of simulator space (write to target memory)
static void store(ThreadContext *tc, const Composite &composite, Aapcs32::State &state)
static HA store(ThreadContext *tc, const HA &ha, Aapcs32Vfp::State &state)
static void prepare(ThreadContext *tc, Aapcs32::State &state)
Register ID: describe an architectural register with its class and index.
static uint64_t floatToBits(double val)
static T loadFromStack(ThreadContext *tc, Aapcs32::State &state)
int allocate(double, int count)
static void prepare(ThreadContext *tc, Aapcs32Vfp::State &state)
ByteOrder byteOrder(const ThreadContext *tc)