e500_emulate.c 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383
  1. /*
  2. * Copyright (C) 2008-2011 Freescale Semiconductor, Inc. All rights reserved.
  3. *
  4. * Author: Yu Liu, <yu.liu@freescale.com>
  5. *
  6. * Description:
  7. * This file is derived from arch/powerpc/kvm/44x_emulate.c,
  8. * by Hollis Blanchard <hollisb@us.ibm.com>.
  9. *
  10. * This program is free software; you can redistribute it and/or modify
  11. * it under the terms of the GNU General Public License, version 2, as
  12. * published by the Free Software Foundation.
  13. */
  14. #include <asm/kvm_ppc.h>
  15. #include <asm/disassemble.h>
  16. #include <asm/dbell.h>
  17. #include "booke.h"
  18. #include "e500.h"
  19. #define XOP_MSGSND 206
  20. #define XOP_MSGCLR 238
  21. #define XOP_TLBIVAX 786
  22. #define XOP_TLBSX 914
  23. #define XOP_TLBRE 946
  24. #define XOP_TLBWE 978
  25. #define XOP_TLBILX 18
  26. #define XOP_EHPRIV 270
  27. #ifdef CONFIG_KVM_E500MC
  28. static int dbell2prio(ulong param)
  29. {
  30. int msg = param & PPC_DBELL_TYPE_MASK;
  31. int prio = -1;
  32. switch (msg) {
  33. case PPC_DBELL_TYPE(PPC_DBELL):
  34. prio = BOOKE_IRQPRIO_DBELL;
  35. break;
  36. case PPC_DBELL_TYPE(PPC_DBELL_CRIT):
  37. prio = BOOKE_IRQPRIO_DBELL_CRIT;
  38. break;
  39. default:
  40. break;
  41. }
  42. return prio;
  43. }
  44. static int kvmppc_e500_emul_msgclr(struct kvm_vcpu *vcpu, int rb)
  45. {
  46. ulong param = vcpu->arch.gpr[rb];
  47. int prio = dbell2prio(param);
  48. if (prio < 0)
  49. return EMULATE_FAIL;
  50. clear_bit(prio, &vcpu->arch.pending_exceptions);
  51. return EMULATE_DONE;
  52. }
  53. static int kvmppc_e500_emul_msgsnd(struct kvm_vcpu *vcpu, int rb)
  54. {
  55. ulong param = vcpu->arch.gpr[rb];
  56. int prio = dbell2prio(rb);
  57. int pir = param & PPC_DBELL_PIR_MASK;
  58. int i;
  59. struct kvm_vcpu *cvcpu;
  60. if (prio < 0)
  61. return EMULATE_FAIL;
  62. kvm_for_each_vcpu(i, cvcpu, vcpu->kvm) {
  63. int cpir = cvcpu->arch.shared->pir;
  64. if ((param & PPC_DBELL_MSG_BRDCAST) || (cpir == pir)) {
  65. set_bit(prio, &cvcpu->arch.pending_exceptions);
  66. kvm_vcpu_kick(cvcpu);
  67. }
  68. }
  69. return EMULATE_DONE;
  70. }
  71. #endif
  72. static int kvmppc_e500_emul_ehpriv(struct kvm_run *run, struct kvm_vcpu *vcpu,
  73. unsigned int inst, int *advance)
  74. {
  75. int emulated = EMULATE_DONE;
  76. switch (get_oc(inst)) {
  77. case EHPRIV_OC_DEBUG:
  78. run->exit_reason = KVM_EXIT_DEBUG;
  79. run->debug.arch.address = vcpu->arch.pc;
  80. run->debug.arch.status = 0;
  81. kvmppc_account_exit(vcpu, DEBUG_EXITS);
  82. emulated = EMULATE_EXIT_USER;
  83. *advance = 0;
  84. break;
  85. default:
  86. emulated = EMULATE_FAIL;
  87. }
  88. return emulated;
  89. }
  90. int kvmppc_core_emulate_op_e500(struct kvm_run *run, struct kvm_vcpu *vcpu,
  91. unsigned int inst, int *advance)
  92. {
  93. int emulated = EMULATE_DONE;
  94. int ra = get_ra(inst);
  95. int rb = get_rb(inst);
  96. int rt = get_rt(inst);
  97. gva_t ea;
  98. switch (get_op(inst)) {
  99. case 31:
  100. switch (get_xop(inst)) {
  101. #ifdef CONFIG_KVM_E500MC
  102. case XOP_MSGSND:
  103. emulated = kvmppc_e500_emul_msgsnd(vcpu, rb);
  104. break;
  105. case XOP_MSGCLR:
  106. emulated = kvmppc_e500_emul_msgclr(vcpu, rb);
  107. break;
  108. #endif
  109. case XOP_TLBRE:
  110. emulated = kvmppc_e500_emul_tlbre(vcpu);
  111. break;
  112. case XOP_TLBWE:
  113. emulated = kvmppc_e500_emul_tlbwe(vcpu);
  114. break;
  115. case XOP_TLBSX:
  116. ea = kvmppc_get_ea_indexed(vcpu, ra, rb);
  117. emulated = kvmppc_e500_emul_tlbsx(vcpu, ea);
  118. break;
  119. case XOP_TLBILX: {
  120. int type = rt & 0x3;
  121. ea = kvmppc_get_ea_indexed(vcpu, ra, rb);
  122. emulated = kvmppc_e500_emul_tlbilx(vcpu, type, ea);
  123. break;
  124. }
  125. case XOP_TLBIVAX:
  126. ea = kvmppc_get_ea_indexed(vcpu, ra, rb);
  127. emulated = kvmppc_e500_emul_tlbivax(vcpu, ea);
  128. break;
  129. case XOP_EHPRIV:
  130. emulated = kvmppc_e500_emul_ehpriv(run, vcpu, inst,
  131. advance);
  132. break;
  133. default:
  134. emulated = EMULATE_FAIL;
  135. }
  136. break;
  137. default:
  138. emulated = EMULATE_FAIL;
  139. }
  140. if (emulated == EMULATE_FAIL)
  141. emulated = kvmppc_booke_emulate_op(run, vcpu, inst, advance);
  142. return emulated;
  143. }
  144. int kvmppc_core_emulate_mtspr_e500(struct kvm_vcpu *vcpu, int sprn, ulong spr_val)
  145. {
  146. struct kvmppc_vcpu_e500 *vcpu_e500 = to_e500(vcpu);
  147. int emulated = EMULATE_DONE;
  148. switch (sprn) {
  149. #ifndef CONFIG_KVM_BOOKE_HV
  150. case SPRN_PID:
  151. kvmppc_set_pid(vcpu, spr_val);
  152. break;
  153. case SPRN_PID1:
  154. if (spr_val != 0)
  155. return EMULATE_FAIL;
  156. vcpu_e500->pid[1] = spr_val;
  157. break;
  158. case SPRN_PID2:
  159. if (spr_val != 0)
  160. return EMULATE_FAIL;
  161. vcpu_e500->pid[2] = spr_val;
  162. break;
  163. case SPRN_MAS0:
  164. vcpu->arch.shared->mas0 = spr_val;
  165. break;
  166. case SPRN_MAS1:
  167. vcpu->arch.shared->mas1 = spr_val;
  168. break;
  169. case SPRN_MAS2:
  170. vcpu->arch.shared->mas2 = spr_val;
  171. break;
  172. case SPRN_MAS3:
  173. vcpu->arch.shared->mas7_3 &= ~(u64)0xffffffff;
  174. vcpu->arch.shared->mas7_3 |= spr_val;
  175. break;
  176. case SPRN_MAS4:
  177. vcpu->arch.shared->mas4 = spr_val;
  178. break;
  179. case SPRN_MAS6:
  180. vcpu->arch.shared->mas6 = spr_val;
  181. break;
  182. case SPRN_MAS7:
  183. vcpu->arch.shared->mas7_3 &= (u64)0xffffffff;
  184. vcpu->arch.shared->mas7_3 |= (u64)spr_val << 32;
  185. break;
  186. #endif
  187. case SPRN_L1CSR0:
  188. vcpu_e500->l1csr0 = spr_val;
  189. vcpu_e500->l1csr0 &= ~(L1CSR0_DCFI | L1CSR0_CLFC);
  190. break;
  191. case SPRN_L1CSR1:
  192. vcpu_e500->l1csr1 = spr_val;
  193. break;
  194. case SPRN_HID0:
  195. vcpu_e500->hid0 = spr_val;
  196. break;
  197. case SPRN_HID1:
  198. vcpu_e500->hid1 = spr_val;
  199. break;
  200. case SPRN_MMUCSR0:
  201. emulated = kvmppc_e500_emul_mt_mmucsr0(vcpu_e500,
  202. spr_val);
  203. break;
  204. /* extra exceptions */
  205. case SPRN_IVOR32:
  206. vcpu->arch.ivor[BOOKE_IRQPRIO_SPE_UNAVAIL] = spr_val;
  207. break;
  208. case SPRN_IVOR33:
  209. vcpu->arch.ivor[BOOKE_IRQPRIO_SPE_FP_DATA] = spr_val;
  210. break;
  211. case SPRN_IVOR34:
  212. vcpu->arch.ivor[BOOKE_IRQPRIO_SPE_FP_ROUND] = spr_val;
  213. break;
  214. case SPRN_IVOR35:
  215. vcpu->arch.ivor[BOOKE_IRQPRIO_PERFORMANCE_MONITOR] = spr_val;
  216. break;
  217. #ifdef CONFIG_KVM_BOOKE_HV
  218. case SPRN_IVOR36:
  219. vcpu->arch.ivor[BOOKE_IRQPRIO_DBELL] = spr_val;
  220. break;
  221. case SPRN_IVOR37:
  222. vcpu->arch.ivor[BOOKE_IRQPRIO_DBELL_CRIT] = spr_val;
  223. break;
  224. #endif
  225. default:
  226. emulated = kvmppc_booke_emulate_mtspr(vcpu, sprn, spr_val);
  227. }
  228. return emulated;
  229. }
  230. int kvmppc_core_emulate_mfspr_e500(struct kvm_vcpu *vcpu, int sprn, ulong *spr_val)
  231. {
  232. struct kvmppc_vcpu_e500 *vcpu_e500 = to_e500(vcpu);
  233. int emulated = EMULATE_DONE;
  234. switch (sprn) {
  235. #ifndef CONFIG_KVM_BOOKE_HV
  236. case SPRN_PID:
  237. *spr_val = vcpu_e500->pid[0];
  238. break;
  239. case SPRN_PID1:
  240. *spr_val = vcpu_e500->pid[1];
  241. break;
  242. case SPRN_PID2:
  243. *spr_val = vcpu_e500->pid[2];
  244. break;
  245. case SPRN_MAS0:
  246. *spr_val = vcpu->arch.shared->mas0;
  247. break;
  248. case SPRN_MAS1:
  249. *spr_val = vcpu->arch.shared->mas1;
  250. break;
  251. case SPRN_MAS2:
  252. *spr_val = vcpu->arch.shared->mas2;
  253. break;
  254. case SPRN_MAS3:
  255. *spr_val = (u32)vcpu->arch.shared->mas7_3;
  256. break;
  257. case SPRN_MAS4:
  258. *spr_val = vcpu->arch.shared->mas4;
  259. break;
  260. case SPRN_MAS6:
  261. *spr_val = vcpu->arch.shared->mas6;
  262. break;
  263. case SPRN_MAS7:
  264. *spr_val = vcpu->arch.shared->mas7_3 >> 32;
  265. break;
  266. #endif
  267. case SPRN_DECAR:
  268. *spr_val = vcpu->arch.decar;
  269. break;
  270. case SPRN_TLB0CFG:
  271. *spr_val = vcpu->arch.tlbcfg[0];
  272. break;
  273. case SPRN_TLB1CFG:
  274. *spr_val = vcpu->arch.tlbcfg[1];
  275. break;
  276. case SPRN_TLB0PS:
  277. if (!has_feature(vcpu, VCPU_FTR_MMU_V2))
  278. return EMULATE_FAIL;
  279. *spr_val = vcpu->arch.tlbps[0];
  280. break;
  281. case SPRN_TLB1PS:
  282. if (!has_feature(vcpu, VCPU_FTR_MMU_V2))
  283. return EMULATE_FAIL;
  284. *spr_val = vcpu->arch.tlbps[1];
  285. break;
  286. case SPRN_L1CSR0:
  287. *spr_val = vcpu_e500->l1csr0;
  288. break;
  289. case SPRN_L1CSR1:
  290. *spr_val = vcpu_e500->l1csr1;
  291. break;
  292. case SPRN_HID0:
  293. *spr_val = vcpu_e500->hid0;
  294. break;
  295. case SPRN_HID1:
  296. *spr_val = vcpu_e500->hid1;
  297. break;
  298. case SPRN_SVR:
  299. *spr_val = vcpu_e500->svr;
  300. break;
  301. case SPRN_MMUCSR0:
  302. *spr_val = 0;
  303. break;
  304. case SPRN_MMUCFG:
  305. *spr_val = vcpu->arch.mmucfg;
  306. break;
  307. case SPRN_EPTCFG:
  308. if (!has_feature(vcpu, VCPU_FTR_MMU_V2))
  309. return EMULATE_FAIL;
  310. /*
  311. * Legacy Linux guests access EPTCFG register even if the E.PT
  312. * category is disabled in the VM. Give them a chance to live.
  313. */
  314. *spr_val = vcpu->arch.eptcfg;
  315. break;
  316. /* extra exceptions */
  317. case SPRN_IVOR32:
  318. *spr_val = vcpu->arch.ivor[BOOKE_IRQPRIO_SPE_UNAVAIL];
  319. break;
  320. case SPRN_IVOR33:
  321. *spr_val = vcpu->arch.ivor[BOOKE_IRQPRIO_SPE_FP_DATA];
  322. break;
  323. case SPRN_IVOR34:
  324. *spr_val = vcpu->arch.ivor[BOOKE_IRQPRIO_SPE_FP_ROUND];
  325. break;
  326. case SPRN_IVOR35:
  327. *spr_val = vcpu->arch.ivor[BOOKE_IRQPRIO_PERFORMANCE_MONITOR];
  328. break;
  329. #ifdef CONFIG_KVM_BOOKE_HV
  330. case SPRN_IVOR36:
  331. *spr_val = vcpu->arch.ivor[BOOKE_IRQPRIO_DBELL];
  332. break;
  333. case SPRN_IVOR37:
  334. *spr_val = vcpu->arch.ivor[BOOKE_IRQPRIO_DBELL_CRIT];
  335. break;
  336. #endif
  337. default:
  338. emulated = kvmppc_booke_emulate_mfspr(vcpu, sprn, spr_val);
  339. }
  340. return emulated;
  341. }