test.h 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. /*
  2. * Copyright (c) 2014 Richard Braun.
  3. *
  4. * This program is free software: you can redistribute it and/or modify
  5. * it under the terms of the GNU General Public License as published by
  6. * the Free Software Foundation, either version 3 of the License, or
  7. * (at your option) any later version.
  8. *
  9. * This program is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. * GNU General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU General Public License
  15. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  16. */
  17. #ifndef TEST_TEST_H
  18. #define TEST_TEST_H
  19. #include <stdint.h>
  20. #include <kern/fmt.h>
  21. #include <kern/init.h>
  22. #include <kern/macros.h>
  23. #include <kern/panic.h>
  24. // Test exit status.
  25. #define TEST_OK 0
  26. #define TEST_SKIPPED 1
  27. #define TEST_RUNNING 2
  28. #define TEST_FAILED 3
  29. void __init test_setup (void);
  30. /*
  31. * Tests can be classified in 2 types: inline and deferred.
  32. * Inline tests are run as they are discovered, whereas deferred tests create a
  33. * detached thread that runs once the needed subsystems are up.
  34. * In order for tests to be inline, they must only use the most basic of the
  35. * functionalities that the kernel provides, since test discovery is run very
  36. * early (Before application processors are up).
  37. */
  38. #define TEST_PREFIX test_F
  39. #define TEST_INLINE_CHAR I
  40. // Convert 'name' to 'test_FI_name'.
  41. #define TEST_INLINE(name) \
  42. int __init CONCAT (TEST_PREFIX, \
  43. CONCAT (TEST_INLINE_CHAR, CONCAT (_, name))) (void)
  44. // Convert 'name' to 'test_F_name'.
  45. #define TEST_DEFERRED(name) \
  46. int __init CONCAT (TEST_PREFIX, CONCAT (_, name)) (void)
  47. // Utilities for the test module.
  48. struct thread;
  49. int test_util_create_thr (struct thread **out, void (*fn) (void *),
  50. void *arg, const char *name);
  51. void test_thread_wait_state (struct thread *thr, uint32_t state);
  52. // Test assertions.
  53. #define test_fmt_get_spec(x) \
  54. _Generic ((x), \
  55. bool: "%d", \
  56. char: "%c", \
  57. unsigned char: "%d", \
  58. short: "%d", \
  59. unsigned short: "%d", \
  60. int: "%d", \
  61. unsigned int: "%u", \
  62. long: "%ld", \
  63. unsigned long: "%lu", \
  64. long long: "%lld", \
  65. unsigned long long: "%llu", \
  66. const char*: "%s", \
  67. default: "%p")
  68. #define test_fmt_any(x, out) \
  69. (fmt_sprintf ((out), test_fmt_get_spec (x), (x)), (out))
  70. #define test_assert_op(x, y, op) \
  71. ({ \
  72. _Auto left_ = (x); \
  73. typeof (left_) right_ = (typeof (left_))(y); \
  74. if (!(left_ op right_)) \
  75. { \
  76. char buf1_[22], buf2_[22]; \
  77. panic ("assertion failed: %s %s %s at %s:%d", \
  78. test_fmt_any (left_, buf1_), \
  79. QUOTE (op), \
  80. test_fmt_any (right_, buf2_), \
  81. __FILE__, __LINE__); \
  82. } \
  83. })
  84. #define test_assert_eq(x, y) test_assert_op (x, y, ==)
  85. #define test_assert_lt(x, y) test_assert_op (x, y, <)
  86. #define test_assert_le(x, y) test_assert_op (x, y, <=)
  87. #define test_assert_gt(x, y) test_assert_op (x, y, >)
  88. #define test_assert_ge(x, y) test_assert_op (x, y, >=)
  89. #define test_assert_ne(x, y) test_assert_op (x, y, !=)
  90. #define test_assert_zero(x) test_assert_eq ((x), (typeof (x))0)
  91. #define test_assert_nonnull(x) \
  92. ({ \
  93. _Auto tmp_ = (x); \
  94. if (! tmp_) \
  95. panic ("assertion failed at %s:%d: " QUOTE (x) " is null", \
  96. __FILE__, __LINE__); \
  97. }) \
  98. #define test_assert_streq(x, y) \
  99. ({ \
  100. const char *x_ = (const char *)(x), *y_ = (const char *)(y); \
  101. if (strcmp (x_, y_) != 0) \
  102. panic ("assertion failed: %s is not equal to %s at %s:%d", \
  103. x_, y_, __FILE__, __LINE__); \
  104. }) \
  105. #define test_assert_or(cond1, ...) \
  106. ({ \
  107. const bool conds_[] = { (cond1), ##__VA_ARGS__ }; \
  108. bool works_ = false; \
  109. for (size_t i_ = 0; i_ < ARRAY_SIZE (conds_) && !works_; ++i_) \
  110. works_ = conds_[i_]; \
  111. if (!works_) \
  112. panic ("assertion failed at %s:%d", __FILE__, __LINE__); \
  113. })
  114. #endif