Go to the documentation of this file.
49 #include "debug/Semihosting.hh"
54 #include "params/ArmSemihosting.hh"
99 "r",
"rb",
"r+",
"r+b",
100 "w",
"wb",
"w+",
"w+b",
101 "a",
"ab",
"a+",
"a+b",
105 { 0x20000,
"semi:ADP_Stopped_BranchThroughZero" },
106 { 0x20001,
"semi:ADP_Stopped_UndefinedInstr" },
107 { 0x20002,
"semi:ADP_Stopped_SoftwareInterrupt" },
108 { 0x20003,
"semi:ADP_Stopped_PrefetchAbort" },
109 { 0x20004,
"semi:ADP_Stopped_DataAbort" },
110 { 0x20005,
"semi:ADP_Stopped_AddressException" },
111 { 0x20006,
"semi:ADP_Stopped_IRQ" },
112 { 0x20007,
"semi:ADP_Stopped_FIQ" },
114 { 0x20020,
"semi:ADP_Stopped_BreakPoint" },
115 { 0x20021,
"semi:ADP_Stopped_WatchPoint" },
116 { 0x20022,
"semi:ADP_Stopped_StepComplete" },
117 { 0x20023,
"semi:ADP_Stopped_RunTimeErrorUnknown" },
118 { 0x20024,
"semi:ADP_Stopped_InternalError" },
119 { 0x20025,
"semi:ADP_Stopped_UserInterruption" },
120 { 0x20026,
"semi:ADP_Stopped_ApplicationExit" },
121 { 0x20027,
"semi:ADP_Stopped_StackOverflow" },
122 { 0x20028,
"semi:ADP_Stopped_DivisionByZero" },
123 { 0x20029,
"semi:ADP_Stopped_DivisionByZero" },
128 0x53, 0x48, 0x46, 0x42,
136 {
"stdout", ::stdout},
138 {
"stderr", ::stderr},
144 memReserve(
p.mem_reserve),
145 stackSize(
p.stack_size),
146 timeBase([
p]{
struct tm
t =
p.time;
return mkutctime(&
t); }()),
147 tickShift(calcTickShift()),
149 filesRootDir(!
p.files_root_dir.empty() &&
150 p.files_root_dir.back() !=
'/' ?
151 p.files_root_dir +
'/' :
p.files_root_dir),
152 stdin(getSTDIO(
"stdin",
p.stdin,
"r")),
153 stdout(getSTDIO(
"stdout",
p.stdout,
"w")),
154 stderr(
p.stderr ==
p.stdout ?
155 stdout : getSTDIO(
"stderr",
p.stderr,
"w"))
159 files.push_back(
nullptr);
162 inform(
"Semihosting: Shifting elapsed ticks by %i bits.",
171 unrecognizedCall<Abi64>(
172 tc,
"Gem5 semihosting op (0x%x) disabled from here.",
op);
177 if (it ==
calls.end()) {
178 unrecognizedCall<Abi64>(
179 tc,
"Unknown aarch64 semihosting call: op = 0x%x",
op);
184 DPRINTF(Semihosting,
"Semihosting call64: %s\n", call.
dump64(tc));
187 DPRINTF(Semihosting,
"\t ->: 0x%x, %i\n",
err.first,
err.second);
197 unrecognizedCall<Abi32>(
198 tc,
"Gem5 semihosting op (0x%x) disabled from here.",
op);
203 if (it ==
calls.end()) {
204 unrecognizedCall<Abi32>(
205 tc,
"Unknown aarch32 semihosting call: op = 0x%x",
op);
210 DPRINTF(Semihosting,
"Semihosting call32: %s\n", call.
dump32(tc));
213 DPRINTF(Semihosting,
"\t ->: 0x%x, %i\n",
err.first,
err.second);
224 for (
int i = 0;
i <
files.size();
i++) {
240 files.resize(num_files);
241 for (
int i = 0;
i < num_files;
i++)
248 static std::unique_ptr<PortProxy> port_proxy_s;
249 static System *secure_sys =
nullptr;
253 if (sys != secure_sys) {
265 return *port_proxy_s;
280 return std::string(buf.data());
285 int fmode,
size_t name_size)
289 DPRINTF(Semihosting,
"Semihosting SYS_OPEN(0x%x, %i[%s], %i)\n",
290 name_base, fmode,
mode ?
mode :
"-", name_size);
291 if (!
mode || !name_base)
294 std::string fname =
readString(tc, name_base, name_size);
295 if (!fname.empty() && fname.front() !=
'/')
298 std::unique_ptr<ArmSemihosting::FileBase> file =
300 int64_t ret = file->open();
301 DPRINTF(Semihosting,
"Semihosting SYS_OPEN(\"%s\", %i[%s]): %i\n",
302 fname, fmode,
mode, ret);
306 files.push_back(std::move(file));
314 if (handle >
files.size()) {
315 DPRINTF(Semihosting,
"Semihosting SYS_CLOSE(%i): Illegal file\n");
319 std::unique_ptr<FileBase> &file =
files[handle];
320 int64_t error = file->close();
321 DPRINTF(Semihosting,
"Semihosting SYS_CLOSE(%i[%s]): %i\n",
322 handle, file->fileName(), error);
328 files[handle].reset();
338 DPRINTF(Semihosting,
"Semihosting SYS_WRITEC('%c')\n",
c);
347 DPRINTF(Semihosting,
"Semihosting SYS_WRITE0(...)\n");
351 std::cout.write(str.c_str(), str.size());
366 int64_t ret =
files[handle]->write(buffer.data(), buffer.size());
385 int64_t ret =
files[handle]->read(buffer.data(), buffer.size());
389 panic_if(ret > buffer.size(),
"Read longer than buffer size.");
394 return retOK(size - ret);
401 return retOK((
char)std::cin.get());
416 int64_t ret =
files[handle]->isTTY();
420 return retOK(ret ? 1 : 0);
430 int64_t ret =
files[handle]->seek(pos);
444 int64_t ret =
files[handle]->flen();
456 std::string
path =
"";
457 int64_t unlink_call_ret = 0;
462 unlink_call_ret = unlink(
path.c_str());
464 }
while ((unlink_call_ret < 0) && (errno == EBUSY));
466 const size_t path_len =
path.length();
467 if (path_len >= size)
477 std::string fname =
readString(tc, name_base, name_size);
479 if (remove(fname.c_str()) != 0) {
488 Addr to_addr,
size_t to_size)
490 std::string from =
readString(tc, from_addr, from_size);
491 std::string to =
readString(tc, to_addr, to_size);
493 if (rename(from.c_str(), to.c_str()) != 0) {
515 const std::string cmd =
readString(tc, cmd_addr, cmd_size);
516 warn(
"Semihosting: SYS_SYSTEM not implemented. Guest tried to run: %s\n",
535 size_t size = size_arg.
read(tc, endian);
537 if (
cmdLine.size() + 1 < size) {
549 Addr &stack_base,
Addr &stack_limit)
553 fatal_if(memories.size() < 1,
"No memories reported from System");
554 warn_if(memories.size() > 1,
"Multiple physical memory ranges available. "
555 "Using first range heap/stack.");
562 const Addr phys_max = (1
ULL << 32) - 1;
564 "Physical memory out of range for a 32-bit guest.");
565 if (mem_end > phys_max) {
566 warn(
"Some physical memory out of range for a 32-bit guest.");
572 "Physical memory too small to fit desired stack and a heap.");
574 heap_base = mem_start;
576 stack_base = (mem_end + 1) & ~0x7ULL;
577 stack_limit = heap_limit;
579 inform(
"Reporting heap/stack info to guest:\n"
580 "\tHeap base: 0x%x\n"
581 "\tHeap limit: 0x%x\n"
582 "\tStack base: 0x%x\n"
583 "\tStack limit: 0x%x\n",
584 heap_base, heap_limit, stack_base, stack_limit);
590 uint64_t heap_base, heap_limit, stack_base, stack_limit;
591 gatherHeapInfo(tc,
false, heap_base, heap_limit, stack_base, stack_limit);
593 std::array<uint32_t, 4> block = {{
594 (uint32_t)heap_base, (uint32_t)heap_limit,
595 (uint32_t)stack_base, (uint32_t)stack_limit
605 uint64_t heap_base, heap_limit, stack_base, stack_limit;
606 gatherHeapInfo(tc,
true, heap_base, heap_limit, stack_base, stack_limit);
608 std::array<uint64_t, 4> block = {{
609 heap_base, heap_limit, stack_base, stack_limit
632 uint64_t code, uint64_t subcode)
657 low.
write(tc, tick, endian);
658 high.
write(tc, tick >> 32, endian);
710 template <
typename T>
712 public Argument<ArmSemihosting::Abi32, T>
714 template <
typename T>
716 public Argument<ArmSemihosting::Abi64, T>
728 if (PseudoInst::pseudoInst<SemiPseudoAbi32>(tc, func, ret))
741 if (PseudoInst::pseudoInst<SemiPseudoAbi64>(tc, func, ret))
749 const std::string &
name,
const char *
mode)
755 fatal(
"Failed to open %s (%s): %s\n",
756 stream_name,
name, strerror(errno));
764 std::unique_ptr<ArmSemihosting::FileBase>
768 std::unique_ptr<FileBase> file;
769 if (fname ==
":semihosting-features") {
778 std::unique_ptr<ArmSemihosting::FileBase>
782 std::unique_ptr<FileBase> file;
789 std::string fname,
mode;
792 file = create(parent, fname,
mode.c_str());
794 file->unserialize(
cp);
883 const char *_name,
const char *_perms)
898 panic_if(file,
"Trying to open an already open file.\n");
900 if (_name ==
":tt") {
901 if (
mode[0] ==
'r') {
903 }
else if (
mode[0] ==
'w') {
904 file = parent.stdout;
905 }
else if (
mode[0] ==
'a') {
906 file = parent.stderr;
908 warn(
"Unknown file mode for the ':tt' special file");
912 std::string real_mode(this->
mode);
915 if (in_cpt && real_mode[0] ==
'w')
918 file = fopen(_name.c_str(), real_mode.c_str());
921 return file ? 0 : -errno;
927 panic_if(!file,
"Trying to close an already closed file.\n");
940 return file == parent.stdout ||
941 file == parent.stderr ||
942 file == parent.stdin;
948 panic_if(!file,
"Trying to read from a closed file");
950 size_t ret = fread(buffer, 1, size, file);
954 return ferror(file) ? -EINVAL : 0;
963 panic_if(!file,
"Trying to write to a closed file");
966 size_t ret = fwrite(buffer, 1, size, file);
979 panic_if(!file,
"Trying to seek in a closed file");
982 if (fseek(file, _pos, SEEK_SET) == 0)
992 long pos = ftell(file);
996 if (fseek(file, 0, SEEK_END) != 0)
999 long len = ftell(file);
1003 if (fseek(file, pos, SEEK_SET) != 0)
1016 long pos = file ? ftell(file) : 0;
1017 panic_if(pos < 0,
"Failed to get file position.");
1027 if (openImpl(
true) < 0) {
1028 fatal(
"Failed to open file: %s", _name);
1034 if (fseek(file, pos, SEEK_SET) != 0) {
1035 fatal(
"Failed seek to current position (%i) in '%s'", pos, _name);
void gatherHeapInfo(ThreadContext *tc, bool aarch64, Addr &heap_base, Addr &heap_limit, Addr &stack_base, Addr &stack_limit)
#define fatal(...)
This implements a cprintf based fatal() function.
Implementation of the ':semihosting-features' magic file.
static std::unique_ptr< FileBase > create(ArmSemihosting &parent, const std::string &fname, const char *mode)
static RetErrno retError(SemiErrno e)
RetErrno callWrite0(ThreadContext *tc, InPlaceArg str)
RetErrno callWrite(ThreadContext *tc, Handle handle, Addr buffer, size_t size)
RetErrno callExit64(ThreadContext *tc, uint64_t code, uint64_t subcode)
double s
These variables equal the number of ticks in the unit of time they're named after in a double.
virtual int64_t flen()
Get the length of a file in bytes.
@ SECURE
The request targets the secure memory space.
RetErrno callElapsed32(ThreadContext *tc, InPlaceArg low, InPlaceArg high)
static const std::map< const std::string, FILE * > stdioMap
RetErrno callTickFreq(ThreadContext *tc)
RetErrno callRename(ThreadContext *tc, Addr from_addr, size_t from_size, Addr to_addr, size_t to_size)
static RetErrno retOK(uint64_t r)
RetErrno callElapsed64(ThreadContext *tc, InPlaceArg ticks)
#define UNSERIALIZE_SCALAR(scalar)
RetErrno callWriteC(ThreadContext *tc, InPlaceArg c)
ByteOrder byteOrder(const ThreadContext *tc)
RetErrno callOpen(ThreadContext *tc, const Addr name_base, int fmode, size_t name_size)
time_t mkutctime(struct tm *time)
int64_t openImpl(bool unserialize)
uint64_t semiTick(Tick tick) const
const std::string cmdLine
static const std::vector< uint8_t > features
State(const ThreadContext *tc)
static const std::vector< const char * > fmodes
RetErrno callErrno(ThreadContext *tc)
RetErrno callRemove(ThreadContext *tc, Addr name_base, size_t name_size)
virtual int64_t seek(uint64_t pos)
Seek to an absolute position in the file.
State(const ThreadContext *tc)
RetErrno callExit32(ThreadContext *tc, InPlaceArg code)
RetErrno callClock(ThreadContext *tc)
std::string resolve(const std::string &name) const
Returns relative file names prepended with name of this directory.
static void decodeAddrOffset(Addr offset, uint8_t &func)
bool call32(ThreadContext *tc, bool gem5_ops)
Perform an Arm Semihosting call from aarch32 code.
bool FullSystem
The FullSystem variable can be used to determine the current mode of simulation.
RetErrno callGetCmdLine(ThreadContext *tc, Addr addr, InPlaceArg size_arg)
void write(ThreadContext *tc, uint64_t val, ByteOrder endian)
RetErrno callSeek(ThreadContext *tc, Handle handle, uint64_t pos)
RetErrno callTime(ThreadContext *tc)
PhysicalMemory & getPhysMem()
Get a pointer to access the physical memory of the system.
int64_t flen() override
Get the length of a file in bytes.
RetErrno callHeapInfo32(ThreadContext *tc, Addr block_addr)
ArmSemihosting(const ArmSemihostingParams &p)
Tick Frequency
The simulated frequency of curTick(). (In ticks per second)
void writeBlob(Addr addr, const void *p, int size) const
Same as tryWriteBlob, but insists on success.
int64_t read(uint8_t *buffer, uint64_t size) override
Read data from file.
File(ArmSemihosting &_parent, const char *name, const char *mode)
virtual int64_t read(uint8_t *buffer, uint64_t size)
Read data from file.
ThreadContext is the external interface to all thread state for anything outside of the CPU.
void readString(std::string &str, Addr addr) const
Same as tryReadString, but insists on success.
The AddrRange class encapsulates an address range, and supports a number of tests to check if two ran...
bool call64(ThreadContext *tc, bool gem5_ops)
Perform an Arm Semihosting call from aarch64 code.
bool isTTY() const override
Check if a file corresponds to a TTY device.
static const std::map< uint32_t, SemiCall > calls
RetErrno callSystem(ThreadContext *tc, Addr cmd_addr, size_t cmd_size)
std::string readString(ThreadContext *tc, Addr ptr, size_t len)
RetErrno callIsTTY(ThreadContext *tc, Handle handle)
RetErrno callHeapInfo64(ThreadContext *tc, Addr block_addr)
static const std::map< uint64_t, const char * > exitCodes
int64_t close() override
Close the file.
void semiExit(uint64_t code, uint64_t subcode)
std::ostream & operator<<(std::ostream &os, const ArmSemihosting::InPlaceArg &ipa)
void exitSimLoop(const std::string &message, int exit_code, Tick when, Tick repeat, bool serialize)
Schedule an event to exit the simulation loop (returning to Python) at the end of the current cycle (...
int64_t seek(uint64_t pos) override
Seek to an absolute position in the file.
void serialize(CheckpointOut &cp) const override
Serialize an object.
static FILE * getSTDIO(const char *stream_name, const std::string &name, const char *mode)
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
int64_t seek(uint64_t pos) override
Seek to an absolute position in the file.
const std::string & name()
#define SERIALIZE_SCALAR(scalar)
void paramOut(CheckpointOut &cp, const std::string &name, ExtMachInst const &machInst)
uint64_t read(ThreadContext *tc, ByteOrder endian)
virtual const std::string name() const
virtual PortProxy & getVirtProxy()=0
RetErrno callGem5PseudoOp32(ThreadContext *tc, uint32_t encoded_func)
RetErrno callExitExtended(ThreadContext *tc, uint64_t code, uint64_t subcode)
#define warn_if(cond,...)
Conditional warning macro that checks the supplied condition and only prints a warning if the conditi...
This proxy attempts to translate virtual addresses using the TLBs.
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
The physical memory encapsulates all memories in the system and provides basic functionality for acce...
static PortProxy & portProxy(ThreadContext *tc)
This object is a proxy for a port or other object which implements the functional response protocol,...
Semihosting call information structure.
void serialize(CheckpointOut &cp) const override
Serialize an object.
T read(Addr address) const
Read sizeof(T) bytes from address and return as object T.
int64_t read(uint8_t *buffer, uint64_t size) override
Read data from file.
RetErrno callFLen(ThreadContext *tc, Handle handle)
RetErrno callReadC(ThreadContext *tc)
FileFeatures(ArmSemihosting &_parent, const char *name, const char *mode)
void ccprintf(cp::Print &print)
void unserialize(CheckpointIn &cp) override
Unserialize an object.
void unserialize(CheckpointIn &cp) override
Unserialize an object.
int64_t write(const uint8_t *buffer, uint64_t size) override
Write data to file.
virtual int64_t write(const uint8_t *buffer, uint64_t size)
Write data to file.
std::ostream CheckpointOut
static const std::string & currentSection()
Gets the fully-qualified name of the active section.
Tick curTick()
The universal simulation clock.
AddrRangeList getConfAddrRanges() const
Get the memory ranges for all memories that are to be reported to the configuration table.
RetErrno callClose(ThreadContext *tc, Handle handle)
virtual RegVal readIntReg(RegIndex reg_idx) const =0
#define fatal_if(cond,...)
Conditional fatal macro that checks the supplied condition and only causes a fatal error if the condi...
void paramIn(CheckpointIn &cp, const std::string &name, ExtMachInst &machInst)
std::vector< std::unique_ptr< FileBase > > files
void readBlob(Addr addr, void *p, int size) const
Higher level interfaces based on the above.
const time_t timeBase
Base time when the simulation started.
Semihosting for AArch32 and AArch64.
RetErrno callTmpNam(ThreadContext *tc, Addr buffer, uint64_t id, size_t size)
RetErrno callRead(ThreadContext *tc, Handle handle, Addr buffer, size_t size)
void serialize(CheckpointOut &cp) const override
Serialize an object.
RetErrno callIsError(ThreadContext *tc, int64_t status)
std::string csprintf(const char *format, const Args &...args)
std::pair< uint64_t, SemiErrno > RetErrno
void serialize(CheckpointOut &cp) const override
Serialize an object.
static std::stack< std::string > path
#define ULL(N)
uint64_t constant
void write(Addr address, const T &data) const
Write object T to address.
bool isSecure(ThreadContext *tc)
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Internal state for open files.
virtual System * getSystemPtr()=0
RetErrno callGem5PseudoOp64(ThreadContext *tc, uint64_t encoded_func)
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Abstract superclass for simulation objects.
Generated on Tue Mar 23 2021 19:41:20 for gem5 by doxygen 1.8.17