gem5  v21.0.1.0
semihosting.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2018, 2019 ARM Limited
3  * All rights reserved
4  *
5  * The license below extends only to copyright in the software and shall
6  * not be construed as granting a license to any other intellectual
7  * property including but not limited to intellectual property relating
8  * to a hardware implementation of the functionality of the software
9  * licensed hereunder. You may use the software subject to the license
10  * terms below provided that you ensure that this notice is replicated
11  * unmodified and in its entirety in all distributions of the software,
12  * modified or unmodified, in source code or in binary form.
13  *
14  * Redistribution and use in source and binary forms, with or without
15  * modification, are permitted provided that the following conditions are
16  * met: redistributions of source code must retain the above copyright
17  * notice, this list of conditions and the following disclaimer;
18  * redistributions in binary form must reproduce the above copyright
19  * notice, this list of conditions and the following disclaimer in the
20  * documentation and/or other materials provided with the distribution;
21  * neither the name of the copyright holders nor the names of its
22  * contributors may be used to endorse or promote products derived from
23  * this software without specific prior written permission.
24  *
25  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
26  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
27  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
28  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
29  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
30  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
31  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
32  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
33  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
34  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
35  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
36  */
37 
38 #include "arch/arm/semihosting.hh"
39 
40 #include <unistd.h>
41 
42 #include <cerrno>
43 #include <cstdio>
44 
45 #include "arch/arm/utility.hh"
46 #include "base/logging.hh"
47 #include "base/output.hh"
48 #include "base/time.hh"
49 #include "debug/Semihosting.hh"
50 #include "dev/serial/serial.hh"
51 #include "mem/physical.hh"
54 #include "params/ArmSemihosting.hh"
55 #include "sim/byteswap.hh"
56 #include "sim/full_system.hh"
57 #include "sim/pseudo_inst.hh"
58 #include "sim/sim_exit.hh"
59 #include "sim/system.hh"
60 
61 const std::map<uint32_t, ArmSemihosting::SemiCall> ArmSemihosting::calls{
62  { SYS_OPEN, { "SYS_OPEN", &ArmSemihosting::callOpen } },
63  { SYS_CLOSE, { "SYS_CLOSE", &ArmSemihosting::callClose } },
64  { SYS_WRITEC, { "SYS_WRITEC", &ArmSemihosting::callWriteC } },
65  { SYS_WRITE0, { "SYS_WRITE0", &ArmSemihosting::callWrite0 } },
66  { SYS_WRITE, { "SYS_WRITE", &ArmSemihosting::callWrite } },
67  { SYS_READ, { "SYS_READ", &ArmSemihosting::callRead } },
68  { SYS_READC, { "SYS_READC", &ArmSemihosting::callReadC } },
69  { SYS_ISERROR, { "SYS_ISERROR", &ArmSemihosting::callIsError } },
70  { SYS_ISTTY, { "SYS_ISTTY", &ArmSemihosting::callIsTTY } },
71  { SYS_SEEK, { "SYS_SEEK", &ArmSemihosting::callSeek } },
72  { SYS_FLEN, { "SYS_FLEN", &ArmSemihosting::callFLen } },
73  { SYS_TMPNAM, { "SYS_TMPNAM", &ArmSemihosting::callTmpNam } },
74  { SYS_REMOVE, { "SYS_REMOVE", &ArmSemihosting::callRemove } },
75  { SYS_RENAME, { "SYS_RENAME", &ArmSemihosting::callRename } },
76  { SYS_CLOCK, { "SYS_CLOCK", &ArmSemihosting::callClock } },
77  { SYS_TIME, { "SYS_TIME", &ArmSemihosting::callTime } },
78  { SYS_SYSTEM, { "SYS_SYSTEM", &ArmSemihosting::callSystem } },
79  { SYS_ERRNO, { "SYS_ERRNO", &ArmSemihosting::callErrno } },
80  { SYS_GET_CMDLINE,
81  { "SYS_GET_CMDLINE", &ArmSemihosting::callGetCmdLine } },
82  { SYS_HEAPINFO, { "SYS_HEAPINFO", &ArmSemihosting::callHeapInfo32,
84 
85  { SYS_EXIT, { "SYS_EXIT", &ArmSemihosting::callExit32,
87  { SYS_EXIT_EXTENDED,
88  { "SYS_EXIT_EXTENDED", &ArmSemihosting::callExitExtended } },
89 
90  { SYS_ELAPSED, { "SYS_ELAPSED", &ArmSemihosting::callElapsed32,
92  { SYS_TICKFREQ, { "SYS_TICKFREQ", &ArmSemihosting::callTickFreq } },
93  { SYS_GEM5_PSEUDO_OP,
94  { "SYS_GEM5_PSEUDO_OP", &ArmSemihosting::callGem5PseudoOp32,
96 };
97 
99  "r", "rb", "r+", "r+b",
100  "w", "wb", "w+", "w+b",
101  "a", "ab", "a+", "a+b",
102 };
103 
104 const std::map<uint64_t, const char *> ArmSemihosting::exitCodes{
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" },
113 
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" },
124 };
125 
126 
128  0x53, 0x48, 0x46, 0x42, // Magic
129  0x3, // EXT_EXIT_EXTENDED, EXT_STDOUT_STDERR
130 };
131 
132 const std::map<const std::string, FILE *> ArmSemihosting::stdioMap{
133  {"cin", ::stdin},
134  {"stdin", ::stdin},
135  {"cout", ::stdout},
136  {"stdout", ::stdout},
137  {"cerr", ::stderr},
138  {"stderr", ::stderr},
139 };
140 
141 ArmSemihosting::ArmSemihosting(const ArmSemihostingParams &p)
142  : SimObject(p),
143  cmdLine(p.cmd_line),
144  memReserve(p.mem_reserve),
145  stackSize(p.stack_size),
146  timeBase([p]{ struct tm t = p.time; return mkutctime(&t); }()),
147  tickShift(calcTickShift()),
148  semiErrno(0),
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"))
156 {
157  // Create an empty place-holder file for position 0 as semi-hosting
158  // calls typically expect non-zero file handles.
159  files.push_back(nullptr);
160 
161  if (tickShift > 0)
162  inform("Semihosting: Shifting elapsed ticks by %i bits.",
163  tickShift);
164 }
165 
166 bool
168 {
170  if (op > MaxStandardOp && !gem5_ops) {
171  unrecognizedCall<Abi64>(
172  tc, "Gem5 semihosting op (0x%x) disabled from here.", op);
173  return false;
174  }
175 
176  auto it = calls.find(op);
177  if (it == calls.end()) {
178  unrecognizedCall<Abi64>(
179  tc, "Unknown aarch64 semihosting call: op = 0x%x", op);
180  return false;
181  }
182  const SemiCall &call = it->second;
183 
184  DPRINTF(Semihosting, "Semihosting call64: %s\n", call.dump64(tc));
185  auto err = call.call64(this, tc);
186  semiErrno = err.second;
187  DPRINTF(Semihosting, "\t ->: 0x%x, %i\n", err.first, err.second);
188 
189  return true;
190 }
191 
192 bool
194 {
196  if (op > MaxStandardOp && !gem5_ops) {
197  unrecognizedCall<Abi32>(
198  tc, "Gem5 semihosting op (0x%x) disabled from here.", op);
199  return false;
200  }
201 
202  auto it = calls.find(op);
203  if (it == calls.end()) {
204  unrecognizedCall<Abi32>(
205  tc, "Unknown aarch32 semihosting call: op = 0x%x", op);
206  return false;
207  }
208  const SemiCall &call = it->second;
209 
210  DPRINTF(Semihosting, "Semihosting call32: %s\n", call.dump32(tc));
211  auto err = call.call32(this, tc);
212  semiErrno = err.second;
213  DPRINTF(Semihosting, "\t ->: 0x%x, %i\n", err.first, err.second);
214 
215  return true;
216 }
217 
218 void
220 {
222 
223  paramOut(cp, "num_files", files.size());
224  for (int i = 0; i < files.size(); i++) {
225  // File closed?
226  if (!files[i])
227  continue;
228 
229  files[i]->serializeSection(cp, csprintf("file%i", i));
230  }
231 }
232 
233 void
235 {
237 
238  size_t num_files;
239  paramIn(cp, "num_files", num_files);
240  files.resize(num_files);
241  for (int i = 0; i < num_files; i++)
242  files[i] = FileBase::create(*this, cp, csprintf("file%i", i));
243 }
244 
245 PortProxy &
247 {
248  static std::unique_ptr<PortProxy> port_proxy_s;
249  static System *secure_sys = nullptr;
250 
251  if (ArmISA::isSecure(tc)) {
252  System *sys = tc->getSystemPtr();
253  if (sys != secure_sys) {
254  if (FullSystem) {
255  port_proxy_s.reset(
257  } else {
258  port_proxy_s.reset(
261  Request::SECURE));
262  }
263  }
264  secure_sys = sys;
265  return *port_proxy_s;
266  } else {
267  return tc->getVirtProxy();
268  }
269 }
270 
271 
272 std::string
274 {
275  std::vector<char> buf(len + 1);
276 
277  buf[len] = '\0';
278  portProxy(tc).readBlob(ptr, buf.data(), len);
279 
280  return std::string(buf.data());
281 }
282 
285  int fmode, size_t name_size)
286 {
287  const char *mode = fmode < fmodes.size() ? fmodes[fmode] : nullptr;
288 
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)
292  return retError(EINVAL);
293 
294  std::string fname = readString(tc, name_base, name_size);
295  if (!fname.empty() && fname.front() != '/')
296  fname = filesRootDir + fname;
297 
298  std::unique_ptr<ArmSemihosting::FileBase> file =
299  FileBase::create(*this, fname, mode);
300  int64_t ret = file->open();
301  DPRINTF(Semihosting, "Semihosting SYS_OPEN(\"%s\", %i[%s]): %i\n",
302  fname, fmode, mode, ret);
303  if (ret < 0) {
304  return retError(-ret);
305  } else {
306  files.push_back(std::move(file));
307  return retOK(files.size() - 1);
308  }
309 }
310 
313 {
314  if (handle > files.size()) {
315  DPRINTF(Semihosting, "Semihosting SYS_CLOSE(%i): Illegal file\n");
316  return retError(EBADF);
317  }
318 
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);
323  if (error < 0) {
324  return retError(-error);
325  } else {
326  // Zap the pointer and free the entry in the file table as
327  // well.
328  files[handle].reset();
329  return retOK(0);
330  }
331 }
332 
335 {
336  const char c = portProxy(tc).read<char>(arg.addr);
337 
338  DPRINTF(Semihosting, "Semihosting SYS_WRITEC('%c')\n", c);
339  std::cout.put(c);
340 
341  return retOK(0);
342 }
343 
346 {
347  DPRINTF(Semihosting, "Semihosting SYS_WRITE0(...)\n");
348  PortProxy &proxy = portProxy(tc);
349  std::string str;
350  proxy.readString(str, arg.addr);
351  std::cout.write(str.c_str(), str.size());
352 
353  return retOK(0);
354 }
355 
358  size_t size)
359 {
360  if (handle > files.size() || !files[handle])
361  return RetErrno(size, EBADF);
362 
363  std::vector<uint8_t> buffer(size);
364  portProxy(tc).readBlob(addr, buffer.data(), buffer.size());
365 
366  int64_t ret = files[handle]->write(buffer.data(), buffer.size());
367  if (ret < 0) {
368  // No bytes written (we're returning the number of bytes not
369  // written)
370  return RetErrno(size, -ret);
371  } else {
372  // Return the number of bytes not written
373  return RetErrno(size - ret, 0);
374  }
375 }
376 
379  size_t size)
380 {
381  if (handle > files.size() || !files[handle])
382  return RetErrno(size, EBADF);
383 
384  std::vector<uint8_t> buffer(size);
385  int64_t ret = files[handle]->read(buffer.data(), buffer.size());
386  if (ret < 0) {
387  return RetErrno(size, -ret);
388  } else {
389  panic_if(ret > buffer.size(), "Read longer than buffer size.");
390 
391  portProxy(tc).writeBlob(addr, buffer.data(), ret);
392 
393  // Return the number of bytes not written
394  return retOK(size - ret);
395  }
396 }
397 
400 {
401  return retOK((char)std::cin.get());
402 }
403 
406 {
407  return retOK(status < 0 ? 1 : 0);
408 }
409 
412 {
413  if (handle > files.size() || !files[handle])
414  return retError(EBADF);
415 
416  int64_t ret = files[handle]->isTTY();
417  if (ret < 0) {
418  return retError(-ret);
419  } else {
420  return retOK(ret ? 1 : 0);
421  }
422 }
423 
426 {
427  if (handle > files.size() || !files[handle])
428  return retError(EBADF);
429 
430  int64_t ret = files[handle]->seek(pos);
431  if (ret < 0) {
432  return retError(-ret);
433  } else {
434  return retOK(0);
435  }
436 }
437 
440 {
441  if (handle > files.size() || !files[handle])
442  return retError(EBADF);
443 
444  int64_t ret = files[handle]->flen();
445  if (ret < 0) {
446  return retError(-ret);
447  } else {
448  return retOK(ret);
449  }
450 }
451 
454  size_t size)
455 {
456  std::string path = "";
457  int64_t unlink_call_ret = 0;
458 
459  do {
460  path = simout.resolve(csprintf("%s.tmp%05i", name(), tmpNameIndex++));
461  // remove the (potentially existing) file of the given path
462  unlink_call_ret = unlink(path.c_str());
463  // if the file is busy, find another name
464  } while ((unlink_call_ret < 0) && (errno == EBUSY));
465 
466  const size_t path_len = path.length();
467  if (path_len >= size)
468  return retError(ENOSPC);
469 
470  portProxy(tc).writeBlob(addr, path.c_str(), path_len + 1);
471  return retOK(0);
472 }
473 
475 ArmSemihosting::callRemove(ThreadContext *tc, Addr name_base, size_t name_size)
476 {
477  std::string fname = readString(tc, name_base, name_size);
478 
479  if (remove(fname.c_str()) != 0) {
480  return retError(errno);
481  } else {
482  return retOK(0);
483  }
484 }
485 
487 ArmSemihosting::callRename(ThreadContext *tc, Addr from_addr, size_t from_size,
488  Addr to_addr, size_t to_size)
489 {
490  std::string from = readString(tc, from_addr, from_size);
491  std::string to = readString(tc, to_addr, to_size);
492 
493  if (rename(from.c_str(), to.c_str()) != 0) {
494  return retError(errno);
495  } else {
496  return retOK(0);
497  }
498 }
499 
502 {
503  return retOK(curTick() / (SimClock::Int::s / 100));
504 }
505 
508 {
509  return retOK(timeBase + round(curTick() / SimClock::Float::s));
510 }
511 
513 ArmSemihosting::callSystem(ThreadContext *tc, Addr cmd_addr, size_t cmd_size)
514 {
515  const std::string cmd = readString(tc, cmd_addr, cmd_size);
516  warn("Semihosting: SYS_SYSTEM not implemented. Guest tried to run: %s\n",
517  cmd);
518  return retError(EINVAL);
519 
520 }
521 
524 {
525  // Preserve errno by returning it in errno as well.
526  return RetErrno(semiErrno, semiErrno);
527 }
528 
531  InPlaceArg size_arg)
532 {
533  PortProxy &proxy = portProxy(tc);
534  ByteOrder endian = ArmISA::byteOrder(tc);
535  size_t size = size_arg.read(tc, endian);
536 
537  if (cmdLine.size() + 1 < size) {
538  proxy.writeBlob(addr, cmdLine.c_str(), cmdLine.size() + 1);
539  size_arg.write(tc, cmdLine.size(), endian);
540  return retOK(0);
541  } else {
542  return retError(0);
543  }
544 }
545 
546 void
548  Addr &heap_base, Addr &heap_limit,
549  Addr &stack_base, Addr &stack_limit)
550 {
551  const PhysicalMemory &phys = tc->getSystemPtr()->getPhysMem();
552  const AddrRangeList memories = phys.getConfAddrRanges();
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.");
556  const AddrRange memory = *memories.begin();
557  const Addr mem_start = memory.start() + memReserve;
558  Addr mem_end = memory.end();
559 
560  // Make sure that 32-bit guests can access their memory.
561  if (!aarch64) {
562  const Addr phys_max = (1ULL << 32) - 1;
563  panic_if(mem_start > phys_max,
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.");
567  mem_end = phys_max;
568  }
569  }
570 
571  fatal_if(mem_start + stackSize >= mem_end,
572  "Physical memory too small to fit desired stack and a heap.");
573 
574  heap_base = mem_start;
575  heap_limit = mem_end - stackSize + 1;
576  stack_base = (mem_end + 1) & ~0x7ULL; // 8 byte stack alignment
577  stack_limit = heap_limit;
578 
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);
585 }
586 
589 {
590  uint64_t heap_base, heap_limit, stack_base, stack_limit;
591  gatherHeapInfo(tc, false, heap_base, heap_limit, stack_base, stack_limit);
592 
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
596  }};
597  portProxy(tc).write(block_addr, block, ArmISA::byteOrder(tc));
598 
599  return retOK(0);
600 }
601 
604 {
605  uint64_t heap_base, heap_limit, stack_base, stack_limit;
606  gatherHeapInfo(tc, true, heap_base, heap_limit, stack_base, stack_limit);
607 
608  std::array<uint64_t, 4> block = {{
609  heap_base, heap_limit, stack_base, stack_limit
610  }};
611  portProxy(tc).write(block_addr, block, ArmISA::byteOrder(tc));
612 
613  return retOK(0);
614 }
615 
618 {
619  semiExit(code.addr, 0);
620  return retOK(0);
621 }
622 
624 ArmSemihosting::callExit64(ThreadContext *tc, uint64_t code, uint64_t subcode)
625 {
626  semiExit(code, subcode);
627  return retOK(0);
628 }
629 
632  uint64_t code, uint64_t subcode)
633 {
634  semiExit(code, subcode);
635  return retOK(0);
636 }
637 
638 void
639 ArmSemihosting::semiExit(uint64_t code, uint64_t subcode)
640 {
641  auto it = exitCodes.find(code);
642  if (it != exitCodes.end()) {
643  exitSimLoop(it->second, subcode);
644  } else {
645  exitSimLoop(csprintf("semi:0x%x", code), subcode);
646  }
647 }
648 
649 
652  InPlaceArg high)
653 {
654  ByteOrder endian = ArmISA::byteOrder(tc);
655  uint64_t tick = semiTick(curTick());
656 
657  low.write(tc, tick, endian);
658  high.write(tc, tick >> 32, endian);
659 
660  return retOK(0);
661 }
662 
663 
666 {
667  ticks.write(tc, semiTick(curTick()), ArmISA::byteOrder(tc));
668  return retOK(0);
669 }
670 
671 
674 {
676 }
677 
678 
680 {
682  {
683  public:
684  State(const ThreadContext *tc) : ArmSemihosting::Abi32::State(tc)
685  {
686  // Use getAddr() to skip the func number in the first slot.
687  getAddr();
688  }
689  };
690 };
691 
693 {
695  {
696  public:
697  State(const ThreadContext *tc) : ArmSemihosting::Abi64::State(tc)
698  {
699  // Use getAddr() to skip the func number in the first slot.
700  getAddr();
701  }
702  };
703 };
704 
705 namespace GuestABI
706 {
707 
708 // Handle arguments the same as for semihosting operations. Skipping the first
709 // slot is handled internally by the State type.
710 template <typename T>
712  public Argument<ArmSemihosting::Abi32, T>
713 {};
714 template <typename T>
716  public Argument<ArmSemihosting::Abi64, T>
717 {};
718 
719 } // namespace GuestABI
720 
723 {
724  uint8_t func;
725  PseudoInst::decodeAddrOffset(encoded_func, func);
726 
727  uint64_t ret;
728  if (PseudoInst::pseudoInst<SemiPseudoAbi32>(tc, func, ret))
729  return retOK(ret);
730  else
731  return retError(EINVAL);
732 }
733 
736 {
737  uint8_t func;
738  PseudoInst::decodeAddrOffset(encoded_func, func);
739 
740  uint64_t ret;
741  if (PseudoInst::pseudoInst<SemiPseudoAbi64>(tc, func, ret))
742  return retOK(ret);
743  else
744  return retError(EINVAL);
745 }
746 
747 FILE *
748 ArmSemihosting::getSTDIO(const char *stream_name,
749  const std::string &name, const char *mode)
750 {
751  auto it = stdioMap.find(name);
752  if (it == stdioMap.end()) {
753  FILE *f = fopen(name.c_str(), mode);
754  if (!f) {
755  fatal("Failed to open %s (%s): %s\n",
756  stream_name, name, strerror(errno));
757  }
758  return f;
759  } else {
760  return it->second;
761  }
762 }
763 
764 std::unique_ptr<ArmSemihosting::FileBase>
766  ArmSemihosting &parent, const std::string &fname, const char *mode)
767 {
768  std::unique_ptr<FileBase> file;
769  if (fname == ":semihosting-features") {
770  file.reset(new FileFeatures(parent, fname.c_str(), mode));
771  } else {
772  file.reset(new File(parent, fname.c_str(), mode));
773  }
774 
775  return file;
776 }
777 
778 std::unique_ptr<ArmSemihosting::FileBase>
780  CheckpointIn &cp, const std::string &sec)
781 {
782  std::unique_ptr<FileBase> file;
783  ScopedCheckpointSection _sec(cp, sec);
784 
785  // Was the file open when the checkpoint was created?
786  if (!cp.sectionExists(Serializable::currentSection()))
787  return file;
788 
789  std::string fname, mode;
790  paramIn(cp, "name", fname);
791  paramIn(cp, "mode", mode);
792  file = create(parent, fname, mode.c_str());
793  assert(file);
794  file->unserialize(cp);
795 
796  return file;
797 }
798 
799 void
801 {
802  paramOut(cp, "name", _name);
804 }
805 
806 void
808 {
809  /* Unserialization of name and mode happens in
810  * ArmSemihosting::FileBase::create() */
811 }
812 
813 int64_t
814 ArmSemihosting::FileBase::read(uint8_t *buffer, uint64_t size)
815 {
816  return -EINVAL;
817 }
818 
819 int64_t
820 ArmSemihosting::FileBase::write(const uint8_t *buffer, uint64_t size)
821 {
822  return -EINVAL;
823 }
824 
825 int64_t
827 {
828  return -EINVAL;
829 }
830 
831 int64_t
833 {
834  return -EINVAL;
835 }
836 
837 
839  ArmSemihosting &_parent, const char *_name, const char *_mode)
840  : FileBase(_parent, _name, _mode)
841 {
842 }
843 
844 int64_t
845 ArmSemihosting::FileFeatures::read(uint8_t *buffer, uint64_t size)
846 {
847  int64_t len = 0;
848 
849  for (; pos < size && pos < ArmSemihosting::features.size(); pos++)
850  buffer[len++] = ArmSemihosting::features[pos];
851 
852  return len;
853 }
854 
855 int64_t
857 {
858  if (_pos < ArmSemihosting::features.size()) {
859  pos = _pos;
860  return 0;
861  } else {
862  return -ENXIO;
863  }
864 }
865 
866 void
868 {
870  SERIALIZE_SCALAR(pos);
871 }
872 
873 void
875 {
877  UNSERIALIZE_SCALAR(pos);
878 }
879 
880 
881 
883  const char *_name, const char *_perms)
884  : FileBase(_parent, _name, _perms),
885  file(nullptr)
886 {
887 }
888 
890 {
891  if (file)
892  close();
893 }
894 
895 int64_t
897 {
898  panic_if(file, "Trying to open an already open file.\n");
899 
900  if (_name == ":tt") {
901  if (mode[0] == 'r') {
902  file = parent.stdin;
903  } else if (mode[0] == 'w') {
904  file = parent.stdout;
905  } else if (mode[0] == 'a') {
906  file = parent.stderr;
907  } else {
908  warn("Unknown file mode for the ':tt' special file");
909  return -EINVAL;
910  }
911  } else {
912  std::string real_mode(this->mode);
913  // Avoid truncating the file if we are restoring from a
914  // checkpoint.
915  if (in_cpt && real_mode[0] == 'w')
916  real_mode[0] = 'a';
917 
918  file = fopen(_name.c_str(), real_mode.c_str());
919  }
920 
921  return file ? 0 : -errno;
922 }
923 
924 int64_t
926 {
927  panic_if(!file, "Trying to close an already closed file.\n");
928 
929  if (needClose()) {
930  fclose(file);
931  }
932  file = nullptr;
933 
934  return 0;
935 }
936 
937 bool
939 {
940  return file == parent.stdout ||
941  file == parent.stderr ||
942  file == parent.stdin;
943 }
944 
945 int64_t
946 ArmSemihosting::File::read(uint8_t *buffer, uint64_t size)
947 {
948  panic_if(!file, "Trying to read from a closed file");
949 
950  size_t ret = fread(buffer, 1, size, file);
951  if (ret == 0) {
952  // Error or EOF. Assume errors are due to invalid file
953  // operations (e.g., reading a write-only stream).
954  return ferror(file) ? -EINVAL : 0;
955  } else {
956  return ret;
957  }
958 }
959 
960 int64_t
961 ArmSemihosting::File::write(const uint8_t *buffer, uint64_t size)
962 {
963  panic_if(!file, "Trying to write to a closed file");
964 
965 
966  size_t ret = fwrite(buffer, 1, size, file);
967  if (ret == 0) {
968  // Assume errors are due to invalid file operations (e.g.,
969  // writing a read-only stream).
970  return -EINVAL;
971  } else {
972  return ret;
973  }
974 }
975 
976 int64_t
978 {
979  panic_if(!file, "Trying to seek in a closed file");
980 
981  errno = 0;
982  if (fseek(file, _pos, SEEK_SET) == 0)
983  return 0;
984  else
985  return -errno;
986 }
987 
988 int64_t
990 {
991  errno = 0;
992  long pos = ftell(file);
993  if (pos < 0)
994  return -errno;
995 
996  if (fseek(file, 0, SEEK_END) != 0)
997  return -errno;
998 
999  long len = ftell(file);
1000  if (len < 0)
1001  return -errno;
1002 
1003  if (fseek(file, pos, SEEK_SET) != 0)
1004  return -errno;
1005 
1006  return len;
1007 }
1008 
1009 
1010 void
1012 {
1014 
1015  if (!isTTY()) {
1016  long pos = file ? ftell(file) : 0;
1017  panic_if(pos < 0, "Failed to get file position.");
1018  SERIALIZE_SCALAR(pos);
1019  }
1020 }
1021 
1022 void
1024 {
1026 
1027  if (openImpl(true) < 0) {
1028  fatal("Failed to open file: %s", _name);
1029  }
1030 
1031  if (!isTTY()) {
1032  long pos = 0;
1033  UNSERIALIZE_SCALAR(pos);
1034  if (fseek(file, pos, SEEK_SET) != 0) {
1035  fatal("Failed seek to current position (%i) in '%s'", pos, _name);
1036  }
1037  }
1038 }
1039 
1040 std::ostream &
1041 operator << (std::ostream &os, const ArmSemihosting::InPlaceArg &ipa)
1042 {
1043  ccprintf(os, "[%#x-%#x)", ipa.addr, ipa.addr + ipa.size - 1);
1044  return os;
1045 }
ArmSemihosting::gatherHeapInfo
void gatherHeapInfo(ThreadContext *tc, bool aarch64, Addr &heap_base, Addr &heap_limit, Addr &stack_base, Addr &stack_limit)
Definition: semihosting.cc:547
fatal
#define fatal(...)
This implements a cprintf based fatal() function.
Definition: logging.hh:183
ArmSemihosting::FileFeatures
Implementation of the ':semihosting-features' magic file.
Definition: semihosting.hh:365
ArmSemihosting::FileBase::create
static std::unique_ptr< FileBase > create(ArmSemihosting &parent, const std::string &fname, const char *mode)
Definition: semihosting.cc:765
ArmSemihosting::retError
static RetErrno retError(SemiErrno e)
Definition: semihosting.hh:432
ArmISA::status
Bitfield< 5, 0 > status
Definition: miscregs_types.hh:417
ArmSemihosting::callWrite0
RetErrno callWrite0(ThreadContext *tc, InPlaceArg str)
Definition: semihosting.cc:345
ArmSemihosting::callWrite
RetErrno callWrite(ThreadContext *tc, Handle handle, Addr buffer, size_t size)
Definition: semihosting.cc:357
ArmSemihosting::Abi32
Definition: semihosting.hh:148
ArmSemihosting::semiErrno
SemiErrno semiErrno
Definition: semihosting.hh:258
ArmSemihosting::callExit64
RetErrno callExit64(ThreadContext *tc, uint64_t code, uint64_t subcode)
Definition: semihosting.cc:624
SimClock::Float::s
double s
These variables equal the number of ticks in the unit of time they're named after in a double.
Definition: core.cc:46
ArmSemihosting::FileBase::flen
virtual int64_t flen()
Get the length of a file in bytes.
Definition: semihosting.cc:832
X86ISA::os
Bitfield< 17 > os
Definition: misc.hh:803
warn
#define warn(...)
Definition: logging.hh:239
Request::SECURE
@ SECURE
The request targets the secure memory space.
Definition: request.hh:177
ArmSemihosting::callElapsed32
RetErrno callElapsed32(ThreadContext *tc, InPlaceArg low, InPlaceArg high)
Definition: semihosting.cc:651
ArmSemihosting::stdioMap
static const std::map< const std::string, FILE * > stdioMap
Definition: semihosting.hh:587
ArmSemihosting::callTickFreq
RetErrno callTickFreq(ThreadContext *tc)
Definition: semihosting.cc:673
SETranslatingPortProxy
Definition: se_translating_port_proxy.hh:46
ArmSemihosting::callRename
RetErrno callRename(ThreadContext *tc, Addr from_addr, size_t from_size, Addr to_addr, size_t to_size)
Definition: semihosting.cc:487
system.hh
ArmSemihosting::filesRootDir
std::string filesRootDir
Definition: semihosting.hh:405
ArmSemihosting::retOK
static RetErrno retOK(uint64_t r)
Definition: semihosting.hh:438
ArmSemihosting::callElapsed64
RetErrno callElapsed64(ThreadContext *tc, InPlaceArg ticks)
Definition: semihosting.cc:665
UNSERIALIZE_SCALAR
#define UNSERIALIZE_SCALAR(scalar)
Definition: serialize.hh:591
ArmSemihosting::callWriteC
RetErrno callWriteC(ThreadContext *tc, InPlaceArg c)
Definition: semihosting.cc:334
memory
Definition: mem.h:38
ArmISA::byteOrder
ByteOrder byteOrder(const ThreadContext *tc)
Definition: utility.hh:430
ArmSemihosting::File::~File
~File()
Definition: semihosting.cc:889
ArmISA::i
Bitfield< 7 > i
Definition: miscregs_types.hh:63
pseudo_inst.hh
ArmSemihosting::callOpen
RetErrno callOpen(ThreadContext *tc, const Addr name_base, int fmode, size_t name_size)
Definition: semihosting.cc:284
mkutctime
time_t mkutctime(struct tm *time)
Definition: time.cc:151
ArmSemihosting::File::openImpl
int64_t openImpl(bool unserialize)
Definition: semihosting.cc:896
ArmSemihosting::semiTick
uint64_t semiTick(Tick tick) const
Definition: semihosting.hh:420
time.hh
ArmSemihosting::cmdLine
const std::string cmdLine
Definition: semihosting.hh:243
ArmSemihosting::File
Definition: semihosting.hh:381
ArmSemihosting::features
static const std::vector< uint8_t > features
Definition: semihosting.hh:586
translating_port_proxy.hh
SemiPseudoAbi64::State::State
State(const ThreadContext *tc)
Definition: semihosting.cc:697
ArmSemihosting::fmodes
static const std::vector< const char * > fmodes
Definition: semihosting.hh:584
SemiPseudoAbi32
Definition: semihosting.cc:679
ArmSemihosting::callErrno
RetErrno callErrno(ThreadContext *tc)
Definition: semihosting.cc:523
ArmSemihosting::callRemove
RetErrno callRemove(ThreadContext *tc, Addr name_base, size_t name_size)
Definition: semihosting.cc:475
ArmSemihosting::FileBase::seek
virtual int64_t seek(uint64_t pos)
Seek to an absolute position in the file.
Definition: semihosting.cc:826
X86ISA::op
Bitfield< 4 > op
Definition: types.hh:79
SemiPseudoAbi32::State::State
State(const ThreadContext *tc)
Definition: semihosting.cc:684
ArmSemihosting::callExit32
RetErrno callExit32(ThreadContext *tc, InPlaceArg code)
Definition: semihosting.cc:617
ArmSemihosting::callClock
RetErrno callClock(ThreadContext *tc)
Definition: semihosting.cc:501
std::vector< const char * >
OutputDirectory::resolve
std::string resolve(const std::string &name) const
Returns relative file names prepended with name of this directory.
Definition: output.cc:201
PseudoInst::decodeAddrOffset
static void decodeAddrOffset(Addr offset, uint8_t &func)
Definition: pseudo_inst.hh:60
ArmSemihosting::call32
bool call32(ThreadContext *tc, bool gem5_ops)
Perform an Arm Semihosting call from aarch32 code.
Definition: semihosting.cc:193
FullSystem
bool FullSystem
The FullSystem variable can be used to determine the current mode of simulation.
Definition: root.cc:204
ArmSemihosting::callGetCmdLine
RetErrno callGetCmdLine(ThreadContext *tc, Addr addr, InPlaceArg size_arg)
Definition: semihosting.cc:530
ArmSemihosting::InPlaceArg::write
void write(ThreadContext *tc, uint64_t val, ByteOrder endian)
Definition: semihosting.hh:188
ArmSemihosting::callSeek
RetErrno callSeek(ThreadContext *tc, Handle handle, uint64_t pos)
Definition: semihosting.cc:425
sim_exit.hh
ArmSemihosting::SemiCall::dump32
Dumper dump32
Definition: semihosting.hh:482
output.hh
ArmSemihosting::callTime
RetErrno callTime(ThreadContext *tc)
Definition: semihosting.cc:507
System::getPhysMem
PhysicalMemory & getPhysMem()
Get a pointer to access the physical memory of the system.
Definition: system.hh:337
ArmSemihosting::File::flen
int64_t flen() override
Get the length of a file in bytes.
Definition: semihosting.cc:989
ArmSemihosting::callHeapInfo32
RetErrno callHeapInfo32(ThreadContext *tc, Addr block_addr)
Definition: semihosting.cc:588
ArmSemihosting::ArmSemihosting
ArmSemihosting(const ArmSemihostingParams &p)
Definition: semihosting.cc:141
ArmSemihosting::tmpNameIndex
uint16_t tmpNameIndex
Definition: semihosting.hh:590
SimClock::Frequency
Tick Frequency
The simulated frequency of curTick(). (In ticks per second)
Definition: core.cc:43
ArmISA::aarch64
Bitfield< 34 > aarch64
Definition: types.hh:90
PortProxy::writeBlob
void writeBlob(Addr addr, const void *p, int size) const
Same as tryWriteBlob, but insists on success.
Definition: port_proxy.hh:187
ArmSemihosting::FileFeatures::read
int64_t read(uint8_t *buffer, uint64_t size) override
Read data from file.
Definition: semihosting.cc:845
ArmSemihosting::File::File
File(ArmSemihosting &_parent, const char *name, const char *mode)
Definition: semihosting.cc:882
cp
Definition: cprintf.cc:37
ArmSemihosting::FileBase::read
virtual int64_t read(uint8_t *buffer, uint64_t size)
Read data from file.
Definition: semihosting.cc:814
ThreadContext
ThreadContext is the external interface to all thread state for anything outside of the CPU.
Definition: thread_context.hh:88
ArmISA::INTREG_R0
@ INTREG_R0
Definition: intregs.hh:54
PortProxy::readString
void readString(std::string &str, Addr addr) const
Same as tryReadString, but insists on success.
Definition: port_proxy.hh:255
AddrRange
The AddrRange class encapsulates an address range, and supports a number of tests to check if two ran...
Definition: addr_range.hh:68
GuestABI
Definition: aapcs32.hh:66
System
Definition: system.hh:73
ArmSemihosting::call64
bool call64(ThreadContext *tc, bool gem5_ops)
Perform an Arm Semihosting call from aarch64 code.
Definition: semihosting.cc:167
DPRINTF
#define DPRINTF(x,...)
Definition: trace.hh:237
ArmSemihosting::Abi32::State
Definition: semihosting.hh:152
ArmSemihosting::File::isTTY
bool isTTY() const override
Check if a file corresponds to a TTY device.
Definition: semihosting.cc:938
ArmSemihosting::calls
static const std::map< uint32_t, SemiCall > calls
Definition: semihosting.hh:583
ArmSemihosting::callSystem
RetErrno callSystem(ThreadContext *tc, Addr cmd_addr, size_t cmd_size)
Definition: semihosting.cc:513
ArmSemihosting::InPlaceArg::size
size_t size
Definition: semihosting.hh:168
ArmSemihosting::readString
std::string readString(ThreadContext *tc, Addr ptr, size_t len)
Definition: semihosting.cc:273
ArmSemihosting::callIsTTY
RetErrno callIsTTY(ThreadContext *tc, Handle handle)
Definition: semihosting.cc:411
SemiPseudoAbi64::State
Definition: semihosting.cc:694
ArmSemihosting::callHeapInfo64
RetErrno callHeapInfo64(ThreadContext *tc, Addr block_addr)
Definition: semihosting.cc:603
ArmSemihosting::exitCodes
static const std::map< uint64_t, const char * > exitCodes
Definition: semihosting.hh:585
ArmSemihosting::stackSize
const Addr stackSize
Definition: semihosting.hh:245
SimClock::Int::s
Tick s
second
Definition: core.cc:59
GuestABI::Argument
Definition: definition.hh:93
SETranslatingPortProxy::NextPage
@ NextPage
Definition: se_translating_port_proxy.hh:53
ArmISA::mode
Bitfield< 4, 0 > mode
Definition: miscregs_types.hh:70
ArmSemihosting::File::close
int64_t close() override
Close the file.
Definition: semihosting.cc:925
ArmSemihosting::semiExit
void semiExit(uint64_t code, uint64_t subcode)
Definition: semihosting.cc:639
operator<<
std::ostream & operator<<(std::ostream &os, const ArmSemihosting::InPlaceArg &ipa)
Definition: semihosting.cc:1041
exitSimLoop
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 (...
Definition: sim_events.cc:85
ArmSemihosting::InPlaceArg::addr
Addr addr
Definition: semihosting.hh:167
ArmISA::INTREG_X0
@ INTREG_X0
Definition: intregs.hh:127
serial.hh
ArmSemihosting::File::seek
int64_t seek(uint64_t pos) override
Seek to an absolute position in the file.
Definition: semihosting.cc:977
std::pair
STL pair class.
Definition: stl.hh:58
ArmSemihosting::serialize
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: semihosting.cc:219
ArmSemihosting::getSTDIO
static FILE * getSTDIO(const char *stream_name, const std::string &name, const char *mode)
Definition: semihosting.cc:748
Addr
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:148
ArmSemihosting::InPlaceArg
Definition: semihosting.hh:165
ArmSemihosting::FileFeatures::seek
int64_t seek(uint64_t pos) override
Seek to an absolute position in the file.
Definition: semihosting.cc:856
Serializable::ScopedCheckpointSection
Definition: serialize.hh:178
name
const std::string & name()
Definition: trace.cc:48
SERIALIZE_SCALAR
#define SERIALIZE_SCALAR(scalar)
Definition: serialize.hh:584
paramOut
void paramOut(CheckpointOut &cp, const std::string &name, ExtMachInst const &machInst)
Definition: types.cc:37
utility.hh
ArmSemihosting::InPlaceArg::read
uint64_t read(ThreadContext *tc, ByteOrder endian)
Definition: semihosting.hh:175
full_system.hh
ArmSemihosting::memReserve
const Addr memReserve
Definition: semihosting.hh:244
X86ISA::addr
Bitfield< 3 > addr
Definition: types.hh:80
SimObject::name
virtual const std::string name() const
Definition: sim_object.hh:182
ThreadContext::getVirtProxy
virtual PortProxy & getVirtProxy()=0
ArmSemihosting::callGem5PseudoOp32
RetErrno callGem5PseudoOp32(ThreadContext *tc, uint32_t encoded_func)
Definition: semihosting.cc:722
SemiPseudoAbi32::State
Definition: semihosting.cc:681
ArmSemihosting::FileBase::mode
std::string mode
Definition: semihosting.hh:361
ArmSemihosting::callExitExtended
RetErrno callExitExtended(ThreadContext *tc, uint64_t code, uint64_t subcode)
Definition: semihosting.cc:631
ArmSemihosting::SemiCall::call64
Dispatcher call64
Definition: semihosting.hh:479
ArmSemihosting::Abi64::State
Definition: semihosting.hh:138
warn_if
#define warn_if(cond,...)
Conditional warning macro that checks the supplied condition and only prints a warning if the conditi...
Definition: logging.hh:263
TranslatingPortProxy
This proxy attempts to translate virtual addresses using the TLBs.
Definition: translating_port_proxy.hh:55
panic_if
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
Definition: logging.hh:197
PhysicalMemory
The physical memory encapsulates all memories in the system and provides basic functionality for acce...
Definition: physical.hh:116
ArmSemihosting::portProxy
static PortProxy & portProxy(ThreadContext *tc)
Definition: semihosting.cc:246
ArmSemihosting::Handle
size_t Handle
Definition: semihosting.hh:407
inform
#define inform(...)
Definition: logging.hh:240
ArmSemihosting::MaxStandardOp
@ MaxStandardOp
Definition: semihosting.hh:226
PortProxy
This object is a proxy for a port or other object which implements the functional response protocol,...
Definition: port_proxy.hh:80
ArmSemihosting::SemiCall
Semihosting call information structure.
Definition: semihosting.hh:450
ArmSemihosting::File::serialize
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: semihosting.cc:1011
semihosting.hh
PortProxy::read
T read(Addr address) const
Read sizeof(T) bytes from address and return as object T.
Definition: port_proxy.hh:282
ArmSemihosting::File::read
int64_t read(uint8_t *buffer, uint64_t size) override
Read data from file.
Definition: semihosting.cc:946
ArmISA::t
Bitfield< 5 > t
Definition: miscregs_types.hh:67
ArmSemihosting::callFLen
RetErrno callFLen(ThreadContext *tc, Handle handle)
Definition: semihosting.cc:439
ArmSemihosting::callReadC
RetErrno callReadC(ThreadContext *tc)
Definition: semihosting.cc:399
ArmISA::len
Bitfield< 18, 16 > len
Definition: miscregs_types.hh:439
physical.hh
ArmSemihosting::FileFeatures::FileFeatures
FileFeatures(ArmSemihosting &_parent, const char *name, const char *mode)
Definition: semihosting.cc:838
ccprintf
void ccprintf(cp::Print &print)
Definition: cprintf.hh:127
logging.hh
ArmSemihosting::FileBase::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: semihosting.cc:807
ArmSemihosting::File::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: semihosting.cc:1023
ArmSemihosting::File::write
int64_t write(const uint8_t *buffer, uint64_t size) override
Write data to file.
Definition: semihosting.cc:961
ArmSemihosting::FileBase::write
virtual int64_t write(const uint8_t *buffer, uint64_t size)
Write data to file.
Definition: semihosting.cc:820
CheckpointOut
std::ostream CheckpointOut
Definition: serialize.hh:64
ArmSemihosting::AbiBase::StateBase::getAddr
Addr getAddr()
Definition: semihosting.hh:113
ArmISA::c
Bitfield< 29 > c
Definition: miscregs_types.hh:50
Serializable::currentSection
static const std::string & currentSection()
Gets the fully-qualified name of the active section.
Definition: serialize.cc:238
se_translating_port_proxy.hh
curTick
Tick curTick()
The universal simulation clock.
Definition: cur_tick.hh:43
ArmSemihosting::SemiCall::dump64
Dumper dump64
Definition: semihosting.hh:483
SemiPseudoAbi64
Definition: semihosting.cc:692
PhysicalMemory::getConfAddrRanges
AddrRangeList getConfAddrRanges() const
Get the memory ranges for all memories that are to be reported to the configuration table.
Definition: physical.cc:256
ArmISA::err
Bitfield< 6 > err
Definition: miscregs_types.hh:744
simout
OutputDirectory simout
Definition: output.cc:59
MipsISA::p
Bitfield< 0 > p
Definition: pra_constants.hh:323
std::list< AddrRange >
ArmSemihosting::callClose
RetErrno callClose(ThreadContext *tc, Handle handle)
Definition: semihosting.cc:312
ThreadContext::readIntReg
virtual RegVal readIntReg(RegIndex reg_idx) const =0
fatal_if
#define fatal_if(cond,...)
Conditional fatal macro that checks the supplied condition and only causes a fatal error if the condi...
Definition: logging.hh:219
paramIn
void paramIn(CheckpointIn &cp, const std::string &name, ExtMachInst &machInst)
Definition: types.cc:69
ArmSemihosting::files
std::vector< std::unique_ptr< FileBase > > files
Definition: semihosting.hh:406
CheckpointIn
Definition: serialize.hh:68
PortProxy::readBlob
void readBlob(Addr addr, void *p, int size) const
Higher level interfaces based on the above.
Definition: port_proxy.hh:177
ArmSemihosting::SemiCall::call32
Dispatcher call32
Definition: semihosting.hh:478
ArmSemihosting::timeBase
const time_t timeBase
Base time when the simulation started.
Definition: semihosting.hh:251
ArmSemihosting
Semihosting for AArch32 and AArch64.
Definition: semihosting.hh:72
ArmSemihosting::callTmpNam
RetErrno callTmpNam(ThreadContext *tc, Addr buffer, uint64_t id, size_t size)
Definition: semihosting.cc:453
ArmSemihosting::callRead
RetErrno callRead(ThreadContext *tc, Handle handle, Addr buffer, size_t size)
Definition: semihosting.cc:378
ArmSemihosting::FileBase::serialize
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: semihosting.cc:800
ArmSemihosting::callIsError
RetErrno callIsError(ThreadContext *tc, int64_t status)
Definition: semihosting.cc:405
csprintf
std::string csprintf(const char *format, const Args &...args)
Definition: cprintf.hh:158
ArmSemihosting::RetErrno
std::pair< uint64_t, SemiErrno > RetErrno
Definition: semihosting.hh:428
ArmSemihosting::FileFeatures::serialize
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: semihosting.cc:867
Serializable::path
static std::stack< std::string > path
Definition: serialize.hh:321
ArmSemihosting::FileBase::parent
ArmSemihosting & parent
Definition: semihosting.hh:359
ULL
#define ULL(N)
uint64_t constant
Definition: types.hh:46
PortProxy::write
void write(Addr address, const T &data) const
Write object T to address.
Definition: port_proxy.hh:291
ArmSemihosting::Abi64
Definition: semihosting.hh:134
RegVal
uint64_t RegVal
Definition: types.hh:174
ArmISA::mask
Bitfield< 28, 24 > mask
Definition: miscregs_types.hh:711
ArmISA::isSecure
bool isSecure(ThreadContext *tc)
Definition: utility.cc:112
ArmSemihosting::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: semihosting.cc:234
ArmSemihosting::FileBase
Internal state for open files.
Definition: semihosting.hh:274
byteswap.hh
ThreadContext::getSystemPtr
virtual System * getSystemPtr()=0
ArmSemihosting::callGem5PseudoOp64
RetErrno callGem5PseudoOp64(ThreadContext *tc, uint64_t encoded_func)
Definition: semihosting.cc:735
ArmISA::f
Bitfield< 6 > f
Definition: miscregs_types.hh:64
ArmSemihosting::FileFeatures::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: semihosting.cc:874
SimObject
Abstract superclass for simulation objects.
Definition: sim_object.hh:141

Generated on Tue Jun 22 2021 15:28:21 for gem5 by doxygen 1.8.17