mock-macro-test.c 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * KUnit test for parameter list parsing macros.
  4. *
  5. * Copyright (C) 2020, Google LLC.
  6. * Author: Brendan Higgins <[email protected]>
  7. */
  8. #include <kunit/test.h>
  9. #include <kunit/params.h>
  10. #include <kunit/mock.h>
  11. struct mock_macro_dummy_struct {
  12. int (*one_param)(struct mock_macro_dummy_struct *test_struct);
  13. int (*two_param)(struct mock_macro_dummy_struct *test_struct, int num);
  14. int (*non_first_slot_param)(
  15. int num,
  16. struct mock_macro_dummy_struct *test_struct);
  17. void *(*void_ptr_return)(struct mock_macro_dummy_struct *test_struct);
  18. };
  19. DECLARE_STRUCT_CLASS_MOCK_PREREQS(mock_macro_dummy_struct);
  20. DEFINE_STRUCT_CLASS_MOCK(METHOD(one_param), CLASS(mock_macro_dummy_struct),
  21. RETURNS(int),
  22. PARAMS(struct mock_macro_dummy_struct *));
  23. DEFINE_STRUCT_CLASS_MOCK(METHOD(two_param), CLASS(mock_macro_dummy_struct),
  24. RETURNS(int),
  25. PARAMS(struct mock_macro_dummy_struct *, int));
  26. DEFINE_STRUCT_CLASS_MOCK_HANDLE_INDEX(METHOD(non_first_slot_param),
  27. CLASS(mock_macro_dummy_struct), HANDLE_INDEX(1),
  28. RETURNS(int),
  29. PARAMS(int, struct mock_macro_dummy_struct *));
  30. DEFINE_STRUCT_CLASS_MOCK(METHOD(void_ptr_return),
  31. CLASS(mock_macro_dummy_struct),
  32. RETURNS(void *),
  33. PARAMS(struct mock_macro_dummy_struct *));
  34. static int mock_macro_dummy_struct_init(
  35. struct kunit *test, struct MOCK(mock_macro_dummy_struct) *mock_test_struct)
  36. {
  37. struct mock_macro_dummy_struct *test_struct =
  38. mock_get_trgt(mock_test_struct);
  39. test_struct->one_param = one_param;
  40. test_struct->two_param = two_param;
  41. test_struct->non_first_slot_param = non_first_slot_param;
  42. return 0;
  43. }
  44. DEFINE_STRUCT_CLASS_MOCK_INIT(mock_macro_dummy_struct,
  45. mock_macro_dummy_struct_init);
  46. DECLARE_VOID_CLASS_MOCK_HANDLE_INDEX(METHOD(test_void_ptr_func),
  47. HANDLE_INDEX(0),
  48. RETURNS(int),
  49. PARAMS(void*, int));
  50. DEFINE_VOID_CLASS_MOCK_HANDLE_INDEX(METHOD(test_void_ptr_func),
  51. HANDLE_INDEX(0),
  52. RETURNS(int),
  53. PARAMS(void*, int));
  54. DEFINE_FUNCTION_MOCK(add, RETURNS(int), PARAMS(int, int));
  55. struct mock_macro_context {
  56. struct MOCK(mock_macro_dummy_struct) *mock_test_struct;
  57. struct MOCK(void) *mock_void_ptr;
  58. };
  59. #define TO_STR_INTERNAL(...) #__VA_ARGS__
  60. #define TO_STR(...) TO_STR_INTERNAL(__VA_ARGS__)
  61. static void mock_macro_is_equal(struct kunit *test)
  62. {
  63. KUNIT_EXPECT_STREQ(test, "dropped, 1", TO_STR(EQUAL(1, 1)));
  64. KUNIT_EXPECT_EQ(test, 1, DROP_FIRST_ARG(dropped, 1, 0));
  65. KUNIT_EXPECT_EQ(test, 0, DROP_FIRST_ARG(1, 0));
  66. KUNIT_EXPECT_EQ(test, 0, DROP_FIRST_ARG(EQUAL(1, 0), 0));
  67. KUNIT_EXPECT_EQ(test, 1, IS_EQUAL(1, 1));
  68. KUNIT_EXPECT_EQ(test, 0, IS_EQUAL(1, 0));
  69. }
  70. static void mock_macro_if(struct kunit *test)
  71. {
  72. KUNIT_EXPECT_STREQ(test, "body", ""IF(1)("body"));
  73. KUNIT_EXPECT_STREQ(test, "", ""IF(0)("body"));
  74. KUNIT_EXPECT_STREQ(test, "body", ""IF(IS_EQUAL(1, 1))("body"));
  75. KUNIT_EXPECT_STREQ(test, "", ""IF(IS_EQUAL(0, 1))("body"));
  76. }
  77. static void mock_macro_apply_tokens(struct kunit *test)
  78. {
  79. KUNIT_EXPECT_STREQ(test, "type", TO_STR(APPLY_TOKENS(type, 1, 0)));
  80. KUNIT_EXPECT_STREQ(test, ", type", TO_STR(APPLY_TOKENS(type, 1, 1)));
  81. KUNIT_EXPECT_STREQ(test, "", TO_STR(APPLY_TOKENS(type, 0, 1)));
  82. }
  83. #define IDENTITY(context, type, index) type
  84. static void mock_macro_param_list_recurse(struct kunit *test)
  85. {
  86. KUNIT_EXPECT_STREQ(test, "", TO_STR(PARAM_LIST_RECURSE(0,
  87. 0,
  88. IDENTITY,
  89. FILTER_NONE,
  90. not_used)));
  91. KUNIT_EXPECT_STREQ(test,
  92. "type",
  93. TO_STR(EXPAND(PARAM_LIST_RECURSE(0,
  94. 1,
  95. IDENTITY,
  96. FILTER_NONE,
  97. not_used,
  98. type))));
  99. KUNIT_EXPECT_STREQ(test,
  100. "type0 , type1 , type2 , type3 , type4 , type5 , "
  101. "type6 , type7 , type8 , type9 , type10 , type11 , "
  102. "type12 , type13 , type14 , type15",
  103. TO_STR(EXPAND(PARAM_LIST_RECURSE(0, 16,
  104. IDENTITY,
  105. FILTER_NONE,
  106. not_used,
  107. type0, type1, type2,
  108. type3, type4, type5,
  109. type6, type7, type8,
  110. type9, type10,
  111. type11, type12,
  112. type13, type14,
  113. type15))));
  114. }
  115. static void mock_macro_for_each_param(struct kunit *test) {
  116. KUNIT_EXPECT_STREQ(test,
  117. "type0 , type1",
  118. TO_STR(FOR_EACH_PARAM(IDENTITY,
  119. FILTER_NONE,
  120. not_used,
  121. type0,
  122. type1)));
  123. KUNIT_EXPECT_STREQ(test,
  124. "type1",
  125. TO_STR(FOR_EACH_PARAM(IDENTITY,
  126. FILTER_INDEX,
  127. 0,
  128. type0,
  129. type1)));
  130. }
  131. static void mock_macro_param_list_from_types_basic(struct kunit *test)
  132. {
  133. KUNIT_EXPECT_STREQ(test, "", TO_STR(PARAM_LIST_FROM_TYPES()));
  134. KUNIT_EXPECT_STREQ(test, "int arg0",
  135. TO_STR(PARAM_LIST_FROM_TYPES(int)));
  136. KUNIT_EXPECT_STREQ(test, "struct kunit_struct * arg0 , int arg1",
  137. TO_STR(PARAM_LIST_FROM_TYPES(struct kunit_struct *,
  138. int)));
  139. KUNIT_EXPECT_STREQ(test,
  140. "type0 arg0 , type1 arg1 , type2 arg2 , type3 arg3 , "
  141. "type4 arg4 , type5 arg5 , type6 arg6 , type7 arg7 , "
  142. "type8 arg8 , type9 arg9 , type10 arg10 , "
  143. "type11 arg11 , type12 arg12 , type13 arg13 , "
  144. "type14 arg14 , type15 arg15",
  145. TO_STR(PARAM_LIST_FROM_TYPES(type0, type1, type2,
  146. type3, type4, type5,
  147. type6, type7, type8,
  148. type9, type10, type11,
  149. type12, type13, type14,
  150. type15)));
  151. }
  152. static void mock_macro_arg_names_from_types(struct kunit *test)
  153. {
  154. KUNIT_EXPECT_STREQ(test, "", TO_STR(ARG_NAMES_FROM_TYPES(0)));
  155. KUNIT_EXPECT_STREQ(test, "", TO_STR(ARG_NAMES_FROM_TYPES(0, int)));
  156. KUNIT_EXPECT_STREQ(test,
  157. "arg1",
  158. TO_STR(ARG_NAMES_FROM_TYPES(0,
  159. struct kunit_struct *,
  160. int)));
  161. KUNIT_EXPECT_STREQ(test,
  162. "arg0 , arg1 , arg3 , arg4 , arg5 , arg6 , arg7 , "
  163. "arg8 , arg9 , arg10 , arg11 , arg12 , arg13 , "
  164. "arg14 , arg15",
  165. TO_STR(ARG_NAMES_FROM_TYPES(2, type0, type1, type2,
  166. type3, type4, type5,
  167. type6, type7, type8,
  168. type9, type10, type11,
  169. type12, type13, type14,
  170. type15)));
  171. }
  172. static void mock_macro_test_generated_method_code_works(struct kunit *test)
  173. {
  174. struct mock_macro_context *ctx = test->priv;
  175. struct MOCK(mock_macro_dummy_struct) *mock_test_struct =
  176. ctx->mock_test_struct;
  177. struct mock_macro_dummy_struct *test_struct =
  178. mock_get_trgt(mock_test_struct);
  179. struct mock_expectation *handle;
  180. handle = KUNIT_EXPECT_CALL(one_param(mock_get_ctrl(mock_test_struct)));
  181. handle->action = kunit_int_return(test, 0);
  182. handle = KUNIT_EXPECT_CALL(two_param(mock_get_ctrl(mock_test_struct),
  183. kunit_int_eq(test, 5)));
  184. handle->action = kunit_int_return(test, 1);
  185. handle = KUNIT_EXPECT_CALL(non_first_slot_param(
  186. kunit_int_eq(test, 5),
  187. mock_get_ctrl(mock_test_struct)));
  188. handle->action = kunit_int_return(test, 1);
  189. test_struct->one_param(test_struct);
  190. test_struct->two_param(test_struct, 5);
  191. test_struct->non_first_slot_param(5, test_struct);
  192. }
  193. static void mock_macro_test_generated_method_void_code_works(struct kunit *test)
  194. {
  195. struct mock_macro_context *ctx = test->priv;
  196. struct MOCK(void) *mock_void_ptr = ctx->mock_void_ptr;
  197. struct mock_expectation *handle;
  198. handle = KUNIT_EXPECT_CALL(test_void_ptr_func(mock_get_ctrl(mock_void_ptr),
  199. kunit_int_eq(test, 3)));
  200. handle->action = kunit_int_return(test, 0);
  201. test_void_ptr_func(mock_void_ptr, 3);
  202. }
  203. static void mock_macro_test_generated_function_code_works(struct kunit *test)
  204. {
  205. struct mock_expectation *handle;
  206. handle = KUNIT_EXPECT_CALL(add(kunit_int_eq(test, 4), kunit_int_eq(test, 3)));
  207. handle->action = kunit_int_return(test, 7);
  208. KUNIT_EXPECT_EQ(test, 7, add(4, 3));
  209. }
  210. static int mock_macro_test_init(struct kunit *test)
  211. {
  212. struct mock_macro_context *ctx;
  213. ctx = kunit_kzalloc(test, sizeof(*ctx), GFP_KERNEL);
  214. if (!ctx)
  215. return -ENOMEM;
  216. test->priv = ctx;
  217. ctx->mock_test_struct = CONSTRUCT_MOCK(mock_macro_dummy_struct, test);
  218. if (!ctx->mock_test_struct)
  219. return -EINVAL;
  220. ctx->mock_void_ptr = CONSTRUCT_MOCK(void, test);
  221. if (!ctx->mock_void_ptr)
  222. return -EINVAL;
  223. return 0;
  224. }
  225. static struct kunit_case mock_macro_test_cases[] = {
  226. KUNIT_CASE(mock_macro_is_equal),
  227. KUNIT_CASE(mock_macro_if),
  228. KUNIT_CASE(mock_macro_apply_tokens),
  229. KUNIT_CASE(mock_macro_param_list_recurse),
  230. KUNIT_CASE(mock_macro_for_each_param),
  231. KUNIT_CASE(mock_macro_param_list_from_types_basic),
  232. KUNIT_CASE(mock_macro_arg_names_from_types),
  233. KUNIT_CASE(mock_macro_test_generated_method_code_works),
  234. KUNIT_CASE(mock_macro_test_generated_method_void_code_works),
  235. KUNIT_CASE(mock_macro_test_generated_function_code_works),
  236. {}
  237. };
  238. static struct kunit_suite mock_macro_test_suite = {
  239. .name = "mock-macro-test",
  240. .init = mock_macro_test_init,
  241. .test_cases = mock_macro_test_cases,
  242. };
  243. kunit_test_suite(mock_macro_test_suite);