gem5  v20.0.0.3
misc64.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2011-2013,2017-2020 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/insts/misc64.hh"
39 #include "arch/arm/isa.hh"
40 
41 std::string
43 {
44  std::stringstream ss;
45  printMnemonic(ss, "", false);
46  ccprintf(ss, "#0x%x", imm);
47  return ss.str();
48 }
49 
50 std::string
52  Addr pc, const Loader::SymbolTable *symtab) const
53 {
54  std::stringstream ss;
55  printMnemonic(ss, "", false);
56  printIntReg(ss, dest);
57  ss << ", ";
58  printIntReg(ss, op1);
59  ccprintf(ss, ", #%d, #%d", imm1, imm2);
60  return ss.str();
61 }
62 
63 std::string
65  Addr pc, const Loader::SymbolTable *symtab) const
66 {
67  std::stringstream ss;
68  printMnemonic(ss, "", false);
69  printIntReg(ss, dest);
70  ss << ", ";
71  printIntReg(ss, op1);
72  ss << ", ";
73  printIntReg(ss, op2);
74  ccprintf(ss, ", #%d", imm);
75  return ss.str();
76 }
77 
78 std::string
80  Addr pc, const Loader::SymbolTable *symtab) const
81 {
82  return csprintf("%-10s (inst %#08x)", "unknown", encoding());
83 }
84 
85 Fault
87  ExceptionLevel el, uint32_t immediate) const
88 {
90 
91  // Check for traps to supervisor (FP/SIMD regs)
92  if (el <= EL1 && checkEL1Trap(tc, misc_reg, el, ec, immediate)) {
93  return std::make_shared<SupervisorTrap>(machInst, immediate, ec);
94  }
95 
96  // Check for traps to hypervisor
97  if ((ArmSystem::haveVirtualization(tc) && el <= EL2) &&
98  checkEL2Trap(tc, misc_reg, el, ec, immediate)) {
99  return std::make_shared<HypervisorTrap>(machInst, immediate, ec);
100  }
101 
102  // Check for traps to secure monitor
103  if ((ArmSystem::haveSecurity(tc) && el <= EL3) &&
104  checkEL3Trap(tc, misc_reg, el, ec, immediate)) {
105  return std::make_shared<SecureMonitorTrap>(machInst, immediate, ec);
106  }
107 
108  return NoFault;
109 }
110 
111 bool
114  uint32_t &immediate) const
115 {
116  const CPACR cpacr = tc->readMiscReg(MISCREG_CPACR_EL1);
117 
118  bool trap_to_sup = false;
119  switch (misc_reg) {
120  case MISCREG_FPCR:
121  case MISCREG_FPSR:
122  case MISCREG_FPEXC32_EL2:
123  if ((el == EL0 && cpacr.fpen != 0x3) ||
124  (el == EL1 && !(cpacr.fpen & 0x1))) {
125  trap_to_sup = true;
126  ec = EC_TRAPPED_SIMD_FP;
127  immediate = 0x1E00000;
128  }
129  break;
130  // Generic Timer
132  trap_to_sup = el == EL0 &&
133  isGenericTimerSystemAccessTrapEL1(misc_reg, tc);
134  break;
135  default:
136  break;
137  }
138  return trap_to_sup;
139 }
140 
141 bool
144  uint32_t &immediate) const
145 {
146  const CPTR cptr = tc->readMiscReg(MISCREG_CPTR_EL2);
147  const HCR hcr = tc->readMiscReg(MISCREG_HCR_EL2);
148  const SCR scr = tc->readMiscReg(MISCREG_SCR_EL3);
149  const CPSR cpsr = tc->readMiscReg(MISCREG_CPSR);
150 
151  bool trap_to_hyp = false;
152 
153  if (!inSecureState(scr, cpsr) && (el != EL2)) {
154  switch (misc_reg) {
155  // FP/SIMD regs
156  case MISCREG_FPCR:
157  case MISCREG_FPSR:
158  case MISCREG_FPEXC32_EL2:
159  trap_to_hyp = cptr.tfp;
160  ec = EC_TRAPPED_SIMD_FP;
161  immediate = 0x1E00000;
162  break;
163  // CPACR
164  case MISCREG_CPACR_EL1:
165  trap_to_hyp = cptr.tcpac && el == EL1;
166  break;
167  // Virtual memory control regs
168  case MISCREG_SCTLR_EL1:
169  case MISCREG_TTBR0_EL1:
170  case MISCREG_TTBR1_EL1:
171  case MISCREG_TCR_EL1:
172  case MISCREG_ESR_EL1:
173  case MISCREG_FAR_EL1:
174  case MISCREG_AFSR0_EL1:
175  case MISCREG_AFSR1_EL1:
176  case MISCREG_MAIR_EL1:
177  case MISCREG_AMAIR_EL1:
179  trap_to_hyp =
180  ((hcr.trvm && miscRead) || (hcr.tvm && !miscRead)) &&
181  el == EL1;
182  break;
183  // TLB maintenance instructions
196  trap_to_hyp = hcr.ttlb && el == EL1;
197  break;
198  // Cache maintenance instructions to the point of unification
199  case MISCREG_IC_IVAU_Xt:
200  case MISCREG_ICIALLU:
201  case MISCREG_ICIALLUIS:
202  case MISCREG_DC_CVAU_Xt:
203  trap_to_hyp = hcr.tpu && el <= EL1;
204  break;
205  // Data/Unified cache maintenance instructions to the
206  // point of coherency
207  case MISCREG_DC_IVAC_Xt:
208  case MISCREG_DC_CIVAC_Xt:
209  case MISCREG_DC_CVAC_Xt:
210  trap_to_hyp = hcr.tpc && el <= EL1;
211  break;
212  // Data/Unified cache maintenance instructions by set/way
213  case MISCREG_DC_ISW_Xt:
214  case MISCREG_DC_CSW_Xt:
215  case MISCREG_DC_CISW_Xt:
216  trap_to_hyp = hcr.tsw && el == EL1;
217  break;
218  // ACTLR
219  case MISCREG_ACTLR_EL1:
220  trap_to_hyp = hcr.tacr && el == EL1;
221  break;
222 
233  trap_to_hyp = el==EL1 && hcr.apk == 0;
234  break;
235  // @todo: Trap implementation-dependent functionality based on
236  // hcr.tidcp
237 
238  // ID regs, group 3
239  case MISCREG_ID_PFR0_EL1:
240  case MISCREG_ID_PFR1_EL1:
241  case MISCREG_ID_DFR0_EL1:
242  case MISCREG_ID_AFR0_EL1:
253  case MISCREG_MVFR0_EL1:
254  case MISCREG_MVFR1_EL1:
255  case MISCREG_MVFR2_EL1:
267  assert(miscRead);
268  trap_to_hyp = hcr.tid3 && el == EL1;
269  break;
270  // ID regs, group 2
271  case MISCREG_CTR_EL0:
272  case MISCREG_CCSIDR_EL1:
273  case MISCREG_CLIDR_EL1:
274  case MISCREG_CSSELR_EL1:
275  trap_to_hyp = hcr.tid2 && el <= EL1;
276  break;
277  // ID regs, group 1
278  case MISCREG_AIDR_EL1:
279  case MISCREG_REVIDR_EL1:
280  assert(miscRead);
281  trap_to_hyp = hcr.tid1 && el == EL1;
282  break;
284  trap_to_hyp = hcr.tidcp && el == EL1;
285  break;
286  // GICv3 regs
288  {
289  auto *isa = static_cast<ArmISA::ISA *>(tc->getIsaPtr());
290  if (isa->haveGICv3CpuIfc())
291  trap_to_hyp = hcr.fmo && el == EL1;
292  }
293  break;
296  {
297  auto *isa = static_cast<ArmISA::ISA *>(tc->getIsaPtr());
298  if (isa->haveGICv3CpuIfc())
299  trap_to_hyp = hcr.imo && el == EL1;
300  }
301  break;
302  // Generic Timer
304  trap_to_hyp = el <= EL1 &&
305  isGenericTimerSystemAccessTrapEL2(misc_reg, tc);
306  break;
307  default:
308  break;
309  }
310  }
311  return trap_to_hyp;
312 }
313 
314 bool
317  uint32_t &immediate) const
318 {
319  const CPTR cptr = tc->readMiscReg(MISCREG_CPTR_EL3);
320  const SCR scr = tc->readMiscReg(MISCREG_SCR_EL3);
321  bool trap_to_mon = false;
322 
323  switch (misc_reg) {
324  // FP/SIMD regs
325  case MISCREG_FPCR:
326  case MISCREG_FPSR:
327  case MISCREG_FPEXC32_EL2:
328  trap_to_mon = cptr.tfp;
329  ec = EC_TRAPPED_SIMD_FP;
330  immediate = 0x1E00000;
331  break;
332  // CPACR, CPTR
333  case MISCREG_CPACR_EL1:
334  if (el == EL1 || el == EL2) {
335  trap_to_mon = cptr.tcpac;
336  }
337  break;
338  case MISCREG_CPTR_EL2:
339  if (el == EL2) {
340  trap_to_mon = cptr.tcpac;
341  }
342  break;
353  trap_to_mon = (el==EL1 || el==EL2) && scr.apk==0 && ELIs64(tc, EL3);
354  break;
355  // Generic Timer
357  trap_to_mon = el == EL1 &&
358  isGenericTimerSystemAccessTrapEL3(misc_reg, tc);
359  break;
360  default:
361  break;
362  }
363  return trap_to_mon;
364 }
365 
366 RegVal
368 {
369  if (dest == MISCREG_SPSEL) {
370  return imm & 0x1;
371  } else if (dest == MISCREG_PAN) {
372  return (imm & 0x1) << 22;
373  } else {
374  panic("Not a valid PSTATE field register\n");
375  }
376 }
377 
378 std::string
380  Addr pc, const Loader::SymbolTable *symtab) const
381 {
382  std::stringstream ss;
383  printMnemonic(ss);
384  printMiscReg(ss, dest);
385  ss << ", ";
386  ccprintf(ss, "#0x%x", imm);
387  return ss.str();
388 }
389 
390 std::string
392  Addr pc, const Loader::SymbolTable *symtab) const
393 {
394  std::stringstream ss;
395  printMnemonic(ss);
396  printMiscReg(ss, dest);
397  ss << ", ";
398  printIntReg(ss, op1);
399  return ss.str();
400 }
401 
402 std::string
404  Addr pc, const Loader::SymbolTable *symtab) const
405 {
406  std::stringstream ss;
407  printMnemonic(ss);
408  printIntReg(ss, dest);
409  ss << ", ";
410  printMiscReg(ss, op1);
411  return ss.str();
412 }
413 
414 Fault
416  Trace::InstRecord *traceData) const
417 {
418  auto tc = xc->tcBase();
419  const CPSR cpsr = tc->readMiscReg(MISCREG_CPSR);
420  const ExceptionLevel el = (ExceptionLevel) (uint8_t) cpsr.el;
421 
422  Fault fault = trap(tc, miscReg, el, imm);
423 
424  if (fault != NoFault) {
425  return fault;
426 
427  } else if (warning) {
428  warn_once("\tinstruction '%s' unimplemented\n", fullMnemonic.c_str());
429  return NoFault;
430 
431  } else {
432  return std::make_shared<UndefinedInstruction>(machInst, false,
433  mnemonic);
434  }
435 }
436 
437 std::string
439  Addr pc, const Loader::SymbolTable *symtab) const
440 {
441  return csprintf("%-10s (implementation defined)", fullMnemonic.c_str());
442 }
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:163
void ccprintf(cp::Print &print)
Definition: cprintf.hh:127
void printMnemonic(std::ostream &os, const std::string &suffix="", bool withPred=true, bool withCond64=false, ConditionCode cond64=COND_UC) const
Definition: static_inst.cc:372
MiscRegIndex
Definition: miscregs.hh:56
decltype(nullptr) constexpr NoFault
Definition: types.hh:243
std::string generateDisassembly(Addr pc, const Loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition: misc64.cc:42
bool haveSecurity() const
Returns true if this system implements the Security Extensions.
Definition: system.hh:150
std::string generateDisassembly(Addr pc, const Loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition: misc64.cc:391
std::string generateDisassembly(Addr pc, const Loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition: misc64.cc:379
Fault trap(ThreadContext *tc, MiscRegIndex misc_reg, ExceptionLevel el, uint32_t immediate) const
Definition: misc64.cc:86
RegVal miscRegImm() const
Returns the "register view" of the immediate field.
Definition: misc64.cc:367
const char * mnemonic
Base mnemonic (e.g., "add").
Definition: static_inst.hh:246
uint64_t RegVal
Definition: types.hh:166
bool checkEL1Trap(ThreadContext *tc, const MiscRegIndex misc_reg, ExceptionLevel el, ExceptionClass &ec, uint32_t &immediate) const
Definition: misc64.cc:112
ThreadContext is the external interface to all thread state for anything outside of the CPU...
bool isGenericTimerSystemAccessTrapEL1(const MiscRegIndex miscReg, ThreadContext *tc)
Definition: utility.cc:898
bool checkEL3Trap(ThreadContext *tc, const MiscRegIndex misc_reg, ExceptionLevel el, ExceptionClass &ec, uint32_t &immediate) const
Definition: misc64.cc:315
ExceptionLevel
Definition: types.hh:583
bool ELIs64(ThreadContext *tc, ExceptionLevel el)
Definition: utility.cc:336
const ExtMachInst machInst
The binary machine instruction.
Definition: static_inst.hh:231
std::string generateDisassembly(Addr pc, const Loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition: misc64.cc:51
Bitfield< 3, 2 > el
The ExecContext is an abstract base class the provides the interface used by the ISA to manipulate th...
Definition: exec_context.hh:70
Bitfield< 4 > pc
std::string csprintf(const char *format, const Args &...args)
Definition: cprintf.hh:158
std::string generateDisassembly(Addr pc, const Loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition: misc64.cc:64
void printIntReg(std::ostream &os, RegIndex reg_idx, uint8_t opWidth=0) const
Print a register name for disassembly given the unique dependence tag number (FP or int)...
Definition: static_inst.cc:294
std::string generateDisassembly(Addr pc, const Loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition: misc64.cc:403
Bitfield< 21 > ss
Fault execute(ExecContext *xc, Trace::InstRecord *traceData) const override
Definition: misc64.cc:415
virtual ThreadContext * tcBase() const =0
Returns a pointer to the ThreadContext.
bool haveVirtualization() const
Returns true if this system implements the virtualization Extensions.
Definition: system.hh:159
virtual BaseISA * getIsaPtr()=0
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:140
#define warn_once(...)
Definition: logging.hh:212
std::string generateDisassembly(Addr pc, const Loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition: misc64.cc:79
ExceptionClass
Definition: types.hh:610
void printMiscReg(std::ostream &os, RegIndex reg_idx) const
Definition: static_inst.cc:365
uint64_t imm
Definition: misc64.hh:46
bool inSecureState(ThreadContext *tc)
Definition: utility.cc:174
virtual RegVal readMiscReg(RegIndex misc_reg)=0
std::shared_ptr< FaultBase > Fault
Definition: types.hh:238
std::string generateDisassembly(Addr pc, const Loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition: misc64.cc:438
bool isGenericTimerSystemAccessTrapEL3(const MiscRegIndex miscReg, ThreadContext *tc)
Definition: utility.cc:1128
bool isGenericTimerSystemAccessTrapEL2(const MiscRegIndex miscReg, ThreadContext *tc)
Definition: utility.cc:958
MachInst encoding() const
Returns the real encoding of the instruction: the machInst field is in fact always 64 bit wide and co...
Definition: static_inst.hh:532
bool checkEL2Trap(ThreadContext *tc, const MiscRegIndex misc_reg, ExceptionLevel el, ExceptionClass &ec, uint32_t &immediate) const
Definition: misc64.cc:142

Generated on Fri Jul 3 2020 15:42:39 for gem5 by doxygen 1.8.13