fastops_test.c 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. #include "test_util.h"
  3. #include "kvm_util.h"
  4. #include "processor.h"
  5. /*
  6. * Execute a fastop() instruction, with or without forced emulation. BT bit 0
  7. * to set RFLAGS.CF based on whether or not the input is even or odd, so that
  8. * instructions like ADC and SBB are deterministic.
  9. */
  10. #define fastop(__insn) \
  11. "bt $0, %[bt_val]\n\t" \
  12. __insn "\n\t" \
  13. "pushfq\n\t" \
  14. "pop %[flags]\n\t"
  15. #define flags_constraint(flags_val) [flags]"=r"(flags_val)
  16. #define bt_constraint(__bt_val) [bt_val]"rm"((uint32_t)__bt_val)
  17. #define guest_execute_fastop_1(FEP, insn, __val, __flags) \
  18. ({ \
  19. __asm__ __volatile__(fastop(FEP insn " %[val]") \
  20. : [val]"+r"(__val), flags_constraint(__flags) \
  21. : bt_constraint(__val) \
  22. : "cc", "memory"); \
  23. })
  24. #define guest_test_fastop_1(insn, type_t, __val) \
  25. ({ \
  26. type_t val = __val, ex_val = __val, input = __val; \
  27. uint64_t flags, ex_flags; \
  28. \
  29. guest_execute_fastop_1("", insn, ex_val, ex_flags); \
  30. guest_execute_fastop_1(KVM_FEP, insn, val, flags); \
  31. \
  32. __GUEST_ASSERT(val == ex_val, \
  33. "Wanted 0x%lx for '%s 0x%lx', got 0x%lx", \
  34. (uint64_t)ex_val, insn, (uint64_t)input, (uint64_t)val); \
  35. __GUEST_ASSERT(flags == ex_flags, \
  36. "Wanted flags 0x%lx for '%s 0x%lx', got 0x%lx", \
  37. ex_flags, insn, (uint64_t)input, flags); \
  38. })
  39. #define guest_execute_fastop_2(FEP, insn, __input, __output, __flags) \
  40. ({ \
  41. __asm__ __volatile__(fastop(FEP insn " %[input], %[output]") \
  42. : [output]"+r"(__output), flags_constraint(__flags) \
  43. : [input]"r"(__input), bt_constraint(__output) \
  44. : "cc", "memory"); \
  45. })
  46. #define guest_test_fastop_2(insn, type_t, __val1, __val2) \
  47. ({ \
  48. type_t input = __val1, input2 = __val2, output = __val2, ex_output = __val2; \
  49. uint64_t flags, ex_flags; \
  50. \
  51. guest_execute_fastop_2("", insn, input, ex_output, ex_flags); \
  52. guest_execute_fastop_2(KVM_FEP, insn, input, output, flags); \
  53. \
  54. __GUEST_ASSERT(output == ex_output, \
  55. "Wanted 0x%lx for '%s 0x%lx 0x%lx', got 0x%lx", \
  56. (uint64_t)ex_output, insn, (uint64_t)input, \
  57. (uint64_t)input2, (uint64_t)output); \
  58. __GUEST_ASSERT(flags == ex_flags, \
  59. "Wanted flags 0x%lx for '%s 0x%lx, 0x%lx', got 0x%lx", \
  60. ex_flags, insn, (uint64_t)input, (uint64_t)input2, flags); \
  61. })
  62. #define guest_execute_fastop_cl(FEP, insn, __shift, __output, __flags) \
  63. ({ \
  64. __asm__ __volatile__(fastop(FEP insn " %%cl, %[output]") \
  65. : [output]"+r"(__output), flags_constraint(__flags) \
  66. : "c"(__shift), bt_constraint(__output) \
  67. : "cc", "memory"); \
  68. })
  69. #define guest_test_fastop_cl(insn, type_t, __val1, __val2) \
  70. ({ \
  71. type_t output = __val2, ex_output = __val2, input = __val2; \
  72. uint8_t shift = __val1; \
  73. uint64_t flags, ex_flags; \
  74. \
  75. guest_execute_fastop_cl("", insn, shift, ex_output, ex_flags); \
  76. guest_execute_fastop_cl(KVM_FEP, insn, shift, output, flags); \
  77. \
  78. __GUEST_ASSERT(output == ex_output, \
  79. "Wanted 0x%lx for '%s 0x%x, 0x%lx', got 0x%lx", \
  80. (uint64_t)ex_output, insn, shift, (uint64_t)input, \
  81. (uint64_t)output); \
  82. __GUEST_ASSERT(flags == ex_flags, \
  83. "Wanted flags 0x%lx for '%s 0x%x, 0x%lx', got 0x%lx", \
  84. ex_flags, insn, shift, (uint64_t)input, flags); \
  85. })
  86. #define guest_execute_fastop_div(__KVM_ASM_SAFE, insn, __a, __d, __rm, __flags) \
  87. ({ \
  88. uint64_t ign_error_code; \
  89. uint8_t vector; \
  90. \
  91. __asm__ __volatile__(fastop(__KVM_ASM_SAFE(insn " %[denom]")) \
  92. : "+a"(__a), "+d"(__d), flags_constraint(__flags), \
  93. KVM_ASM_SAFE_OUTPUTS(vector, ign_error_code) \
  94. : [denom]"rm"(__rm), bt_constraint(__rm) \
  95. : "cc", "memory", KVM_ASM_SAFE_CLOBBERS); \
  96. vector; \
  97. })
  98. #define guest_test_fastop_div(insn, type_t, __val1, __val2) \
  99. ({ \
  100. type_t _a = __val1, _d = __val1, rm = __val2; \
  101. type_t a = _a, d = _d, ex_a = _a, ex_d = _d; \
  102. uint64_t flags, ex_flags; \
  103. uint8_t v, ex_v; \
  104. \
  105. ex_v = guest_execute_fastop_div(KVM_ASM_SAFE, insn, ex_a, ex_d, rm, ex_flags); \
  106. v = guest_execute_fastop_div(KVM_ASM_SAFE_FEP, insn, a, d, rm, flags); \
  107. \
  108. GUEST_ASSERT_EQ(v, ex_v); \
  109. __GUEST_ASSERT(v == ex_v, \
  110. "Wanted vector 0x%x for '%s 0x%lx:0x%lx/0x%lx', got 0x%x", \
  111. ex_v, insn, (uint64_t)_a, (uint64_t)_d, (uint64_t)rm, v); \
  112. __GUEST_ASSERT(a == ex_a && d == ex_d, \
  113. "Wanted 0x%lx:0x%lx for '%s 0x%lx:0x%lx/0x%lx', got 0x%lx:0x%lx",\
  114. (uint64_t)ex_a, (uint64_t)ex_d, insn, (uint64_t)_a, \
  115. (uint64_t)_d, (uint64_t)rm, (uint64_t)a, (uint64_t)d); \
  116. __GUEST_ASSERT(v || ex_v || (flags == ex_flags), \
  117. "Wanted flags 0x%lx for '%s 0x%lx:0x%lx/0x%lx', got 0x%lx", \
  118. ex_flags, insn, (uint64_t)_a, (uint64_t)_d, (uint64_t)rm, flags);\
  119. })
  120. static const uint64_t vals[] = {
  121. 0,
  122. 1,
  123. 2,
  124. 4,
  125. 7,
  126. 0x5555555555555555,
  127. 0xaaaaaaaaaaaaaaaa,
  128. 0xfefefefefefefefe,
  129. 0xffffffffffffffff,
  130. };
  131. #define guest_test_fastops(type_t, suffix) \
  132. do { \
  133. int i, j; \
  134. \
  135. for (i = 0; i < ARRAY_SIZE(vals); i++) { \
  136. guest_test_fastop_1("dec" suffix, type_t, vals[i]); \
  137. guest_test_fastop_1("inc" suffix, type_t, vals[i]); \
  138. guest_test_fastop_1("neg" suffix, type_t, vals[i]); \
  139. guest_test_fastop_1("not" suffix, type_t, vals[i]); \
  140. \
  141. for (j = 0; j < ARRAY_SIZE(vals); j++) { \
  142. guest_test_fastop_2("add" suffix, type_t, vals[i], vals[j]); \
  143. guest_test_fastop_2("adc" suffix, type_t, vals[i], vals[j]); \
  144. guest_test_fastop_2("and" suffix, type_t, vals[i], vals[j]); \
  145. if (sizeof(type_t) != 1) { \
  146. guest_test_fastop_2("bsf" suffix, type_t, vals[i], vals[j]); \
  147. guest_test_fastop_2("bsr" suffix, type_t, vals[i], vals[j]); \
  148. guest_test_fastop_2("bt" suffix, type_t, vals[i], vals[j]); \
  149. guest_test_fastop_2("btc" suffix, type_t, vals[i], vals[j]); \
  150. guest_test_fastop_2("btr" suffix, type_t, vals[i], vals[j]); \
  151. guest_test_fastop_2("bts" suffix, type_t, vals[i], vals[j]); \
  152. guest_test_fastop_2("imul" suffix, type_t, vals[i], vals[j]); \
  153. } \
  154. guest_test_fastop_2("cmp" suffix, type_t, vals[i], vals[j]); \
  155. guest_test_fastop_2("or" suffix, type_t, vals[i], vals[j]); \
  156. guest_test_fastop_2("sbb" suffix, type_t, vals[i], vals[j]); \
  157. guest_test_fastop_2("sub" suffix, type_t, vals[i], vals[j]); \
  158. guest_test_fastop_2("test" suffix, type_t, vals[i], vals[j]); \
  159. guest_test_fastop_2("xor" suffix, type_t, vals[i], vals[j]); \
  160. \
  161. guest_test_fastop_cl("rol" suffix, type_t, vals[i], vals[j]); \
  162. guest_test_fastop_cl("ror" suffix, type_t, vals[i], vals[j]); \
  163. guest_test_fastop_cl("rcl" suffix, type_t, vals[i], vals[j]); \
  164. guest_test_fastop_cl("rcr" suffix, type_t, vals[i], vals[j]); \
  165. guest_test_fastop_cl("sar" suffix, type_t, vals[i], vals[j]); \
  166. guest_test_fastop_cl("shl" suffix, type_t, vals[i], vals[j]); \
  167. guest_test_fastop_cl("shr" suffix, type_t, vals[i], vals[j]); \
  168. \
  169. guest_test_fastop_div("div" suffix, type_t, vals[i], vals[j]); \
  170. } \
  171. } \
  172. } while (0)
  173. static void guest_code(void)
  174. {
  175. guest_test_fastops(uint8_t, "b");
  176. guest_test_fastops(uint16_t, "w");
  177. guest_test_fastops(uint32_t, "l");
  178. guest_test_fastops(uint64_t, "q");
  179. GUEST_DONE();
  180. }
  181. int main(int argc, char *argv[])
  182. {
  183. struct kvm_vcpu *vcpu;
  184. struct kvm_vm *vm;
  185. TEST_REQUIRE(is_forced_emulation_enabled);
  186. vm = vm_create_with_one_vcpu(&vcpu, guest_code);
  187. vcpu_run(vcpu);
  188. TEST_ASSERT_EQ(get_ucall(vcpu, NULL), UCALL_DONE);
  189. kvm_vm_free(vm);
  190. }