38 #ifndef __ARCH_ARM_SEMIHOSTING_HH__ 39 #define __ARCH_ARM_SEMIHOSTING_HH__ 55 struct ArmSemihostingParams;
80 template <
typename Arg>
105 argPointer +=
sizeof(Arg);
115 argPointer +=
sizeof(Arg);
168 panic(
"Unexpected semihosting argument size %d.", size);
181 panic(
"Unexpected semihosting argument size %d.", size);
263 : parent(_parent), _name(name),
mode(_mode) {}
269 static std::unique_ptr<FileBase> create(
272 static std::unique_ptr<FileBase> create(
296 virtual int64_t
open() {
return 0; }
303 virtual int64_t
close() {
return 0; }
310 virtual bool isTTY()
const {
return false; }
317 virtual int64_t read(uint8_t *buffer, uint64_t size);
324 virtual int64_t write(
const uint8_t *buffer, uint64_t size);
332 virtual int64_t seek(uint64_t pos);
339 virtual int64_t flen();
354 const char *
name,
const char *
mode);
359 int64_t read(uint8_t *buffer, uint64_t size)
override;
360 int64_t seek(uint64_t pos)
override;
375 int64_t
open()
override {
return openImpl(
false); }
376 int64_t close()
override;
377 bool isTTY()
const override;
378 int64_t read(uint8_t *buffer, uint64_t size)
override;
379 int64_t write(
const uint8_t *buffer, uint64_t size)
override;
380 int64_t seek(uint64_t pos)
override;
381 int64_t flen()
override;
384 int64_t openImpl(
bool unserialize);
402 return msb > 31 ? msb - 31 : 0;
409 void semiExit(uint64_t code, uint64_t subcode);
441 template <
typename ...Args>
447 template <
typename ...Args>
452 return (sh->*impl)(tc, args...);
459 std::function<RetErrno(ArmSemihosting *sh, ThreadContext *tc)>;
460 using Dumper = std::function<std::string(ThreadContext *tc)>;
471 template <
typename Abi,
typename ...Args>
477 auto wrapper = wrapImpl(sh, impl);
478 return invokeSimcall<Abi>(tc, wrapper);
483 template <
typename Abi,
typename ...Args>
494 template <
typename ...Args>
496 name(_name), call32(buildDispatcher<
Abi32>(common)),
497 call64(buildDispatcher<
Abi64>(common)),
498 dump32(buildDumper<
Abi32>(_name, common)),
499 dump64(buildDumper<
Abi64>(_name, common))
503 template <
typename ...Args32,
typename ...Args64>
506 name(_name), call32(buildDispatcher<
Abi32>(impl32)),
507 call64(buildDispatcher<
Abi64>(impl64)),
508 dump32(buildDumper<
Abi32>(_name, impl32)),
509 dump64(buildDumper<
Abi64>(_name, impl64))
514 int fmode,
size_t name_size);
519 Addr buffer,
size_t size);
521 Addr buffer,
size_t size);
528 uint64_t
id,
size_t size);
531 Addr to_addr,
size_t to_size);
540 Addr &stack_base,
Addr &stack_limit);
555 template <
typename Abi>
560 std::function<RetErrno(ThreadContext *tc)> retErr =
562 invokeSimcall<Abi>(tc, retErr);
565 static FILE *
getSTDIO(
const char *stream_name,
566 const std::string &
name,
const char *
mode);
568 static const std::map<uint32_t, SemiCall>
calls;
570 static const std::map<uint64_t, const char *>
exitCodes;
572 static const std::map<const std::string, FILE *>
stdioMap;
581 template <
typename Arg>
583 typename std::enable_if<std::is_integral<Arg>::value>
::type>
588 return state.get(tc);
592 template <
typename Arg>
594 typename std::enable_if<std::is_integral<Arg>::value>
::type>
599 if (std::is_signed<Arg>::value)
600 return sext<32>(state.get(tc));
602 return state.get(tc);
606 template <
typename Abi>
608 std::is_base_of<ArmSemihosting::AbiBase, Abi>::value>
::type>
614 state.getAddr(),
sizeof(
typename Abi::State::ArgType));
640 #endif // __ARCH_ARM_SEMIHOSTING_HH__ #define panic(...)
This implements a cprintf based panic() function.
std::string dumpSimcall(std::string name, ThreadContext *tc, std::function< Ret(ThreadContext *, Args...)> target=std::function< Ret(ThreadContext *, Args...)>())
void write(ThreadContext *tc, uint64_t val, ByteOrder endian)
Implementation of the ':semihosting-features' magic file.
RetErrno callHeapInfo64(ThreadContext *tc, Addr block_addr)
RetErrno callSeek(ThreadContext *tc, Handle handle, uint64_t pos)
State(const ThreadContext *tc)
static Dispatcher buildDispatcher(Implementation< Args... > impl)
void serialize(CheckpointOut &cp) const override
Serialize an object.
static FILE * getSTDIO(const char *stream_name, const std::string &name, const char *mode)
RetErrno callErrno(ThreadContext *tc)
ArmSemihosting(const ArmSemihostingParams *p)
std::vector< std::unique_ptr< FileBase > > files
static const std::map< const std::string, FILE * > stdioMap
RetErrno callTime(ThreadContext *tc)
RetErrno callClose(ThreadContext *tc, Handle handle)
void unserialize(CheckpointIn &cp) override
Unserialize an object.
static RetErrno retError(SemiErrno e)
bool call64(ThreadContext *tc, bool gem5_ops)
Perform an Arm Semihosting call from aarch64 code.
RetErrno callGem5PseudoOp32(ThreadContext *tc, uint32_t encoded_func)
T read(Addr address) const
Read sizeof(T) bytes from address and return as object T.
StateBase(const ThreadContext *tc, Addr arg_pointer)
SemiCall(const char *_name, Implementation< Args32... > impl32, Implementation< Args64... > impl64)
const char * name
Call name.
Tick Frequency
The simulated frequency of curTick(). (In ticks per second)
ThreadContext is the external interface to all thread state for anything outside of the CPU...
RetErrno callClock(ThreadContext *tc)
RetErrno callWrite0(ThreadContext *tc, InPlaceArg str)
virtual int64_t open()
Open the the file.
static void store(ThreadContext *tc, const ArmSemihosting::RetErrno &err)
static const std::vector< const char * > fmodes
State(const ThreadContext *tc)
std::function< std::string(ThreadContext *tc)> Dumper
RetErrno callTickFreq(ThreadContext *tc)
const std::string cmdLine
RetErrno(ArmSemihosting::*)(ThreadContext *tc, Args... args) Implementation
RetErrno callElapsed64(ThreadContext *tc, InPlaceArg ticks)
bool call32(ThreadContext *tc, bool gem5_ops)
Perform an Arm Semihosting call from aarch32 code.
PortProxy Object Declaration.
Internal state for open files.
virtual bool isTTY() const
Check if a file corresponds to a TTY device.
static std::function< RetErrno(ThreadContext *tc, Args... args)> wrapImpl(ArmSemihosting *sh, Implementation< Args... > impl)
uint64_t Tick
Tick count type.
const std::string & fileName()
Semihosting for AArch32 and AArch64.
uint64_t semiTick(Tick tick) const
static PortProxy & portProxy(ThreadContext *tc)
static const std::map< uint32_t, SemiCall > calls
FileBase(ArmSemihosting &_parent, const char *name, const char *_mode)
RetErrno callOpen(ThreadContext *tc, const Addr name_base, int fmode, size_t name_size)
std::function< RetErrno(ArmSemihosting *sh, ThreadContext *tc)> Dispatcher
std::pair< uint64_t, SemiErrno > RetErrno
InPlaceArg(Addr _addr, size_t _size)
RetErrno callElapsed32(ThreadContext *tc, InPlaceArg low, InPlaceArg high)
Base class for serial devices such as terminals.
RetErrno callGem5PseudoOp64(ThreadContext *tc, uint64_t encoded_func)
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.
RetErrno callRemove(ThreadContext *tc, Addr name_base, size_t name_size)
std::string readString(ThreadContext *tc, Addr ptr, size_t len)
static RetErrno retOK(uint64_t r)
RetErrno callReadC(ThreadContext *tc)
Basic support for object serialization.
RetErrno callFLen(ThreadContext *tc, Handle handle)
RetErrno callRename(ThreadContext *tc, Addr from_addr, size_t from_size, Addr to_addr, size_t to_size)
static const std::vector< uint8_t > features
RetErrno callSystem(ThreadContext *tc, Addr cmd_addr, size_t cmd_size)
This object is a proxy for a port or other object which implements the functional response protocol...
virtual const std::string name() const
static const std::map< uint64_t, const char * > exitCodes
RetErrno callTmpNam(ThreadContext *tc, Addr buffer, uint64_t id, size_t size)
RetErrno callIsTTY(ThreadContext *tc, Handle handle)
std::ostream CheckpointOut
Bitfield< 31, 29 > format
void semiExit(uint64_t code, uint64_t subcode)
virtual int64_t close()
Close the file.
static Dumper buildDumper(const char *name, Implementation< Args... > impl)
int findMsbSet(uint64_t val)
Returns the bit position of the MSB that is set in the input.
const unsigned tickShift
Number of bits to right shift gem5 ticks to fit in a uint32_t.
const time_t timeBase
Base time when the simulation started.
int64_t open() override
Open the the file.
unsigned calcTickShift() const
RetErrno callIsError(ThreadContext *tc, int64_t status)
RetErrno callExit64(ThreadContext *tc, uint64_t code, uint64_t subcode)
Semihosting call information structure.
RetErrno callRead(ThreadContext *tc, Handle handle, Addr buffer, size_t size)
static void store(ThreadContext *tc, const ArmSemihosting::RetErrno &err)
void unrecognizedCall(ThreadContext *tc, const char *format, uint64_t op)
RetErrno callExit32(ThreadContext *tc, InPlaceArg code)
RetErrno callExitExtended(ThreadContext *tc, uint64_t code, uint64_t subcode)
uint64_t read(ThreadContext *tc, ByteOrder endian)
RetErrno callHeapInfo32(ThreadContext *tc, Addr block_addr)
RetErrno callWrite(ThreadContext *tc, Handle handle, Addr buffer, size_t size)
RetErrno callWriteC(ThreadContext *tc, InPlaceArg c)
Abstract superclass for simulation objects.
SemiCall(const char *_name, Implementation< Args... > common)
RetErrno callGetCmdLine(ThreadContext *tc, Addr addr, InPlaceArg size_arg)
ByteOrder byteOrder(const ThreadContext *tc)
void gatherHeapInfo(ThreadContext *tc, bool aarch64, Addr &heap_base, Addr &heap_limit, Addr &stack_base, Addr &stack_limit)
std::ostream & operator<<(std::ostream &os, const ArmSemihosting::InPlaceArg &ipa)