SDmetamacros.h 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667
  1. /**
  2. * Macros for metaprogramming
  3. * ExtendedC
  4. *
  5. * Copyright (C) 2012 Justin Spahr-Summers
  6. * Released under the MIT license
  7. */
  8. #ifndef EXTC_METAMACROS_H
  9. #define EXTC_METAMACROS_H
  10. /**
  11. * Executes one or more expressions (which may have a void type, such as a call
  12. * to a function that returns no value) and always returns true.
  13. */
  14. #define metamacro_exprify(...) \
  15. ((__VA_ARGS__), true)
  16. /**
  17. * Returns a string representation of VALUE after full macro expansion.
  18. */
  19. #define metamacro_stringify(VALUE) \
  20. metamacro_stringify_(VALUE)
  21. /**
  22. * Returns A and B concatenated after full macro expansion.
  23. */
  24. #define metamacro_concat(A, B) \
  25. metamacro_concat_(A, B)
  26. /**
  27. * Returns the Nth variadic argument (starting from zero). At least
  28. * N + 1 variadic arguments must be given. N must be between zero and twenty,
  29. * inclusive.
  30. */
  31. #define metamacro_at(N, ...) \
  32. metamacro_concat(metamacro_at, N)(__VA_ARGS__)
  33. /**
  34. * Returns the number of arguments (up to twenty) provided to the macro. At
  35. * least one argument must be provided.
  36. *
  37. * Inspired by P99: http://p99.gforge.inria.fr
  38. */
  39. #define metamacro_argcount(...) \
  40. metamacro_at(20, __VA_ARGS__, 20, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1)
  41. /**
  42. * Identical to #metamacro_foreach_cxt, except that no CONTEXT argument is
  43. * given. Only the index and current argument will thus be passed to MACRO.
  44. */
  45. #define metamacro_foreach(MACRO, SEP, ...) \
  46. metamacro_foreach_cxt(metamacro_foreach_iter, SEP, MACRO, __VA_ARGS__)
  47. /**
  48. * For each consecutive variadic argument (up to twenty), MACRO is passed the
  49. * zero-based index of the current argument, CONTEXT, and then the argument
  50. * itself. The results of adjoining invocations of MACRO are then separated by
  51. * SEP.
  52. *
  53. * Inspired by P99: http://p99.gforge.inria.fr
  54. */
  55. #define metamacro_foreach_cxt(MACRO, SEP, CONTEXT, ...) \
  56. metamacro_concat(metamacro_foreach_cxt, metamacro_argcount(__VA_ARGS__))(MACRO, SEP, CONTEXT, __VA_ARGS__)
  57. /**
  58. * Identical to #metamacro_foreach_cxt. This can be used when the former would
  59. * fail due to recursive macro expansion.
  60. */
  61. #define metamacro_foreach_cxt_recursive(MACRO, SEP, CONTEXT, ...) \
  62. metamacro_concat(metamacro_foreach_cxt_recursive, metamacro_argcount(__VA_ARGS__))(MACRO, SEP, CONTEXT, __VA_ARGS__)
  63. /**
  64. * In consecutive order, appends each variadic argument (up to twenty) onto
  65. * BASE. The resulting concatenations are then separated by SEP.
  66. *
  67. * This is primarily useful to manipulate a list of macro invocations into instead
  68. * invoking a different, possibly related macro.
  69. */
  70. #define metamacro_foreach_concat(BASE, SEP, ...) \
  71. metamacro_foreach_cxt(metamacro_foreach_concat_iter, SEP, BASE, __VA_ARGS__)
  72. /**
  73. * Iterates COUNT times, each time invoking MACRO with the current index
  74. * (starting at zero) and CONTEXT. The results of adjoining invocations of MACRO
  75. * are then separated by SEP.
  76. *
  77. * COUNT must be an integer between zero and twenty, inclusive.
  78. */
  79. #define metamacro_for_cxt(COUNT, MACRO, SEP, CONTEXT) \
  80. metamacro_concat(metamacro_for_cxt, COUNT)(MACRO, SEP, CONTEXT)
  81. /**
  82. * Returns the first argument given. At least one argument must be provided.
  83. *
  84. * This is useful when implementing a variadic macro, where you may have only
  85. * one variadic argument, but no way to retrieve it (for example, because \c ...
  86. * always needs to match at least one argument).
  87. *
  88. * @code
  89. #define varmacro(...) \
  90. metamacro_head(__VA_ARGS__)
  91. * @endcode
  92. */
  93. #define metamacro_head(...) \
  94. metamacro_head_(__VA_ARGS__, 0)
  95. /**
  96. * Returns every argument except the first. At least two arguments must be
  97. * provided.
  98. */
  99. #define metamacro_tail(...) \
  100. metamacro_tail_(__VA_ARGS__)
  101. /**
  102. * Returns the first N (up to twenty) variadic arguments as a new argument list.
  103. * At least N variadic arguments must be provided.
  104. */
  105. #define metamacro_take(N, ...) \
  106. metamacro_concat(metamacro_take, N)(__VA_ARGS__)
  107. /**
  108. * Removes the first N (up to twenty) variadic arguments from the given argument
  109. * list. At least N variadic arguments must be provided.
  110. */
  111. #define metamacro_drop(N, ...) \
  112. metamacro_concat(metamacro_drop, N)(__VA_ARGS__)
  113. /**
  114. * Decrements VAL, which must be a number between zero and twenty, inclusive.
  115. *
  116. * This is primarily useful when dealing with indexes and counts in
  117. * metaprogramming.
  118. */
  119. #define metamacro_dec(VAL) \
  120. metamacro_at(VAL, -1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19)
  121. /**
  122. * Increments VAL, which must be a number between zero and twenty, inclusive.
  123. *
  124. * This is primarily useful when dealing with indexes and counts in
  125. * metaprogramming.
  126. */
  127. #define metamacro_inc(VAL) \
  128. metamacro_at(VAL, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21)
  129. /**
  130. * If A is equal to B, the next argument list is expanded; otherwise, the
  131. * argument list after that is expanded. A and B must be numbers between zero
  132. * and twenty, inclusive. Additionally, B must be greater than or equal to A.
  133. *
  134. * @code
  135. // expands to true
  136. metamacro_if_eq(0, 0)(true)(false)
  137. // expands to false
  138. metamacro_if_eq(0, 1)(true)(false)
  139. * @endcode
  140. *
  141. * This is primarily useful when dealing with indexes and counts in
  142. * metaprogramming.
  143. */
  144. #define metamacro_if_eq(A, B) \
  145. metamacro_concat(metamacro_if_eq, A)(B)
  146. /**
  147. * Identical to #metamacro_if_eq. This can be used when the former would fail
  148. * due to recursive macro expansion.
  149. */
  150. #define metamacro_if_eq_recursive(A, B) \
  151. metamacro_concat(metamacro_if_eq_recursive, A)(B)
  152. /**
  153. * Returns 1 if N is an even number, or 0 otherwise. N must be between zero and
  154. * twenty, inclusive.
  155. *
  156. * For the purposes of this test, zero is considered even.
  157. */
  158. #define metamacro_is_even(N) \
  159. metamacro_at(N, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1)
  160. /**
  161. * Returns the logical NOT of B, which must be the number zero or one.
  162. */
  163. #define metamacro_not(B) \
  164. metamacro_at(B, 1, 0)
  165. // IMPLEMENTATION DETAILS FOLLOW!
  166. // Do not write code that depends on anything below this line.
  167. #define metamacro_stringify_(VALUE) # VALUE
  168. #define metamacro_concat_(A, B) A ## B
  169. #define metamacro_foreach_iter(INDEX, MACRO, ARG) MACRO(INDEX, ARG)
  170. #define metamacro_head_(FIRST, ...) FIRST
  171. #define metamacro_tail_(FIRST, ...) __VA_ARGS__
  172. #define metamacro_consume_(...)
  173. #define metamacro_expand_(...) __VA_ARGS__
  174. // implemented from scratch so that metamacro_concat() doesn't end up nesting
  175. #define metamacro_foreach_concat_iter(INDEX, BASE, ARG) metamacro_foreach_concat_iter_(BASE, ARG)
  176. #define metamacro_foreach_concat_iter_(BASE, ARG) BASE ## ARG
  177. // metamacro_at expansions
  178. #define metamacro_at0(...) metamacro_head(__VA_ARGS__)
  179. #define metamacro_at1(_0, ...) metamacro_head(__VA_ARGS__)
  180. #define metamacro_at2(_0, _1, ...) metamacro_head(__VA_ARGS__)
  181. #define metamacro_at3(_0, _1, _2, ...) metamacro_head(__VA_ARGS__)
  182. #define metamacro_at4(_0, _1, _2, _3, ...) metamacro_head(__VA_ARGS__)
  183. #define metamacro_at5(_0, _1, _2, _3, _4, ...) metamacro_head(__VA_ARGS__)
  184. #define metamacro_at6(_0, _1, _2, _3, _4, _5, ...) metamacro_head(__VA_ARGS__)
  185. #define metamacro_at7(_0, _1, _2, _3, _4, _5, _6, ...) metamacro_head(__VA_ARGS__)
  186. #define metamacro_at8(_0, _1, _2, _3, _4, _5, _6, _7, ...) metamacro_head(__VA_ARGS__)
  187. #define metamacro_at9(_0, _1, _2, _3, _4, _5, _6, _7, _8, ...) metamacro_head(__VA_ARGS__)
  188. #define metamacro_at10(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, ...) metamacro_head(__VA_ARGS__)
  189. #define metamacro_at11(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, ...) metamacro_head(__VA_ARGS__)
  190. #define metamacro_at12(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, ...) metamacro_head(__VA_ARGS__)
  191. #define metamacro_at13(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, ...) metamacro_head(__VA_ARGS__)
  192. #define metamacro_at14(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, ...) metamacro_head(__VA_ARGS__)
  193. #define metamacro_at15(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, ...) metamacro_head(__VA_ARGS__)
  194. #define metamacro_at16(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, ...) metamacro_head(__VA_ARGS__)
  195. #define metamacro_at17(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, ...) metamacro_head(__VA_ARGS__)
  196. #define metamacro_at18(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, ...) metamacro_head(__VA_ARGS__)
  197. #define metamacro_at19(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18, ...) metamacro_head(__VA_ARGS__)
  198. #define metamacro_at20(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18, _19, ...) metamacro_head(__VA_ARGS__)
  199. // metamacro_foreach_cxt expansions
  200. #define metamacro_foreach_cxt0(MACRO, SEP, CONTEXT)
  201. #define metamacro_foreach_cxt1(MACRO, SEP, CONTEXT, _0) MACRO(0, CONTEXT, _0)
  202. #define metamacro_foreach_cxt2(MACRO, SEP, CONTEXT, _0, _1) \
  203. metamacro_foreach_cxt1(MACRO, SEP, CONTEXT, _0) \
  204. SEP \
  205. MACRO(1, CONTEXT, _1)
  206. #define metamacro_foreach_cxt3(MACRO, SEP, CONTEXT, _0, _1, _2) \
  207. metamacro_foreach_cxt2(MACRO, SEP, CONTEXT, _0, _1) \
  208. SEP \
  209. MACRO(2, CONTEXT, _2)
  210. #define metamacro_foreach_cxt4(MACRO, SEP, CONTEXT, _0, _1, _2, _3) \
  211. metamacro_foreach_cxt3(MACRO, SEP, CONTEXT, _0, _1, _2) \
  212. SEP \
  213. MACRO(3, CONTEXT, _3)
  214. #define metamacro_foreach_cxt5(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4) \
  215. metamacro_foreach_cxt4(MACRO, SEP, CONTEXT, _0, _1, _2, _3) \
  216. SEP \
  217. MACRO(4, CONTEXT, _4)
  218. #define metamacro_foreach_cxt6(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5) \
  219. metamacro_foreach_cxt5(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4) \
  220. SEP \
  221. MACRO(5, CONTEXT, _5)
  222. #define metamacro_foreach_cxt7(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6) \
  223. metamacro_foreach_cxt6(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5) \
  224. SEP \
  225. MACRO(6, CONTEXT, _6)
  226. #define metamacro_foreach_cxt8(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7) \
  227. metamacro_foreach_cxt7(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6) \
  228. SEP \
  229. MACRO(7, CONTEXT, _7)
  230. #define metamacro_foreach_cxt9(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8) \
  231. metamacro_foreach_cxt8(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7) \
  232. SEP \
  233. MACRO(8, CONTEXT, _8)
  234. #define metamacro_foreach_cxt10(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9) \
  235. metamacro_foreach_cxt9(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8) \
  236. SEP \
  237. MACRO(9, CONTEXT, _9)
  238. #define metamacro_foreach_cxt11(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10) \
  239. metamacro_foreach_cxt10(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9) \
  240. SEP \
  241. MACRO(10, CONTEXT, _10)
  242. #define metamacro_foreach_cxt12(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11) \
  243. metamacro_foreach_cxt11(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10) \
  244. SEP \
  245. MACRO(11, CONTEXT, _11)
  246. #define metamacro_foreach_cxt13(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12) \
  247. metamacro_foreach_cxt12(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11) \
  248. SEP \
  249. MACRO(12, CONTEXT, _12)
  250. #define metamacro_foreach_cxt14(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13) \
  251. metamacro_foreach_cxt13(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12) \
  252. SEP \
  253. MACRO(13, CONTEXT, _13)
  254. #define metamacro_foreach_cxt15(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14) \
  255. metamacro_foreach_cxt14(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13) \
  256. SEP \
  257. MACRO(14, CONTEXT, _14)
  258. #define metamacro_foreach_cxt16(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15) \
  259. metamacro_foreach_cxt15(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14) \
  260. SEP \
  261. MACRO(15, CONTEXT, _15)
  262. #define metamacro_foreach_cxt17(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16) \
  263. metamacro_foreach_cxt16(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15) \
  264. SEP \
  265. MACRO(16, CONTEXT, _16)
  266. #define metamacro_foreach_cxt18(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17) \
  267. metamacro_foreach_cxt17(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16) \
  268. SEP \
  269. MACRO(17, CONTEXT, _17)
  270. #define metamacro_foreach_cxt19(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18) \
  271. metamacro_foreach_cxt18(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17) \
  272. SEP \
  273. MACRO(18, CONTEXT, _18)
  274. #define metamacro_foreach_cxt20(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18, _19) \
  275. metamacro_foreach_cxt19(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18) \
  276. SEP \
  277. MACRO(19, CONTEXT, _19)
  278. // metamacro_foreach_cxt_recursive expansions
  279. #define metamacro_foreach_cxt_recursive0(MACRO, SEP, CONTEXT)
  280. #define metamacro_foreach_cxt_recursive1(MACRO, SEP, CONTEXT, _0) MACRO(0, CONTEXT, _0)
  281. #define metamacro_foreach_cxt_recursive2(MACRO, SEP, CONTEXT, _0, _1) \
  282. metamacro_foreach_cxt_recursive1(MACRO, SEP, CONTEXT, _0) \
  283. SEP \
  284. MACRO(1, CONTEXT, _1)
  285. #define metamacro_foreach_cxt_recursive3(MACRO, SEP, CONTEXT, _0, _1, _2) \
  286. metamacro_foreach_cxt_recursive2(MACRO, SEP, CONTEXT, _0, _1) \
  287. SEP \
  288. MACRO(2, CONTEXT, _2)
  289. #define metamacro_foreach_cxt_recursive4(MACRO, SEP, CONTEXT, _0, _1, _2, _3) \
  290. metamacro_foreach_cxt_recursive3(MACRO, SEP, CONTEXT, _0, _1, _2) \
  291. SEP \
  292. MACRO(3, CONTEXT, _3)
  293. #define metamacro_foreach_cxt_recursive5(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4) \
  294. metamacro_foreach_cxt_recursive4(MACRO, SEP, CONTEXT, _0, _1, _2, _3) \
  295. SEP \
  296. MACRO(4, CONTEXT, _4)
  297. #define metamacro_foreach_cxt_recursive6(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5) \
  298. metamacro_foreach_cxt_recursive5(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4) \
  299. SEP \
  300. MACRO(5, CONTEXT, _5)
  301. #define metamacro_foreach_cxt_recursive7(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6) \
  302. metamacro_foreach_cxt_recursive6(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5) \
  303. SEP \
  304. MACRO(6, CONTEXT, _6)
  305. #define metamacro_foreach_cxt_recursive8(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7) \
  306. metamacro_foreach_cxt_recursive7(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6) \
  307. SEP \
  308. MACRO(7, CONTEXT, _7)
  309. #define metamacro_foreach_cxt_recursive9(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8) \
  310. metamacro_foreach_cxt_recursive8(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7) \
  311. SEP \
  312. MACRO(8, CONTEXT, _8)
  313. #define metamacro_foreach_cxt_recursive10(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9) \
  314. metamacro_foreach_cxt_recursive9(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8) \
  315. SEP \
  316. MACRO(9, CONTEXT, _9)
  317. #define metamacro_foreach_cxt_recursive11(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10) \
  318. metamacro_foreach_cxt_recursive10(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9) \
  319. SEP \
  320. MACRO(10, CONTEXT, _10)
  321. #define metamacro_foreach_cxt_recursive12(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11) \
  322. metamacro_foreach_cxt_recursive11(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10) \
  323. SEP \
  324. MACRO(11, CONTEXT, _11)
  325. #define metamacro_foreach_cxt_recursive13(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12) \
  326. metamacro_foreach_cxt_recursive12(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11) \
  327. SEP \
  328. MACRO(12, CONTEXT, _12)
  329. #define metamacro_foreach_cxt_recursive14(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13) \
  330. metamacro_foreach_cxt_recursive13(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12) \
  331. SEP \
  332. MACRO(13, CONTEXT, _13)
  333. #define metamacro_foreach_cxt_recursive15(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14) \
  334. metamacro_foreach_cxt_recursive14(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13) \
  335. SEP \
  336. MACRO(14, CONTEXT, _14)
  337. #define metamacro_foreach_cxt_recursive16(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15) \
  338. metamacro_foreach_cxt_recursive15(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14) \
  339. SEP \
  340. MACRO(15, CONTEXT, _15)
  341. #define metamacro_foreach_cxt_recursive17(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16) \
  342. metamacro_foreach_cxt_recursive16(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15) \
  343. SEP \
  344. MACRO(16, CONTEXT, _16)
  345. #define metamacro_foreach_cxt_recursive18(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17) \
  346. metamacro_foreach_cxt_recursive17(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16) \
  347. SEP \
  348. MACRO(17, CONTEXT, _17)
  349. #define metamacro_foreach_cxt_recursive19(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18) \
  350. metamacro_foreach_cxt_recursive18(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17) \
  351. SEP \
  352. MACRO(18, CONTEXT, _18)
  353. #define metamacro_foreach_cxt_recursive20(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18, _19) \
  354. metamacro_foreach_cxt_recursive19(MACRO, SEP, CONTEXT, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18) \
  355. SEP \
  356. MACRO(19, CONTEXT, _19)
  357. // metamacro_for_cxt expansions
  358. #define metamacro_for_cxt0(MACRO, SEP, CONTEXT)
  359. #define metamacro_for_cxt1(MACRO, SEP, CONTEXT) MACRO(0, CONTEXT)
  360. #define metamacro_for_cxt2(MACRO, SEP, CONTEXT) \
  361. metamacro_for_cxt1(MACRO, SEP, CONTEXT) \
  362. SEP \
  363. MACRO(1, CONTEXT)
  364. #define metamacro_for_cxt3(MACRO, SEP, CONTEXT) \
  365. metamacro_for_cxt2(MACRO, SEP, CONTEXT) \
  366. SEP \
  367. MACRO(2, CONTEXT)
  368. #define metamacro_for_cxt4(MACRO, SEP, CONTEXT) \
  369. metamacro_for_cxt3(MACRO, SEP, CONTEXT) \
  370. SEP \
  371. MACRO(3, CONTEXT)
  372. #define metamacro_for_cxt5(MACRO, SEP, CONTEXT) \
  373. metamacro_for_cxt4(MACRO, SEP, CONTEXT) \
  374. SEP \
  375. MACRO(4, CONTEXT)
  376. #define metamacro_for_cxt6(MACRO, SEP, CONTEXT) \
  377. metamacro_for_cxt5(MACRO, SEP, CONTEXT) \
  378. SEP \
  379. MACRO(5, CONTEXT)
  380. #define metamacro_for_cxt7(MACRO, SEP, CONTEXT) \
  381. metamacro_for_cxt6(MACRO, SEP, CONTEXT) \
  382. SEP \
  383. MACRO(6, CONTEXT)
  384. #define metamacro_for_cxt8(MACRO, SEP, CONTEXT) \
  385. metamacro_for_cxt7(MACRO, SEP, CONTEXT) \
  386. SEP \
  387. MACRO(7, CONTEXT)
  388. #define metamacro_for_cxt9(MACRO, SEP, CONTEXT) \
  389. metamacro_for_cxt8(MACRO, SEP, CONTEXT) \
  390. SEP \
  391. MACRO(8, CONTEXT)
  392. #define metamacro_for_cxt10(MACRO, SEP, CONTEXT) \
  393. metamacro_for_cxt9(MACRO, SEP, CONTEXT) \
  394. SEP \
  395. MACRO(9, CONTEXT)
  396. #define metamacro_for_cxt11(MACRO, SEP, CONTEXT) \
  397. metamacro_for_cxt10(MACRO, SEP, CONTEXT) \
  398. SEP \
  399. MACRO(10, CONTEXT)
  400. #define metamacro_for_cxt12(MACRO, SEP, CONTEXT) \
  401. metamacro_for_cxt11(MACRO, SEP, CONTEXT) \
  402. SEP \
  403. MACRO(11, CONTEXT)
  404. #define metamacro_for_cxt13(MACRO, SEP, CONTEXT) \
  405. metamacro_for_cxt12(MACRO, SEP, CONTEXT) \
  406. SEP \
  407. MACRO(12, CONTEXT)
  408. #define metamacro_for_cxt14(MACRO, SEP, CONTEXT) \
  409. metamacro_for_cxt13(MACRO, SEP, CONTEXT) \
  410. SEP \
  411. MACRO(13, CONTEXT)
  412. #define metamacro_for_cxt15(MACRO, SEP, CONTEXT) \
  413. metamacro_for_cxt14(MACRO, SEP, CONTEXT) \
  414. SEP \
  415. MACRO(14, CONTEXT)
  416. #define metamacro_for_cxt16(MACRO, SEP, CONTEXT) \
  417. metamacro_for_cxt15(MACRO, SEP, CONTEXT) \
  418. SEP \
  419. MACRO(15, CONTEXT)
  420. #define metamacro_for_cxt17(MACRO, SEP, CONTEXT) \
  421. metamacro_for_cxt16(MACRO, SEP, CONTEXT) \
  422. SEP \
  423. MACRO(16, CONTEXT)
  424. #define metamacro_for_cxt18(MACRO, SEP, CONTEXT) \
  425. metamacro_for_cxt17(MACRO, SEP, CONTEXT) \
  426. SEP \
  427. MACRO(17, CONTEXT)
  428. #define metamacro_for_cxt19(MACRO, SEP, CONTEXT) \
  429. metamacro_for_cxt18(MACRO, SEP, CONTEXT) \
  430. SEP \
  431. MACRO(18, CONTEXT)
  432. #define metamacro_for_cxt20(MACRO, SEP, CONTEXT) \
  433. metamacro_for_cxt19(MACRO, SEP, CONTEXT) \
  434. SEP \
  435. MACRO(19, CONTEXT)
  436. // metamacro_if_eq expansions
  437. #define metamacro_if_eq0(VALUE) \
  438. metamacro_concat(metamacro_if_eq0_, VALUE)
  439. #define metamacro_if_eq0_0(...) __VA_ARGS__ metamacro_consume_
  440. #define metamacro_if_eq0_1(...) metamacro_expand_
  441. #define metamacro_if_eq0_2(...) metamacro_expand_
  442. #define metamacro_if_eq0_3(...) metamacro_expand_
  443. #define metamacro_if_eq0_4(...) metamacro_expand_
  444. #define metamacro_if_eq0_5(...) metamacro_expand_
  445. #define metamacro_if_eq0_6(...) metamacro_expand_
  446. #define metamacro_if_eq0_7(...) metamacro_expand_
  447. #define metamacro_if_eq0_8(...) metamacro_expand_
  448. #define metamacro_if_eq0_9(...) metamacro_expand_
  449. #define metamacro_if_eq0_10(...) metamacro_expand_
  450. #define metamacro_if_eq0_11(...) metamacro_expand_
  451. #define metamacro_if_eq0_12(...) metamacro_expand_
  452. #define metamacro_if_eq0_13(...) metamacro_expand_
  453. #define metamacro_if_eq0_14(...) metamacro_expand_
  454. #define metamacro_if_eq0_15(...) metamacro_expand_
  455. #define metamacro_if_eq0_16(...) metamacro_expand_
  456. #define metamacro_if_eq0_17(...) metamacro_expand_
  457. #define metamacro_if_eq0_18(...) metamacro_expand_
  458. #define metamacro_if_eq0_19(...) metamacro_expand_
  459. #define metamacro_if_eq0_20(...) metamacro_expand_
  460. #define metamacro_if_eq1(VALUE) metamacro_if_eq0(metamacro_dec(VALUE))
  461. #define metamacro_if_eq2(VALUE) metamacro_if_eq1(metamacro_dec(VALUE))
  462. #define metamacro_if_eq3(VALUE) metamacro_if_eq2(metamacro_dec(VALUE))
  463. #define metamacro_if_eq4(VALUE) metamacro_if_eq3(metamacro_dec(VALUE))
  464. #define metamacro_if_eq5(VALUE) metamacro_if_eq4(metamacro_dec(VALUE))
  465. #define metamacro_if_eq6(VALUE) metamacro_if_eq5(metamacro_dec(VALUE))
  466. #define metamacro_if_eq7(VALUE) metamacro_if_eq6(metamacro_dec(VALUE))
  467. #define metamacro_if_eq8(VALUE) metamacro_if_eq7(metamacro_dec(VALUE))
  468. #define metamacro_if_eq9(VALUE) metamacro_if_eq8(metamacro_dec(VALUE))
  469. #define metamacro_if_eq10(VALUE) metamacro_if_eq9(metamacro_dec(VALUE))
  470. #define metamacro_if_eq11(VALUE) metamacro_if_eq10(metamacro_dec(VALUE))
  471. #define metamacro_if_eq12(VALUE) metamacro_if_eq11(metamacro_dec(VALUE))
  472. #define metamacro_if_eq13(VALUE) metamacro_if_eq12(metamacro_dec(VALUE))
  473. #define metamacro_if_eq14(VALUE) metamacro_if_eq13(metamacro_dec(VALUE))
  474. #define metamacro_if_eq15(VALUE) metamacro_if_eq14(metamacro_dec(VALUE))
  475. #define metamacro_if_eq16(VALUE) metamacro_if_eq15(metamacro_dec(VALUE))
  476. #define metamacro_if_eq17(VALUE) metamacro_if_eq16(metamacro_dec(VALUE))
  477. #define metamacro_if_eq18(VALUE) metamacro_if_eq17(metamacro_dec(VALUE))
  478. #define metamacro_if_eq19(VALUE) metamacro_if_eq18(metamacro_dec(VALUE))
  479. #define metamacro_if_eq20(VALUE) metamacro_if_eq19(metamacro_dec(VALUE))
  480. // metamacro_if_eq_recursive expansions
  481. #define metamacro_if_eq_recursive0(VALUE) \
  482. metamacro_concat(metamacro_if_eq_recursive0_, VALUE)
  483. #define metamacro_if_eq_recursive0_0(...) __VA_ARGS__ metamacro_consume_
  484. #define metamacro_if_eq_recursive0_1(...) metamacro_expand_
  485. #define metamacro_if_eq_recursive0_2(...) metamacro_expand_
  486. #define metamacro_if_eq_recursive0_3(...) metamacro_expand_
  487. #define metamacro_if_eq_recursive0_4(...) metamacro_expand_
  488. #define metamacro_if_eq_recursive0_5(...) metamacro_expand_
  489. #define metamacro_if_eq_recursive0_6(...) metamacro_expand_
  490. #define metamacro_if_eq_recursive0_7(...) metamacro_expand_
  491. #define metamacro_if_eq_recursive0_8(...) metamacro_expand_
  492. #define metamacro_if_eq_recursive0_9(...) metamacro_expand_
  493. #define metamacro_if_eq_recursive0_10(...) metamacro_expand_
  494. #define metamacro_if_eq_recursive0_11(...) metamacro_expand_
  495. #define metamacro_if_eq_recursive0_12(...) metamacro_expand_
  496. #define metamacro_if_eq_recursive0_13(...) metamacro_expand_
  497. #define metamacro_if_eq_recursive0_14(...) metamacro_expand_
  498. #define metamacro_if_eq_recursive0_15(...) metamacro_expand_
  499. #define metamacro_if_eq_recursive0_16(...) metamacro_expand_
  500. #define metamacro_if_eq_recursive0_17(...) metamacro_expand_
  501. #define metamacro_if_eq_recursive0_18(...) metamacro_expand_
  502. #define metamacro_if_eq_recursive0_19(...) metamacro_expand_
  503. #define metamacro_if_eq_recursive0_20(...) metamacro_expand_
  504. #define metamacro_if_eq_recursive1(VALUE) metamacro_if_eq_recursive0(metamacro_dec(VALUE))
  505. #define metamacro_if_eq_recursive2(VALUE) metamacro_if_eq_recursive1(metamacro_dec(VALUE))
  506. #define metamacro_if_eq_recursive3(VALUE) metamacro_if_eq_recursive2(metamacro_dec(VALUE))
  507. #define metamacro_if_eq_recursive4(VALUE) metamacro_if_eq_recursive3(metamacro_dec(VALUE))
  508. #define metamacro_if_eq_recursive5(VALUE) metamacro_if_eq_recursive4(metamacro_dec(VALUE))
  509. #define metamacro_if_eq_recursive6(VALUE) metamacro_if_eq_recursive5(metamacro_dec(VALUE))
  510. #define metamacro_if_eq_recursive7(VALUE) metamacro_if_eq_recursive6(metamacro_dec(VALUE))
  511. #define metamacro_if_eq_recursive8(VALUE) metamacro_if_eq_recursive7(metamacro_dec(VALUE))
  512. #define metamacro_if_eq_recursive9(VALUE) metamacro_if_eq_recursive8(metamacro_dec(VALUE))
  513. #define metamacro_if_eq_recursive10(VALUE) metamacro_if_eq_recursive9(metamacro_dec(VALUE))
  514. #define metamacro_if_eq_recursive11(VALUE) metamacro_if_eq_recursive10(metamacro_dec(VALUE))
  515. #define metamacro_if_eq_recursive12(VALUE) metamacro_if_eq_recursive11(metamacro_dec(VALUE))
  516. #define metamacro_if_eq_recursive13(VALUE) metamacro_if_eq_recursive12(metamacro_dec(VALUE))
  517. #define metamacro_if_eq_recursive14(VALUE) metamacro_if_eq_recursive13(metamacro_dec(VALUE))
  518. #define metamacro_if_eq_recursive15(VALUE) metamacro_if_eq_recursive14(metamacro_dec(VALUE))
  519. #define metamacro_if_eq_recursive16(VALUE) metamacro_if_eq_recursive15(metamacro_dec(VALUE))
  520. #define metamacro_if_eq_recursive17(VALUE) metamacro_if_eq_recursive16(metamacro_dec(VALUE))
  521. #define metamacro_if_eq_recursive18(VALUE) metamacro_if_eq_recursive17(metamacro_dec(VALUE))
  522. #define metamacro_if_eq_recursive19(VALUE) metamacro_if_eq_recursive18(metamacro_dec(VALUE))
  523. #define metamacro_if_eq_recursive20(VALUE) metamacro_if_eq_recursive19(metamacro_dec(VALUE))
  524. // metamacro_take expansions
  525. #define metamacro_take0(...)
  526. #define metamacro_take1(...) metamacro_head(__VA_ARGS__)
  527. #define metamacro_take2(...) metamacro_head(__VA_ARGS__), metamacro_take1(metamacro_tail(__VA_ARGS__))
  528. #define metamacro_take3(...) metamacro_head(__VA_ARGS__), metamacro_take2(metamacro_tail(__VA_ARGS__))
  529. #define metamacro_take4(...) metamacro_head(__VA_ARGS__), metamacro_take3(metamacro_tail(__VA_ARGS__))
  530. #define metamacro_take5(...) metamacro_head(__VA_ARGS__), metamacro_take4(metamacro_tail(__VA_ARGS__))
  531. #define metamacro_take6(...) metamacro_head(__VA_ARGS__), metamacro_take5(metamacro_tail(__VA_ARGS__))
  532. #define metamacro_take7(...) metamacro_head(__VA_ARGS__), metamacro_take6(metamacro_tail(__VA_ARGS__))
  533. #define metamacro_take8(...) metamacro_head(__VA_ARGS__), metamacro_take7(metamacro_tail(__VA_ARGS__))
  534. #define metamacro_take9(...) metamacro_head(__VA_ARGS__), metamacro_take8(metamacro_tail(__VA_ARGS__))
  535. #define metamacro_take10(...) metamacro_head(__VA_ARGS__), metamacro_take9(metamacro_tail(__VA_ARGS__))
  536. #define metamacro_take11(...) metamacro_head(__VA_ARGS__), metamacro_take10(metamacro_tail(__VA_ARGS__))
  537. #define metamacro_take12(...) metamacro_head(__VA_ARGS__), metamacro_take11(metamacro_tail(__VA_ARGS__))
  538. #define metamacro_take13(...) metamacro_head(__VA_ARGS__), metamacro_take12(metamacro_tail(__VA_ARGS__))
  539. #define metamacro_take14(...) metamacro_head(__VA_ARGS__), metamacro_take13(metamacro_tail(__VA_ARGS__))
  540. #define metamacro_take15(...) metamacro_head(__VA_ARGS__), metamacro_take14(metamacro_tail(__VA_ARGS__))
  541. #define metamacro_take16(...) metamacro_head(__VA_ARGS__), metamacro_take15(metamacro_tail(__VA_ARGS__))
  542. #define metamacro_take17(...) metamacro_head(__VA_ARGS__), metamacro_take16(metamacro_tail(__VA_ARGS__))
  543. #define metamacro_take18(...) metamacro_head(__VA_ARGS__), metamacro_take17(metamacro_tail(__VA_ARGS__))
  544. #define metamacro_take19(...) metamacro_head(__VA_ARGS__), metamacro_take18(metamacro_tail(__VA_ARGS__))
  545. #define metamacro_take20(...) metamacro_head(__VA_ARGS__), metamacro_take19(metamacro_tail(__VA_ARGS__))
  546. // metamacro_drop expansions
  547. #define metamacro_drop0(...) __VA_ARGS__
  548. #define metamacro_drop1(...) metamacro_tail(__VA_ARGS__)
  549. #define metamacro_drop2(...) metamacro_drop1(metamacro_tail(__VA_ARGS__))
  550. #define metamacro_drop3(...) metamacro_drop2(metamacro_tail(__VA_ARGS__))
  551. #define metamacro_drop4(...) metamacro_drop3(metamacro_tail(__VA_ARGS__))
  552. #define metamacro_drop5(...) metamacro_drop4(metamacro_tail(__VA_ARGS__))
  553. #define metamacro_drop6(...) metamacro_drop5(metamacro_tail(__VA_ARGS__))
  554. #define metamacro_drop7(...) metamacro_drop6(metamacro_tail(__VA_ARGS__))
  555. #define metamacro_drop8(...) metamacro_drop7(metamacro_tail(__VA_ARGS__))
  556. #define metamacro_drop9(...) metamacro_drop8(metamacro_tail(__VA_ARGS__))
  557. #define metamacro_drop10(...) metamacro_drop9(metamacro_tail(__VA_ARGS__))
  558. #define metamacro_drop11(...) metamacro_drop10(metamacro_tail(__VA_ARGS__))
  559. #define metamacro_drop12(...) metamacro_drop11(metamacro_tail(__VA_ARGS__))
  560. #define metamacro_drop13(...) metamacro_drop12(metamacro_tail(__VA_ARGS__))
  561. #define metamacro_drop14(...) metamacro_drop13(metamacro_tail(__VA_ARGS__))
  562. #define metamacro_drop15(...) metamacro_drop14(metamacro_tail(__VA_ARGS__))
  563. #define metamacro_drop16(...) metamacro_drop15(metamacro_tail(__VA_ARGS__))
  564. #define metamacro_drop17(...) metamacro_drop16(metamacro_tail(__VA_ARGS__))
  565. #define metamacro_drop18(...) metamacro_drop17(metamacro_tail(__VA_ARGS__))
  566. #define metamacro_drop19(...) metamacro_drop18(metamacro_tail(__VA_ARGS__))
  567. #define metamacro_drop20(...) metamacro_drop19(metamacro_tail(__VA_ARGS__))
  568. #endif