test_overflow.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283
  1. // SPDX-License-Identifier: GPL-2.0 OR MIT
  2. /*
  3. * Test cases for arithmetic overflow checks.
  4. */
  5. #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
  6. #include <linux/init.h>
  7. #include <linux/kernel.h>
  8. #include <linux/module.h>
  9. #include <linux/overflow.h>
  10. #include <linux/types.h>
  11. #define DEFINE_TEST_ARRAY(t) \
  12. static const struct test_ ## t { \
  13. t a, b; \
  14. t sum, diff, prod; \
  15. bool s_of, d_of, p_of; \
  16. } t ## _tests[] __initconst
  17. DEFINE_TEST_ARRAY(u8) = {
  18. {0, 0, 0, 0, 0, false, false, false},
  19. {1, 1, 2, 0, 1, false, false, false},
  20. {0, 1, 1, U8_MAX, 0, false, true, false},
  21. {1, 0, 1, 1, 0, false, false, false},
  22. {0, U8_MAX, U8_MAX, 1, 0, false, true, false},
  23. {U8_MAX, 0, U8_MAX, U8_MAX, 0, false, false, false},
  24. {1, U8_MAX, 0, 2, U8_MAX, true, true, false},
  25. {U8_MAX, 1, 0, U8_MAX-1, U8_MAX, true, false, false},
  26. {U8_MAX, U8_MAX, U8_MAX-1, 0, 1, true, false, true},
  27. {U8_MAX, U8_MAX-1, U8_MAX-2, 1, 2, true, false, true},
  28. {U8_MAX-1, U8_MAX, U8_MAX-2, U8_MAX, 2, true, true, true},
  29. {1U << 3, 1U << 3, 1U << 4, 0, 1U << 6, false, false, false},
  30. {1U << 4, 1U << 4, 1U << 5, 0, 0, false, false, true},
  31. {1U << 4, 1U << 3, 3*(1U << 3), 1U << 3, 1U << 7, false, false, false},
  32. {1U << 7, 1U << 7, 0, 0, 0, true, false, true},
  33. {48, 32, 80, 16, 0, false, false, true},
  34. {128, 128, 0, 0, 0, true, false, true},
  35. {123, 234, 101, 145, 110, true, true, true},
  36. };
  37. DEFINE_TEST_ARRAY(u16) = {
  38. {0, 0, 0, 0, 0, false, false, false},
  39. {1, 1, 2, 0, 1, false, false, false},
  40. {0, 1, 1, U16_MAX, 0, false, true, false},
  41. {1, 0, 1, 1, 0, false, false, false},
  42. {0, U16_MAX, U16_MAX, 1, 0, false, true, false},
  43. {U16_MAX, 0, U16_MAX, U16_MAX, 0, false, false, false},
  44. {1, U16_MAX, 0, 2, U16_MAX, true, true, false},
  45. {U16_MAX, 1, 0, U16_MAX-1, U16_MAX, true, false, false},
  46. {U16_MAX, U16_MAX, U16_MAX-1, 0, 1, true, false, true},
  47. {U16_MAX, U16_MAX-1, U16_MAX-2, 1, 2, true, false, true},
  48. {U16_MAX-1, U16_MAX, U16_MAX-2, U16_MAX, 2, true, true, true},
  49. {1U << 7, 1U << 7, 1U << 8, 0, 1U << 14, false, false, false},
  50. {1U << 8, 1U << 8, 1U << 9, 0, 0, false, false, true},
  51. {1U << 8, 1U << 7, 3*(1U << 7), 1U << 7, 1U << 15, false, false, false},
  52. {1U << 15, 1U << 15, 0, 0, 0, true, false, true},
  53. {123, 234, 357, 65425, 28782, false, true, false},
  54. {1234, 2345, 3579, 64425, 10146, false, true, true},
  55. };
  56. DEFINE_TEST_ARRAY(u32) = {
  57. {0, 0, 0, 0, 0, false, false, false},
  58. {1, 1, 2, 0, 1, false, false, false},
  59. {0, 1, 1, U32_MAX, 0, false, true, false},
  60. {1, 0, 1, 1, 0, false, false, false},
  61. {0, U32_MAX, U32_MAX, 1, 0, false, true, false},
  62. {U32_MAX, 0, U32_MAX, U32_MAX, 0, false, false, false},
  63. {1, U32_MAX, 0, 2, U32_MAX, true, true, false},
  64. {U32_MAX, 1, 0, U32_MAX-1, U32_MAX, true, false, false},
  65. {U32_MAX, U32_MAX, U32_MAX-1, 0, 1, true, false, true},
  66. {U32_MAX, U32_MAX-1, U32_MAX-2, 1, 2, true, false, true},
  67. {U32_MAX-1, U32_MAX, U32_MAX-2, U32_MAX, 2, true, true, true},
  68. {1U << 15, 1U << 15, 1U << 16, 0, 1U << 30, false, false, false},
  69. {1U << 16, 1U << 16, 1U << 17, 0, 0, false, false, true},
  70. {1U << 16, 1U << 15, 3*(1U << 15), 1U << 15, 1U << 31, false, false, false},
  71. {1U << 31, 1U << 31, 0, 0, 0, true, false, true},
  72. {-2U, 1U, -1U, -3U, -2U, false, false, false},
  73. {-4U, 5U, 1U, -9U, -20U, true, false, true},
  74. };
  75. DEFINE_TEST_ARRAY(u64) = {
  76. {0, 0, 0, 0, 0, false, false, false},
  77. {1, 1, 2, 0, 1, false, false, false},
  78. {0, 1, 1, U64_MAX, 0, false, true, false},
  79. {1, 0, 1, 1, 0, false, false, false},
  80. {0, U64_MAX, U64_MAX, 1, 0, false, true, false},
  81. {U64_MAX, 0, U64_MAX, U64_MAX, 0, false, false, false},
  82. {1, U64_MAX, 0, 2, U64_MAX, true, true, false},
  83. {U64_MAX, 1, 0, U64_MAX-1, U64_MAX, true, false, false},
  84. {U64_MAX, U64_MAX, U64_MAX-1, 0, 1, true, false, true},
  85. {U64_MAX, U64_MAX-1, U64_MAX-2, 1, 2, true, false, true},
  86. {U64_MAX-1, U64_MAX, U64_MAX-2, U64_MAX, 2, true, true, true},
  87. {1ULL << 31, 1ULL << 31, 1ULL << 32, 0, 1ULL << 62, false, false, false},
  88. {1ULL << 32, 1ULL << 32, 1ULL << 33, 0, 0, false, false, true},
  89. {1ULL << 32, 1ULL << 31, 3*(1ULL << 31), 1ULL << 31, 1ULL << 63, false, false, false},
  90. {1ULL << 63, 1ULL << 63, 0, 0, 0, true, false, true},
  91. {1000000000ULL /* 10^9 */, 10000000000ULL /* 10^10 */,
  92. 11000000000ULL, 18446744064709551616ULL, 10000000000000000000ULL,
  93. false, true, false},
  94. {-15ULL, 10ULL, -5ULL, -25ULL, -150ULL, false, false, true},
  95. };
  96. DEFINE_TEST_ARRAY(s8) = {
  97. {0, 0, 0, 0, 0, false, false, false},
  98. {0, S8_MAX, S8_MAX, -S8_MAX, 0, false, false, false},
  99. {S8_MAX, 0, S8_MAX, S8_MAX, 0, false, false, false},
  100. {0, S8_MIN, S8_MIN, S8_MIN, 0, false, true, false},
  101. {S8_MIN, 0, S8_MIN, S8_MIN, 0, false, false, false},
  102. {-1, S8_MIN, S8_MAX, S8_MAX, S8_MIN, true, false, true},
  103. {S8_MIN, -1, S8_MAX, -S8_MAX, S8_MIN, true, false, true},
  104. {-1, S8_MAX, S8_MAX-1, S8_MIN, -S8_MAX, false, false, false},
  105. {S8_MAX, -1, S8_MAX-1, S8_MIN, -S8_MAX, false, true, false},
  106. {-1, -S8_MAX, S8_MIN, S8_MAX-1, S8_MAX, false, false, false},
  107. {-S8_MAX, -1, S8_MIN, S8_MIN+2, S8_MAX, false, false, false},
  108. {1, S8_MIN, -S8_MAX, -S8_MAX, S8_MIN, false, true, false},
  109. {S8_MIN, 1, -S8_MAX, S8_MAX, S8_MIN, false, true, false},
  110. {1, S8_MAX, S8_MIN, S8_MIN+2, S8_MAX, true, false, false},
  111. {S8_MAX, 1, S8_MIN, S8_MAX-1, S8_MAX, true, false, false},
  112. {S8_MIN, S8_MIN, 0, 0, 0, true, false, true},
  113. {S8_MAX, S8_MAX, -2, 0, 1, true, false, true},
  114. {-4, -32, -36, 28, -128, false, false, true},
  115. {-4, 32, 28, -36, -128, false, false, false},
  116. };
  117. DEFINE_TEST_ARRAY(s16) = {
  118. {0, 0, 0, 0, 0, false, false, false},
  119. {0, S16_MAX, S16_MAX, -S16_MAX, 0, false, false, false},
  120. {S16_MAX, 0, S16_MAX, S16_MAX, 0, false, false, false},
  121. {0, S16_MIN, S16_MIN, S16_MIN, 0, false, true, false},
  122. {S16_MIN, 0, S16_MIN, S16_MIN, 0, false, false, false},
  123. {-1, S16_MIN, S16_MAX, S16_MAX, S16_MIN, true, false, true},
  124. {S16_MIN, -1, S16_MAX, -S16_MAX, S16_MIN, true, false, true},
  125. {-1, S16_MAX, S16_MAX-1, S16_MIN, -S16_MAX, false, false, false},
  126. {S16_MAX, -1, S16_MAX-1, S16_MIN, -S16_MAX, false, true, false},
  127. {-1, -S16_MAX, S16_MIN, S16_MAX-1, S16_MAX, false, false, false},
  128. {-S16_MAX, -1, S16_MIN, S16_MIN+2, S16_MAX, false, false, false},
  129. {1, S16_MIN, -S16_MAX, -S16_MAX, S16_MIN, false, true, false},
  130. {S16_MIN, 1, -S16_MAX, S16_MAX, S16_MIN, false, true, false},
  131. {1, S16_MAX, S16_MIN, S16_MIN+2, S16_MAX, true, false, false},
  132. {S16_MAX, 1, S16_MIN, S16_MAX-1, S16_MAX, true, false, false},
  133. {S16_MIN, S16_MIN, 0, 0, 0, true, false, true},
  134. {S16_MAX, S16_MAX, -2, 0, 1, true, false, true},
  135. };
  136. DEFINE_TEST_ARRAY(s32) = {
  137. {0, 0, 0, 0, 0, false, false, false},
  138. {0, S32_MAX, S32_MAX, -S32_MAX, 0, false, false, false},
  139. {S32_MAX, 0, S32_MAX, S32_MAX, 0, false, false, false},
  140. {0, S32_MIN, S32_MIN, S32_MIN, 0, false, true, false},
  141. {S32_MIN, 0, S32_MIN, S32_MIN, 0, false, false, false},
  142. {-1, S32_MIN, S32_MAX, S32_MAX, S32_MIN, true, false, true},
  143. {S32_MIN, -1, S32_MAX, -S32_MAX, S32_MIN, true, false, true},
  144. {-1, S32_MAX, S32_MAX-1, S32_MIN, -S32_MAX, false, false, false},
  145. {S32_MAX, -1, S32_MAX-1, S32_MIN, -S32_MAX, false, true, false},
  146. {-1, -S32_MAX, S32_MIN, S32_MAX-1, S32_MAX, false, false, false},
  147. {-S32_MAX, -1, S32_MIN, S32_MIN+2, S32_MAX, false, false, false},
  148. {1, S32_MIN, -S32_MAX, -S32_MAX, S32_MIN, false, true, false},
  149. {S32_MIN, 1, -S32_MAX, S32_MAX, S32_MIN, false, true, false},
  150. {1, S32_MAX, S32_MIN, S32_MIN+2, S32_MAX, true, false, false},
  151. {S32_MAX, 1, S32_MIN, S32_MAX-1, S32_MAX, true, false, false},
  152. {S32_MIN, S32_MIN, 0, 0, 0, true, false, true},
  153. {S32_MAX, S32_MAX, -2, 0, 1, true, false, true},
  154. };
  155. DEFINE_TEST_ARRAY(s64) = {
  156. {0, 0, 0, 0, 0, false, false, false},
  157. {0, S64_MAX, S64_MAX, -S64_MAX, 0, false, false, false},
  158. {S64_MAX, 0, S64_MAX, S64_MAX, 0, false, false, false},
  159. {0, S64_MIN, S64_MIN, S64_MIN, 0, false, true, false},
  160. {S64_MIN, 0, S64_MIN, S64_MIN, 0, false, false, false},
  161. {-1, S64_MIN, S64_MAX, S64_MAX, S64_MIN, true, false, true},
  162. {S64_MIN, -1, S64_MAX, -S64_MAX, S64_MIN, true, false, true},
  163. {-1, S64_MAX, S64_MAX-1, S64_MIN, -S64_MAX, false, false, false},
  164. {S64_MAX, -1, S64_MAX-1, S64_MIN, -S64_MAX, false, true, false},
  165. {-1, -S64_MAX, S64_MIN, S64_MAX-1, S64_MAX, false, false, false},
  166. {-S64_MAX, -1, S64_MIN, S64_MIN+2, S64_MAX, false, false, false},
  167. {1, S64_MIN, -S64_MAX, -S64_MAX, S64_MIN, false, true, false},
  168. {S64_MIN, 1, -S64_MAX, S64_MAX, S64_MIN, false, true, false},
  169. {1, S64_MAX, S64_MIN, S64_MIN+2, S64_MAX, true, false, false},
  170. {S64_MAX, 1, S64_MIN, S64_MAX-1, S64_MAX, true, false, false},
  171. {S64_MIN, S64_MIN, 0, 0, 0, true, false, true},
  172. {S64_MAX, S64_MAX, -2, 0, 1, true, false, true},
  173. {-1, -1, -2, 0, 1, false, false, false},
  174. {-1, -128, -129, 127, 128, false, false, false},
  175. {-128, -1, -129, -127, 128, false, false, false},
  176. {0, -S64_MAX, -S64_MAX, S64_MAX, 0, false, false, false},
  177. };
  178. #define check_one_op(t, fmt, op, sym, a, b, r, of) do { \
  179. t _r; \
  180. bool _of; \
  181. \
  182. _of = check_ ## op ## _overflow(a, b, &_r); \
  183. if (_of != of) { \
  184. pr_warn("expected "fmt" "sym" "fmt \
  185. " to%s overflow (type %s)\n", \
  186. a, b, of ? "" : " not", #t); \
  187. } \
  188. if (_r != r) { \
  189. pr_warn("expected "fmt" "sym" "fmt" == " \
  190. fmt", got "fmt" (type %s)\n", \
  191. a, b, r, _r, #t); \
  192. } \
  193. } while (0)
  194. #define DEFINE_TEST_FUNC(t, fmt) \
  195. static void __init do_test_ ## t(const struct test_ ## t *p) \
  196. { \
  197. check_one_op(t, fmt, add, "+", p->a, p->b, p->sum, p->s_of); \
  198. check_one_op(t, fmt, add, "+", p->b, p->a, p->sum, p->s_of); \
  199. check_one_op(t, fmt, sub, "-", p->a, p->b, p->diff, p->d_of); \
  200. check_one_op(t, fmt, mul, "*", p->a, p->b, p->prod, p->p_of); \
  201. check_one_op(t, fmt, mul, "*", p->b, p->a, p->prod, p->p_of); \
  202. } \
  203. \
  204. static void __init test_ ## t ## _overflow(void) { \
  205. unsigned i; \
  206. \
  207. pr_info("%-3s: %zu tests\n", #t, ARRAY_SIZE(t ## _tests)); \
  208. for (i = 0; i < ARRAY_SIZE(t ## _tests); ++i) \
  209. do_test_ ## t(&t ## _tests[i]); \
  210. }
  211. DEFINE_TEST_FUNC(u8, "%d");
  212. DEFINE_TEST_FUNC(s8, "%d");
  213. DEFINE_TEST_FUNC(u16, "%d");
  214. DEFINE_TEST_FUNC(s16, "%d");
  215. DEFINE_TEST_FUNC(u32, "%u");
  216. DEFINE_TEST_FUNC(s32, "%d");
  217. #if BITS_PER_LONG == 64
  218. DEFINE_TEST_FUNC(u64, "%llu");
  219. DEFINE_TEST_FUNC(s64, "%lld");
  220. #endif
  221. static int __init test_overflow(void)
  222. {
  223. test_u8_overflow();
  224. test_s8_overflow();
  225. test_u16_overflow();
  226. test_s16_overflow();
  227. test_u32_overflow();
  228. test_s32_overflow();
  229. #if BITS_PER_LONG == 64
  230. test_u64_overflow();
  231. test_s64_overflow();
  232. #endif
  233. pr_info("done\n");
  234. return 0;
  235. }
  236. static void __exit test_module_exit(void)
  237. { }
  238. module_init(test_overflow);
  239. module_exit(test_module_exit);
  240. MODULE_LICENSE("Dual MIT/GPL");