1 /* 2 * Copyright (c) 2025 Antmicro 3 * Originally created by William Swanson in 2012, donated to the public domain. 4 */ 5 6 #ifndef MAP_H_INCLUDED 7 #define MAP_H_INCLUDED 8 9 #define EVAL0(...) __VA_ARGS__ 10 #define EVAL1(...) EVAL0(EVAL0(EVAL0(__VA_ARGS__))) 11 #define EVAL2(...) EVAL1(EVAL1(EVAL1(__VA_ARGS__))) 12 #define EVAL3(...) EVAL2(EVAL2(EVAL2(__VA_ARGS__))) 13 #define EVAL4(...) EVAL3(EVAL3(EVAL3(__VA_ARGS__))) 14 #define EVAL(...) EVAL4(EVAL4(EVAL4(__VA_ARGS__))) 15 16 #define MAP_END(...) 17 #define MAP_OUT 18 #define MAP_COMMA , 19 20 #define MAP_GET_END2() 0, MAP_END 21 #define MAP_GET_END1(...) MAP_GET_END2 22 #define MAP_GET_END(...) MAP_GET_END1 23 #define MAP_NEXT0(test, next, ...) next MAP_OUT 24 #define MAP_NEXT1(test, next) MAP_NEXT0(test, next, 0) 25 #define MAP_NEXT(test, next) MAP_NEXT1(MAP_GET_END test, next) 26 27 #define MAP0(f, x, peek, ...) f(x) MAP_NEXT(peek, MAP1)(f, peek, __VA_ARGS__) 28 #define MAP1(f, x, peek, ...) f(x) MAP_NEXT(peek, MAP0)(f, peek, __VA_ARGS__) 29 30 #define MAP_LIST_NEXT1(test, next) MAP_NEXT0(test, MAP_COMMA next, 0) 31 #define MAP_LIST_NEXT(test, next) MAP_LIST_NEXT1(MAP_GET_END test, next) 32 33 #define MAP_LIST0(f, x, peek, ...) f(x) MAP_LIST_NEXT(peek, MAP_LIST1)(f, peek, __VA_ARGS__) 34 #define MAP_LIST1(f, x, peek, ...) f(x) MAP_LIST_NEXT(peek, MAP_LIST0)(f, peek, __VA_ARGS__) 35 36 /** 37 * Applies the function macro `f` to each of the remaining parameters. 38 */ 39 #define MAP(f, ...) EVAL(MAP1(f, __VA_ARGS__, ()()(), ()()(), ()()(), 0)) 40 41 /** 42 * Applies the function macro `f` to each of the remaining parameters and 43 * inserts commas between the results. 44 */ 45 #define MAP_LIST(f, ...) EVAL(MAP_LIST1(f, __VA_ARGS__, ()()(), ()()(), ()()(), 0)) 46 47 #endif 48