util_internal.h 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. /*
  2. * Copyright (c) 2011-2014, Wind River Systems, Inc.
  3. * Copyright (c) 2020, Nordic Semiconductor ASA
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. */
  7. /**
  8. * @file
  9. * @brief Misc utilities
  10. *
  11. * Repetitive or obscure helper macros needed by sys/util.h.
  12. */
  13. #ifndef ZEPHYR_INCLUDE_SYS_UTIL_INTERNAL_H_
  14. #define ZEPHYR_INCLUDE_SYS_UTIL_INTERNAL_H_
  15. #include "util_loops.h"
  16. /* IS_ENABLED() helpers */
  17. /* This is called from IS_ENABLED(), and sticks on a "_XXXX" prefix,
  18. * it will now be "_XXXX1" if config_macro is "1", or just "_XXXX" if it's
  19. * undefined.
  20. * ENABLED: Z_IS_ENABLED2(_XXXX1)
  21. * DISABLED Z_IS_ENABLED2(_XXXX)
  22. */
  23. #define Z_IS_ENABLED1(config_macro) Z_IS_ENABLED2(_XXXX##config_macro)
  24. /* Here's the core trick, we map "_XXXX1" to "_YYYY," (i.e. a string
  25. * with a trailing comma), so it has the effect of making this a
  26. * two-argument tuple to the preprocessor only in the case where the
  27. * value is defined to "1"
  28. * ENABLED: _YYYY, <--- note comma!
  29. * DISABLED: _XXXX
  30. */
  31. #define _XXXX1 _YYYY,
  32. /* Then we append an extra argument to fool the gcc preprocessor into
  33. * accepting it as a varargs macro.
  34. * arg1 arg2 arg3
  35. * ENABLED: Z_IS_ENABLED3(_YYYY, 1, 0)
  36. * DISABLED Z_IS_ENABLED3(_XXXX 1, 0)
  37. */
  38. #define Z_IS_ENABLED2(one_or_two_args) Z_IS_ENABLED3(one_or_two_args 1, 0)
  39. /* And our second argument is thus now cooked to be 1 in the case
  40. * where the value is defined to 1, and 0 if not:
  41. */
  42. #define Z_IS_ENABLED3(ignore_this, val, ...) val
  43. /* Used internally by COND_CODE_1 and COND_CODE_0. */
  44. #define Z_COND_CODE_1(_flag, _if_1_code, _else_code) \
  45. __COND_CODE(_XXXX##_flag, _if_1_code, _else_code)
  46. #define Z_COND_CODE_0(_flag, _if_0_code, _else_code) \
  47. __COND_CODE(_ZZZZ##_flag, _if_0_code, _else_code)
  48. #define _ZZZZ0 _YYYY,
  49. #define __COND_CODE(one_or_two_args, _if_code, _else_code) \
  50. __GET_ARG2_DEBRACKET(one_or_two_args _if_code, _else_code)
  51. /* Gets second argument and removes brackets around that argument. It
  52. * is expected that the parameter is provided in brackets/parentheses.
  53. */
  54. #define __GET_ARG2_DEBRACKET(ignore_this, val, ...) __DEBRACKET val
  55. /* Used to remove brackets from around a single argument. */
  56. #define __DEBRACKET(...) __VA_ARGS__
  57. /* Used by IS_EMPTY() */
  58. #define Z_IS_EMPTY_(...) Z_IS_EMPTY__(__VA_ARGS__)
  59. #define Z_IS_EMPTY__(a, ...) Z_IS_EMPTY___(_ZZ##a##ZZ0, __VA_ARGS__)
  60. #define Z_IS_EMPTY___(...) GET_ARG_N(3, __VA_ARGS__)
  61. /* Used by LIST_DROP_EMPTY() */
  62. /* Adding ',' after each element would add empty element at the end of
  63. * list, which is hard to remove, so instead precede each element with ',',
  64. * this way first element is empty, and this one is easy to drop.
  65. */
  66. #define Z_LIST_ADD_ELEM(e) EMPTY, e
  67. #define Z_LIST_DROP_FIRST(...) GET_ARGS_LESS_N(1, __VA_ARGS__)
  68. #define Z_LIST_NO_EMPTIES(e) \
  69. COND_CODE_1(IS_EMPTY(e), (), (Z_LIST_ADD_ELEM(e)))
  70. #define UTIL_CAT(a, ...) UTIL_PRIMITIVE_CAT(a, __VA_ARGS__)
  71. #define UTIL_PRIMITIVE_CAT(a, ...) a##__VA_ARGS__
  72. #define UTIL_CHECK_N(x, n, ...) n
  73. #define UTIL_CHECK(...) UTIL_CHECK_N(__VA_ARGS__, 0,)
  74. #define UTIL_NOT(x) UTIL_CHECK(UTIL_PRIMITIVE_CAT(UTIL_NOT_, x))
  75. #define UTIL_NOT_0 ~, 1,
  76. #define UTIL_COMPL(b) UTIL_PRIMITIVE_CAT(UTIL_COMPL_, b)
  77. #define UTIL_COMPL_0 1
  78. #define UTIL_COMPL_1 0
  79. #define UTIL_BOOL(x) UTIL_COMPL(UTIL_NOT(x))
  80. #define UTIL_EVAL(...) __VA_ARGS__
  81. #define UTIL_EXPAND(...) __VA_ARGS__
  82. #define UTIL_REPEAT(...) UTIL_LISTIFY(__VA_ARGS__)
  83. /* Implementation details for NUM_VA_ARGS_LESS_1 */
  84. #define NUM_VA_ARGS_LESS_1_IMPL( \
  85. _ignored, \
  86. _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, \
  87. _11, _12, _13, _14, _15, _16, _17, _18, _19, _20, \
  88. _21, _22, _23, _24, _25, _26, _27, _28, _29, _30, \
  89. _31, _32, _33, _34, _35, _36, _37, _38, _39, _40, \
  90. _41, _42, _43, _44, _45, _46, _47, _48, _49, _50, \
  91. _51, _52, _53, _54, _55, _56, _57, _58, _59, _60, \
  92. _61, _62, N, ...) N
  93. /* Used by MACRO_MAP_CAT */
  94. #define MACRO_MAP_CAT_(...) \
  95. /* To make sure it works also for 2 arguments in total */ \
  96. MACRO_MAP_CAT_N(NUM_VA_ARGS_LESS_1(__VA_ARGS__), __VA_ARGS__)
  97. #define MACRO_MAP_CAT_N_(N, ...) UTIL_CAT(MACRO_MC_, N)(__VA_ARGS__,)
  98. #define MACRO_MC_0(...)
  99. #define MACRO_MC_1(m, a, ...) m(a)
  100. #define MACRO_MC_2(m, a, ...) UTIL_CAT(m(a), MACRO_MC_1(m, __VA_ARGS__,))
  101. #define MACRO_MC_3(m, a, ...) UTIL_CAT(m(a), MACRO_MC_2(m, __VA_ARGS__,))
  102. #define MACRO_MC_4(m, a, ...) UTIL_CAT(m(a), MACRO_MC_3(m, __VA_ARGS__,))
  103. #define MACRO_MC_5(m, a, ...) UTIL_CAT(m(a), MACRO_MC_4(m, __VA_ARGS__,))
  104. #define MACRO_MC_6(m, a, ...) UTIL_CAT(m(a), MACRO_MC_5(m, __VA_ARGS__,))
  105. #define MACRO_MC_7(m, a, ...) UTIL_CAT(m(a), MACRO_MC_6(m, __VA_ARGS__,))
  106. #define MACRO_MC_8(m, a, ...) UTIL_CAT(m(a), MACRO_MC_7(m, __VA_ARGS__,))
  107. #define MACRO_MC_9(m, a, ...) UTIL_CAT(m(a), MACRO_MC_8(m, __VA_ARGS__,))
  108. #define MACRO_MC_10(m, a, ...) UTIL_CAT(m(a), MACRO_MC_9(m, __VA_ARGS__,))
  109. #define MACRO_MC_11(m, a, ...) UTIL_CAT(m(a), MACRO_MC_10(m, __VA_ARGS__,))
  110. #define MACRO_MC_12(m, a, ...) UTIL_CAT(m(a), MACRO_MC_11(m, __VA_ARGS__,))
  111. #define MACRO_MC_13(m, a, ...) UTIL_CAT(m(a), MACRO_MC_12(m, __VA_ARGS__,))
  112. #define MACRO_MC_14(m, a, ...) UTIL_CAT(m(a), MACRO_MC_13(m, __VA_ARGS__,))
  113. #define MACRO_MC_15(m, a, ...) UTIL_CAT(m(a), MACRO_MC_14(m, __VA_ARGS__,))
  114. #endif /* ZEPHYR_INCLUDE_SYS_UTIL_INTERNAL_H_ */