test_fpu_glue.c 1.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Test cases for using floating point operations inside a kernel module.
  4. *
  5. * This tests kernel_fpu_begin() and kernel_fpu_end() functions, especially
  6. * when userland has modified the floating point control registers. The kernel
  7. * state might depend on the state set by the userland thread that was active
  8. * before a syscall.
  9. *
  10. * To facilitate the test, this module registers file
  11. * /sys/kernel/debug/selftest_helpers/test_fpu, which when read causes a
  12. * sequence of floating point operations. If the operations fail, either the
  13. * read returns error status or the kernel crashes.
  14. * If the operations succeed, the read returns "1\n".
  15. */
  16. #include <linux/module.h>
  17. #include <linux/kernel.h>
  18. #include <linux/debugfs.h>
  19. #include <linux/fpu.h>
  20. #include "test_fpu.h"
  21. static int test_fpu_get(void *data, u64 *val)
  22. {
  23. int status = -EINVAL;
  24. kernel_fpu_begin();
  25. status = test_fpu();
  26. kernel_fpu_end();
  27. *val = 1;
  28. return status;
  29. }
  30. DEFINE_DEBUGFS_ATTRIBUTE(test_fpu_fops, test_fpu_get, NULL, "%lld\n");
  31. static struct dentry *selftest_dir;
  32. static int __init test_fpu_init(void)
  33. {
  34. if (!kernel_fpu_available())
  35. return -EINVAL;
  36. selftest_dir = debugfs_create_dir("selftest_helpers", NULL);
  37. if (IS_ERR(selftest_dir))
  38. return -ENOMEM;
  39. debugfs_create_file_unsafe("test_fpu", 0444, selftest_dir, NULL,
  40. &test_fpu_fops);
  41. return 0;
  42. }
  43. static void __exit test_fpu_exit(void)
  44. {
  45. debugfs_remove(selftest_dir);
  46. }
  47. module_init(test_fpu_init);
  48. module_exit(test_fpu_exit);
  49. MODULE_DESCRIPTION("Test cases for floating point operations");
  50. MODULE_LICENSE("GPL");