e500_emulate.c 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  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. #ifdef CONFIG_KVM_E500MC
  27. static int dbell2prio(ulong param)
  28. {
  29. int msg = param & PPC_DBELL_TYPE_MASK;
  30. int prio = -1;
  31. switch (msg) {
  32. case PPC_DBELL_TYPE(PPC_DBELL):
  33. prio = BOOKE_IRQPRIO_DBELL;
  34. break;
  35. case PPC_DBELL_TYPE(PPC_DBELL_CRIT):
  36. prio = BOOKE_IRQPRIO_DBELL_CRIT;
  37. break;
  38. default:
  39. break;
  40. }
  41. return prio;
  42. }
  43. static int kvmppc_e500_emul_msgclr(struct kvm_vcpu *vcpu, int rb)
  44. {
  45. ulong param = vcpu->arch.gpr[rb];
  46. int prio = dbell2prio(param);
  47. if (prio < 0)
  48. return EMULATE_FAIL;
  49. clear_bit(prio, &vcpu->arch.pending_exceptions);
  50. return EMULATE_DONE;
  51. }
  52. static int kvmppc_e500_emul_msgsnd(struct kvm_vcpu *vcpu, int rb)
  53. {
  54. ulong param = vcpu->arch.gpr[rb];
  55. int prio = dbell2prio(rb);
  56. int pir = param & PPC_DBELL_PIR_MASK;
  57. int i;
  58. struct kvm_vcpu *cvcpu;
  59. if (prio < 0)
  60. return EMULATE_FAIL;
  61. kvm_for_each_vcpu(i, cvcpu, vcpu->kvm) {
  62. int cpir = cvcpu->arch.shared->pir;
  63. if ((param & PPC_DBELL_MSG_BRDCAST) || (cpir == pir)) {
  64. set_bit(prio, &cvcpu->arch.pending_exceptions);
  65. kvm_vcpu_kick(cvcpu);
  66. }
  67. }
  68. return EMULATE_DONE;
  69. }
  70. #endif
  71. int kvmppc_core_emulate_op(struct kvm_run *run, struct kvm_vcpu *vcpu,
  72. unsigned int inst, int *advance)
  73. {
  74. int emulated = EMULATE_DONE;
  75. int ra = get_ra(inst);
  76. int rb = get_rb(inst);
  77. int rt = get_rt(inst);
  78. switch (get_op(inst)) {
  79. case 31:
  80. switch (get_xop(inst)) {
  81. #ifdef CONFIG_KVM_E500MC
  82. case XOP_MSGSND:
  83. emulated = kvmppc_e500_emul_msgsnd(vcpu, rb);
  84. break;
  85. case XOP_MSGCLR:
  86. emulated = kvmppc_e500_emul_msgclr(vcpu, rb);
  87. break;
  88. #endif
  89. case XOP_TLBRE:
  90. emulated = kvmppc_e500_emul_tlbre(vcpu);
  91. break;
  92. case XOP_TLBWE:
  93. emulated = kvmppc_e500_emul_tlbwe(vcpu);
  94. break;
  95. case XOP_TLBSX:
  96. emulated = kvmppc_e500_emul_tlbsx(vcpu,rb);
  97. break;
  98. case XOP_TLBILX:
  99. emulated = kvmppc_e500_emul_tlbilx(vcpu, rt, ra, rb);
  100. break;
  101. case XOP_TLBIVAX:
  102. emulated = kvmppc_e500_emul_tlbivax(vcpu, ra, rb);
  103. break;
  104. default:
  105. emulated = EMULATE_FAIL;
  106. }
  107. break;
  108. default:
  109. emulated = EMULATE_FAIL;
  110. }
  111. if (emulated == EMULATE_FAIL)
  112. emulated = kvmppc_booke_emulate_op(run, vcpu, inst, advance);
  113. return emulated;
  114. }
  115. int kvmppc_core_emulate_mtspr(struct kvm_vcpu *vcpu, int sprn, ulong spr_val)
  116. {
  117. struct kvmppc_vcpu_e500 *vcpu_e500 = to_e500(vcpu);
  118. int emulated = EMULATE_DONE;
  119. switch (sprn) {
  120. #ifndef CONFIG_KVM_BOOKE_HV
  121. case SPRN_PID:
  122. kvmppc_set_pid(vcpu, spr_val);
  123. break;
  124. case SPRN_PID1:
  125. if (spr_val != 0)
  126. return EMULATE_FAIL;
  127. vcpu_e500->pid[1] = spr_val;
  128. break;
  129. case SPRN_PID2:
  130. if (spr_val != 0)
  131. return EMULATE_FAIL;
  132. vcpu_e500->pid[2] = spr_val;
  133. break;
  134. case SPRN_MAS0:
  135. vcpu->arch.shared->mas0 = spr_val;
  136. break;
  137. case SPRN_MAS1:
  138. vcpu->arch.shared->mas1 = spr_val;
  139. break;
  140. case SPRN_MAS2:
  141. vcpu->arch.shared->mas2 = spr_val;
  142. break;
  143. case SPRN_MAS3:
  144. vcpu->arch.shared->mas7_3 &= ~(u64)0xffffffff;
  145. vcpu->arch.shared->mas7_3 |= spr_val;
  146. break;
  147. case SPRN_MAS4:
  148. vcpu->arch.shared->mas4 = spr_val;
  149. break;
  150. case SPRN_MAS6:
  151. vcpu->arch.shared->mas6 = spr_val;
  152. break;
  153. case SPRN_MAS7:
  154. vcpu->arch.shared->mas7_3 &= (u64)0xffffffff;
  155. vcpu->arch.shared->mas7_3 |= (u64)spr_val << 32;
  156. break;
  157. #endif
  158. case SPRN_L1CSR0:
  159. vcpu_e500->l1csr0 = spr_val;
  160. vcpu_e500->l1csr0 &= ~(L1CSR0_DCFI | L1CSR0_CLFC);
  161. break;
  162. case SPRN_L1CSR1:
  163. vcpu_e500->l1csr1 = spr_val;
  164. break;
  165. case SPRN_HID0:
  166. vcpu_e500->hid0 = spr_val;
  167. break;
  168. case SPRN_HID1:
  169. vcpu_e500->hid1 = spr_val;
  170. break;
  171. case SPRN_MMUCSR0:
  172. emulated = kvmppc_e500_emul_mt_mmucsr0(vcpu_e500,
  173. spr_val);
  174. break;
  175. /* extra exceptions */
  176. case SPRN_IVOR32:
  177. vcpu->arch.ivor[BOOKE_IRQPRIO_SPE_UNAVAIL] = spr_val;
  178. break;
  179. case SPRN_IVOR33:
  180. vcpu->arch.ivor[BOOKE_IRQPRIO_SPE_FP_DATA] = spr_val;
  181. break;
  182. case SPRN_IVOR34:
  183. vcpu->arch.ivor[BOOKE_IRQPRIO_SPE_FP_ROUND] = spr_val;
  184. break;
  185. case SPRN_IVOR35:
  186. vcpu->arch.ivor[BOOKE_IRQPRIO_PERFORMANCE_MONITOR] = spr_val;
  187. break;
  188. #ifdef CONFIG_KVM_BOOKE_HV
  189. case SPRN_IVOR36:
  190. vcpu->arch.ivor[BOOKE_IRQPRIO_DBELL] = spr_val;
  191. break;
  192. case SPRN_IVOR37:
  193. vcpu->arch.ivor[BOOKE_IRQPRIO_DBELL_CRIT] = spr_val;
  194. break;
  195. #endif
  196. default:
  197. emulated = kvmppc_booke_emulate_mtspr(vcpu, sprn, spr_val);
  198. }
  199. return emulated;
  200. }
  201. int kvmppc_core_emulate_mfspr(struct kvm_vcpu *vcpu, int sprn, ulong *spr_val)
  202. {
  203. struct kvmppc_vcpu_e500 *vcpu_e500 = to_e500(vcpu);
  204. int emulated = EMULATE_DONE;
  205. switch (sprn) {
  206. #ifndef CONFIG_KVM_BOOKE_HV
  207. case SPRN_PID:
  208. *spr_val = vcpu_e500->pid[0];
  209. break;
  210. case SPRN_PID1:
  211. *spr_val = vcpu_e500->pid[1];
  212. break;
  213. case SPRN_PID2:
  214. *spr_val = vcpu_e500->pid[2];
  215. break;
  216. case SPRN_MAS0:
  217. *spr_val = vcpu->arch.shared->mas0;
  218. break;
  219. case SPRN_MAS1:
  220. *spr_val = vcpu->arch.shared->mas1;
  221. break;
  222. case SPRN_MAS2:
  223. *spr_val = vcpu->arch.shared->mas2;
  224. break;
  225. case SPRN_MAS3:
  226. *spr_val = (u32)vcpu->arch.shared->mas7_3;
  227. break;
  228. case SPRN_MAS4:
  229. *spr_val = vcpu->arch.shared->mas4;
  230. break;
  231. case SPRN_MAS6:
  232. *spr_val = vcpu->arch.shared->mas6;
  233. break;
  234. case SPRN_MAS7:
  235. *spr_val = vcpu->arch.shared->mas7_3 >> 32;
  236. break;
  237. #endif
  238. case SPRN_DECAR:
  239. *spr_val = vcpu->arch.decar;
  240. break;
  241. case SPRN_TLB0CFG:
  242. *spr_val = vcpu->arch.tlbcfg[0];
  243. break;
  244. case SPRN_TLB1CFG:
  245. *spr_val = vcpu->arch.tlbcfg[1];
  246. break;
  247. case SPRN_L1CSR0:
  248. *spr_val = vcpu_e500->l1csr0;
  249. break;
  250. case SPRN_L1CSR1:
  251. *spr_val = vcpu_e500->l1csr1;
  252. break;
  253. case SPRN_HID0:
  254. *spr_val = vcpu_e500->hid0;
  255. break;
  256. case SPRN_HID1:
  257. *spr_val = vcpu_e500->hid1;
  258. break;
  259. case SPRN_SVR:
  260. *spr_val = vcpu_e500->svr;
  261. break;
  262. case SPRN_MMUCSR0:
  263. *spr_val = 0;
  264. break;
  265. case SPRN_MMUCFG:
  266. *spr_val = vcpu->arch.mmucfg;
  267. break;
  268. /* extra exceptions */
  269. case SPRN_IVOR32:
  270. *spr_val = vcpu->arch.ivor[BOOKE_IRQPRIO_SPE_UNAVAIL];
  271. break;
  272. case SPRN_IVOR33:
  273. *spr_val = vcpu->arch.ivor[BOOKE_IRQPRIO_SPE_FP_DATA];
  274. break;
  275. case SPRN_IVOR34:
  276. *spr_val = vcpu->arch.ivor[BOOKE_IRQPRIO_SPE_FP_ROUND];
  277. break;
  278. case SPRN_IVOR35:
  279. *spr_val = vcpu->arch.ivor[BOOKE_IRQPRIO_PERFORMANCE_MONITOR];
  280. break;
  281. #ifdef CONFIG_KVM_BOOKE_HV
  282. case SPRN_IVOR36:
  283. *spr_val = vcpu->arch.ivor[BOOKE_IRQPRIO_DBELL];
  284. break;
  285. case SPRN_IVOR37:
  286. *spr_val = vcpu->arch.ivor[BOOKE_IRQPRIO_DBELL_CRIT];
  287. break;
  288. #endif
  289. default:
  290. emulated = kvmppc_booke_emulate_mfspr(vcpu, sprn, spr_val);
  291. }
  292. return emulated;
  293. }