46 using namespace ArmISA;
52 printMnemonic(
ss,
"",
false);
62 printMnemonic(
ss,
"",
false);
63 printIntReg(
ss, dest);
75 printMnemonic(
ss,
"",
false);
76 printIntReg(
ss, dest);
99 if (
el <=
EL1 && checkEL1Trap(tc, misc_reg,
el,
ec, immediate)) {
100 return std::make_shared<SupervisorTrap>(machInst, immediate,
ec);
105 checkEL2Trap(tc, misc_reg,
el,
ec, immediate)) {
106 return std::make_shared<HypervisorTrap>(machInst, immediate,
ec);
111 checkEL3Trap(tc, misc_reg,
el,
ec, immediate)) {
112 return std::make_shared<SecureMonitorTrap>(machInst, immediate,
ec);
121 uint32_t &immediate)
const
128 bool trap_to_sup =
false;
131 trap_to_sup = !scr.ns && !scr.eel2 && !sctlr.uma &&
el ==
EL0;
132 trap_to_sup = trap_to_sup ||
133 (
el ==
EL0 && (scr.ns || scr.eel2) && !hcr.tge && !sctlr.uma);
142 trap_to_sup =
el ==
EL0 && !sctlr.uci;
147 if ((
el ==
EL0 && cpacr.fpen != 0x3) ||
148 (
el ==
EL1 && !(cpacr.fpen & 0x1))) {
151 immediate = 0x1E00000;
155 trap_to_sup = !sctlr.uci && (!hcr.tge || (!scr.ns && !scr.eel2)) &&
159 trap_to_sup =
el ==
EL0 && !sctlr.uct &&
160 (!hcr.tge || (!scr.ns && !scr.eel2));
165 trap_to_sup =
el ==
EL0 && mdscr.tdcc &&
166 (hcr.tge == 0x0 || ( scr.ns == 0x0));
170 trap_to_sup =
el ==
EL1 && ((cpacr.zen & 0x1) == 0x0);
174 trap_to_sup =
el ==
EL0 &&
186 uint32_t &immediate)
const
195 bool trap_to_hyp =
false;
205 if (isa->haveGICv3CpuIfc())
213 if (isa->haveGICv3CpuIfc())
221 bool from_el2 = (
el ==
EL2) && (scr.ns || scr.eel2) &&
223 ((!hcr.e2h && cptr.tfp) ||
224 (hcr.e2h && (cptr.fpen == 0x0 ||
226 bool from_el1 = (
el ==
EL1) && hcr.nv &&
227 (!hcr.e2h || (hcr.e2h && !hcr.tge));
228 trap_to_hyp = from_el2 || from_el1;
230 immediate = 0x1E00000;
248 bool tvm = miscRead? hcr.trvm: hcr.tvm;
268 (hcr.nv && (hcr.nv1 || !hcr.nv2));
310 if (
el ==
EL0 && el2_en) {
311 const bool in_host = hcr.e2h && hcr.tge;
312 const bool general_trap = el2_en && !in_host && hcr.tge &&
314 const bool tpu_trap = el2_en && !in_host && hcr.tpu;
315 const bool host_trap = el2_en && in_host && !sctlr2.uci;
316 trap_to_hyp = general_trap || tpu_trap || host_trap;
318 else if (
el ==
EL1 && el2_en) {
319 trap_to_hyp = hcr.tpu;
331 if (
el ==
EL0 && el2_en) {
333 const bool in_host = hcr.e2h && hcr.tge;
334 const bool general_trap = el2_en && !in_host && hcr.tge &&
336 const bool tpc_trap = el2_en && !in_host && hcr.tpc;
337 const bool host_trap = el2_en && in_host && !sctlr2.uci;
338 trap_to_hyp = general_trap || tpc_trap || host_trap;
339 }
else if (
el ==
EL1 && el2_en) {
340 trap_to_hyp = hcr.tpc;
400 if (
el ==
EL0 && el2_en) {
401 const bool in_host = hcr.e2h && hcr.tge;
402 const bool general_trap = el2_en && !in_host && hcr.tge &&
404 const bool tid_trap = el2_en && !in_host && hcr.tid2;
405 const bool host_trap = el2_en && in_host && !sctlr2.uct;
406 trap_to_hyp = general_trap || tid_trap || host_trap;
407 }
else if (
el ==
EL1 && el2_en) {
408 trap_to_hyp = hcr.tid2;
429 (hcr.tge && (hcr.e2h || !sctlr.uma));
540 if (
el ==
EL0 && el2_en) {
541 const bool in_host = hcr.e2h && hcr.tge;
542 const bool general_trap = el2_en && !in_host && hcr.tge &&
544 const bool tdz_trap = el2_en && !in_host && hcr.tdz;
545 const bool host_trap = el2_en && in_host && !sctlr2.dze;
546 trap_to_hyp = general_trap || tdz_trap || host_trap;
547 }
else if (
el ==
EL1 && el2_en) {
548 trap_to_hyp = hcr.tdz;
622 ELIs64(tc,
EL2) && ((!hcr.e2h && cptr.tz) ||
623 (hcr.e2h && ((cptr.zen & 0x1) == 0x0)));
624 bool from_el2 = (
el ==
EL2) && ((!hcr.e2h && cptr.tz) ||
625 (hcr.e2h && ((cptr.zen & 0x1) == 0x0)));
626 trap_to_hyp = from_el1 || from_el2;
634 bool from_el2 = (
el ==
EL2) && ((!hcr.e2h && cptr.tz) ||
635 (hcr.e2h && ((cptr.zen & 0x1) == 0x0)));
636 trap_to_hyp = from_el1 || from_el2;
650 uint32_t &immediate)
const
656 bool trap_to_mon =
false;
663 trap_to_mon = cptr.tfp &&
ELIs64(tc,
EL3);
665 immediate = 0x1E00000;
671 (!hcr.nv2 || hcr.nv1 || !hcr.nv))) ;
678 trap_to_mon = cptr.tcpac;
702 trap_to_mon = (
el ==
EL1 ||
el ==
EL2) && scr.apk == 0 &&
707 trap_to_mon =
el ==
EL1 &&
780 trap_to_mon = !cptr.ez && ((
el ==
EL3) ||
786 trap_to_mon = !cptr.ez && ((
el ==
EL3) ||
792 trap_to_mon = !cptr.ez && (
el ==
EL3);
803 MiscRegImmOp64::miscRegImm()
const
809 return (
imm & 0x1) << 22;
811 return (
imm & 0x1) << 23;
813 panic(
"Not a valid PSTATE field register\n");
818 MiscRegImmOp64::generateDisassembly(
821 std::stringstream
ss;
823 printMiscReg(
ss, dest);
830 MiscRegRegImmOp64::generateDisassembly(
833 std::stringstream
ss;
835 printMiscReg(
ss, dest);
837 printIntReg(
ss, op1);
842 RegMiscRegImmOp64::generateDisassembly(
845 std::stringstream
ss;
847 printIntReg(
ss, dest);
849 printMiscReg(
ss, op1);
867 warn_once(
"\tinstruction '%s' unimplemented\n", fullMnemonic.c_str());
871 return std::make_shared<UndefinedInstruction>(machInst,
false,
877 MiscRegImplDefined64::generateDisassembly(
880 return csprintf(
"%-10s (implementation defined)", fullMnemonic.c_str());
884 RegNone::generateDisassembly(
887 std::stringstream
ss;
889 printIntReg(
ss, dest);
900 bool asid_16bits = ArmSystem::haveLargeAsid64(tc);
922 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
932 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
942 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
952 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
961 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
973 if (hcr.tge && hcr.e2h) {
978 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
979 TLBIVMALL tlbiOp(target_el, secure,
false);
987 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
999 if (hcr.tge && hcr.e2h) {
1004 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
1005 TLBIVMALL tlbiOp(target_el, secure,
false);
1018 static_cast<Addr>(
bits(value, 43, 0)) << 12);
1027 static_cast<Addr>(
bits(value, 43, 0)) << 12);
1039 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
1043 auto asid = asid_16bits ?
bits(value, 63, 48) :
1044 bits(value, 55, 48);
1047 static_cast<Addr>(
bits(value, 43, 0)) << 12,
1052 static_cast<Addr>(
bits(value, 43, 0)) << 12);
1064 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
1068 auto asid = asid_16bits ?
bits(value, 63, 48) :
1069 bits(value, 55, 48);
1072 static_cast<Addr>(
bits(value, 43, 0)) << 12,
1077 static_cast<Addr>(
bits(value, 43, 0)) << 12);
1087 auto asid = asid_16bits ?
bits(value, 63, 48) :
1088 bits(value, 55, 48);
1093 if (hcr.tge && hcr.e2h) {
1098 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
1099 TLBIMVA tlbiOp(target_el, secure,
1100 static_cast<Addr>(
bits(value, 43, 0)) << 12,
1111 auto asid = asid_16bits ?
bits(value, 63, 48) :
1112 bits(value, 55, 48);
1117 if (hcr.tge && hcr.e2h) {
1122 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
1123 TLBIMVA tlbiOp(target_el, secure,
1124 static_cast<Addr>(
bits(value, 43, 0)) << 12,
1134 auto asid = asid_16bits ?
bits(value, 63, 48) :
1135 bits(value, 55, 48);
1140 if (hcr.tge && hcr.e2h) {
1145 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
1154 auto asid = asid_16bits ?
bits(value, 63, 48) :
1155 bits(value, 55, 48);
1160 if (hcr.tge && hcr.e2h) {
1165 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
1181 if (hcr.tge && hcr.e2h) {
1186 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
1188 static_cast<Addr>(
bits(value, 43, 0)) << 12);
1202 if (hcr.tge && hcr.e2h) {
1207 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
1209 static_cast<Addr>(
bits(value, 43, 0)) << 12);
1221 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
1223 static_cast<Addr>(
bits(value, 35, 0)) << 12);
1235 bool secure = release->has(ArmExtension::SECURITY) && !scr.ns;
1237 static_cast<Addr>(
bits(value, 35, 0)) << 12);
1243 panic(
"Invalid TLBI\n");