test_FCMOV.c 2.3 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  1. // SPDX-License-Identifier: GPL-2.0
  2. #undef _GNU_SOURCE
  3. #define _GNU_SOURCE 1
  4. #undef __USE_GNU
  5. #define __USE_GNU 1
  6. #include <unistd.h>
  7. #include <stdlib.h>
  8. #include <string.h>
  9. #include <stdio.h>
  10. #include <signal.h>
  11. #include <sys/types.h>
  12. #include <sys/select.h>
  13. #include <sys/time.h>
  14. #include <sys/wait.h>
  15. #define TEST(insn) \
  16. long double __attribute__((noinline)) insn(long flags) \
  17. { \
  18. long double out; \
  19. asm ("\n" \
  20. " push %1""\n" \
  21. " popf""\n" \
  22. " fldpi""\n" \
  23. " fld1""\n" \
  24. " " #insn " %%st(1), %%st" "\n" \
  25. " ffree %%st(1)" "\n" \
  26. : "=t" (out) \
  27. : "r" (flags) \
  28. ); \
  29. return out; \
  30. }
  31. TEST(fcmovb)
  32. TEST(fcmove)
  33. TEST(fcmovbe)
  34. TEST(fcmovu)
  35. TEST(fcmovnb)
  36. TEST(fcmovne)
  37. TEST(fcmovnbe)
  38. TEST(fcmovnu)
  39. enum {
  40. CF = 1 << 0,
  41. PF = 1 << 2,
  42. ZF = 1 << 6,
  43. };
  44. void sighandler(int sig)
  45. {
  46. printf("[FAIL]\tGot signal %d, exiting\n", sig);
  47. exit(1);
  48. }
  49. int main(int argc, char **argv, char **envp)
  50. {
  51. int err = 0;
  52. /* SIGILL triggers on 32-bit kernels w/o fcomi emulation
  53. * when run with "no387 nofxsr". Other signals are caught
  54. * just in case.
  55. */
  56. signal(SIGILL, sighandler);
  57. signal(SIGFPE, sighandler);
  58. signal(SIGSEGV, sighandler);
  59. printf("[RUN]\tTesting fcmovCC instructions\n");
  60. /* If fcmovCC() returns 1.0, the move wasn't done */
  61. err |= !(fcmovb(0) == 1.0); err |= !(fcmovnb(0) != 1.0);
  62. err |= !(fcmove(0) == 1.0); err |= !(fcmovne(0) != 1.0);
  63. err |= !(fcmovbe(0) == 1.0); err |= !(fcmovnbe(0) != 1.0);
  64. err |= !(fcmovu(0) == 1.0); err |= !(fcmovnu(0) != 1.0);
  65. err |= !(fcmovb(CF) != 1.0); err |= !(fcmovnb(CF) == 1.0);
  66. err |= !(fcmove(CF) == 1.0); err |= !(fcmovne(CF) != 1.0);
  67. err |= !(fcmovbe(CF) != 1.0); err |= !(fcmovnbe(CF) == 1.0);
  68. err |= !(fcmovu(CF) == 1.0); err |= !(fcmovnu(CF) != 1.0);
  69. err |= !(fcmovb(ZF) == 1.0); err |= !(fcmovnb(ZF) != 1.0);
  70. err |= !(fcmove(ZF) != 1.0); err |= !(fcmovne(ZF) == 1.0);
  71. err |= !(fcmovbe(ZF) != 1.0); err |= !(fcmovnbe(ZF) == 1.0);
  72. err |= !(fcmovu(ZF) == 1.0); err |= !(fcmovnu(ZF) != 1.0);
  73. err |= !(fcmovb(PF) == 1.0); err |= !(fcmovnb(PF) != 1.0);
  74. err |= !(fcmove(PF) == 1.0); err |= !(fcmovne(PF) != 1.0);
  75. err |= !(fcmovbe(PF) == 1.0); err |= !(fcmovnbe(PF) != 1.0);
  76. err |= !(fcmovu(PF) != 1.0); err |= !(fcmovnu(PF) == 1.0);
  77. if (!err)
  78. printf("[OK]\tfcmovCC\n");
  79. else
  80. printf("[FAIL]\tfcmovCC errors: %d\n", err);
  81. return err;
  82. }