зеркало из https://github.com/mozilla/gecko-dev.git
Bug 1369622 - Fix MOZ_FOR_EACH with an empty list. r=froydnj
I'm not sure how I tested MOZ_FOR_EACH in bug 1368932, but it turns out it doesn't work with an empty list, despite MOZ_PASTE_PREFIX_AND_ARG_COUNT now supporting 0 arguments. Macros can be tricky, and it ends up being easier to make things work cross-compiler with a separate macro that does the counting, and (re)building MOZ_PASTE_PREFIX_AND_ARG_COUNT on top of that. Then MOZ_FOR_EACH ends up working as expected with an empty list. So this adds a MOZ_ARG_COUNT macro that counts the number of variadic arguments it's given, and derives MOZ_PASTE_PREFIX_AND_ARG_COUNT from it. And this adds a testcase validating that MOZ_FOR_EACH works properly with an empty list as a result. --HG-- extra : rebase_source : 309371d87bd1561fbd2153f44fc1256185045d23
This commit is contained in:
Родитель
1d3666a615
Коммит
38a6de0fe3
|
@ -16,44 +16,35 @@
|
|||
#define MOZ_CONCAT(x, y) MOZ_CONCAT2(x, y)
|
||||
|
||||
/*
|
||||
* MOZ_PASTE_PREFIX_AND_ARG_COUNT(aPrefix, ...) counts the number of variadic
|
||||
* arguments and prefixes it with |aPrefix|. For example:
|
||||
* MOZ_ARG_COUNT(...) counts the number of variadic arguments.
|
||||
* You must pass in between 0 and 50 (inclusive) variadic arguments.
|
||||
* For example:
|
||||
*
|
||||
* MOZ_PASTE_PREFIX_AND_ARG_COUNT(, foo, 42) expands to 2
|
||||
* MOZ_PASTE_PREFIX_AND_ARG_COUNT(A, foo, 42, bar) expands to A3
|
||||
* MOZ_PASTE_PREFIX_AND_ARG_COUNT(A) expands to A0
|
||||
* MOZ_PASTE_PREFIX_AND_ARG_COUNT() expands to 0, but MSVC warns there
|
||||
* aren't enough arguments given.
|
||||
*
|
||||
* You must pass in between 0 and 50 (inclusive) variadic arguments, past
|
||||
* |aPrefix|.
|
||||
* MOZ_ARG_COUNT() expands to 0
|
||||
* MOZ_ARG_COUNT(a) expands to 1
|
||||
* MOZ_ARG_COUNT(a, b) expands to 2
|
||||
*
|
||||
* Implementation notes:
|
||||
* The `##__VA_ARGS__` form is a GCC extension that removes the comma if
|
||||
* __VA_ARGS__ is empty. It is supported by Clang too. MSVC ignores ##,
|
||||
* and its default behavior is already to strip the comma when __VA_ARGS__
|
||||
* is empty.
|
||||
*
|
||||
* So MOZ_MACROARGS_ARG_COUNT_HELPER(prefix) expands to
|
||||
* (_, prefix50, prefix49, ...)
|
||||
* MOZ_MACROARGS_ARG_COUNT_HELPER(prefix, a) expands to
|
||||
* (_, a, prefix50, prefix49, ...)
|
||||
* So MOZ_MACROARGS_ARG_COUNT_HELPER() expands to
|
||||
* (_, 50, 49, ...)
|
||||
* MOZ_MACROARGS_ARG_COUNT_HELPER(a) expands to
|
||||
* (_, a, 50, 49, ...)
|
||||
* etc.
|
||||
*/
|
||||
#define MOZ_PASTE_PREFIX_AND_ARG_COUNT(aPrefix, ...) \
|
||||
MOZ_MACROARGS_ARG_COUNT_HELPER2( \
|
||||
MOZ_MACROARGS_ARG_COUNT_HELPER(aPrefix, ##__VA_ARGS__))
|
||||
#define MOZ_ARG_COUNT(...) \
|
||||
MOZ_MACROARGS_ARG_COUNT_HELPER2(MOZ_MACROARGS_ARG_COUNT_HELPER(__VA_ARGS__))
|
||||
|
||||
#define MOZ_MACROARGS_ARG_COUNT_HELPER(aPrefix, ...) (_, ##__VA_ARGS__, \
|
||||
aPrefix##50, aPrefix##49, aPrefix##48, aPrefix##47, aPrefix##46, \
|
||||
aPrefix##45, aPrefix##44, aPrefix##43, aPrefix##42, aPrefix##41, \
|
||||
aPrefix##40, aPrefix##39, aPrefix##38, aPrefix##37, aPrefix##36, \
|
||||
aPrefix##35, aPrefix##34, aPrefix##33, aPrefix##32, aPrefix##31, \
|
||||
aPrefix##30, aPrefix##29, aPrefix##28, aPrefix##27, aPrefix##26, \
|
||||
aPrefix##25, aPrefix##24, aPrefix##23, aPrefix##22, aPrefix##21, \
|
||||
aPrefix##20, aPrefix##19, aPrefix##18, aPrefix##17, aPrefix##16, \
|
||||
aPrefix##15, aPrefix##14, aPrefix##13, aPrefix##12, aPrefix##11, \
|
||||
aPrefix##10, aPrefix##9, aPrefix##8, aPrefix##7, aPrefix##6, \
|
||||
aPrefix##5, aPrefix##4, aPrefix##3, aPrefix##2, aPrefix##1, aPrefix##0)
|
||||
#define MOZ_MACROARGS_ARG_COUNT_HELPER(...) (_, ##__VA_ARGS__, \
|
||||
50, 49, 48, 47, 46, 45, 44, 43, 42, 41, \
|
||||
40, 39, 38, 37, 36, 35, 34, 33, 32, 31, \
|
||||
30, 29, 28, 27, 26, 25, 24, 23, 22, 21, \
|
||||
20, 19, 18, 17, 16, 15, 14, 13, 12, 11, \
|
||||
10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0)
|
||||
|
||||
#define MOZ_MACROARGS_ARG_COUNT_HELPER2(aArgs) \
|
||||
MOZ_MACROARGS_ARG_COUNT_HELPER3 aArgs
|
||||
|
@ -66,6 +57,24 @@
|
|||
a41, a42, a43, a44, a45, a46, a47, a48, a49, a50, \
|
||||
a51, ...) a51
|
||||
|
||||
/*
|
||||
* MOZ_PASTE_PREFIX_AND_ARG_COUNT(aPrefix, ...) counts the number of variadic
|
||||
* arguments and prefixes it with |aPrefix|. For example:
|
||||
*
|
||||
* MOZ_PASTE_PREFIX_AND_ARG_COUNT(, foo, 42) expands to 2
|
||||
* MOZ_PASTE_PREFIX_AND_ARG_COUNT(A, foo, 42, bar) expands to A3
|
||||
* MOZ_PASTE_PREFIX_AND_ARG_COUNT(A) expands to A0
|
||||
* MOZ_PASTE_PREFIX_AND_ARG_COUNT() expands to 0, but MSVC warns there
|
||||
* aren't enough arguments given.
|
||||
*
|
||||
* You must pass in between 0 and 50 (inclusive) variadic arguments, past
|
||||
* |aPrefix|.
|
||||
*/
|
||||
#define MOZ_PASTE_PREFIX_AND_ARG_COUNT_GLUE(a, b) a b
|
||||
#define MOZ_PASTE_PREFIX_AND_ARG_COUNT(aPrefix, ...) \
|
||||
MOZ_PASTE_PREFIX_AND_ARG_COUNT_GLUE( \
|
||||
MOZ_CONCAT, (aPrefix, MOZ_ARG_COUNT(__VA_ARGS__)))
|
||||
|
||||
/*
|
||||
* MOZ_ARGS_AFTER_N expands to its arguments excluding the first |N|
|
||||
* arguments. For example:
|
||||
|
|
|
@ -6,6 +6,11 @@
|
|||
|
||||
#include "mozilla/MacroArgs.h"
|
||||
|
||||
static_assert(MOZ_ARG_COUNT() == 0, "");
|
||||
static_assert(MOZ_ARG_COUNT(a) == 1, "");
|
||||
static_assert(MOZ_ARG_COUNT(a, b) == 2, "");
|
||||
static_assert(MOZ_ARG_COUNT(a, b, c) == 3, "");
|
||||
|
||||
static_assert(MOZ_PASTE_PREFIX_AND_ARG_COUNT(100) == 1000, "");
|
||||
static_assert(MOZ_PASTE_PREFIX_AND_ARG_COUNT(100, a) == 1001, "");
|
||||
static_assert(MOZ_PASTE_PREFIX_AND_ARG_COUNT(100, a, b) == 1002, "");
|
||||
|
|
|
@ -16,6 +16,9 @@ static_assert(MOZ_FOR_EACH_SEPARATED(HELPER_IDENTITY, (+),
|
|||
static_assert(MOZ_FOR_EACH_SEPARATED(HELPER_IDENTITY, (+),
|
||||
(), (1, 1, 1)) == 3, "");
|
||||
|
||||
#define HELPER_ONE_PLUS(x) HELPER_IDENTITY_PLUS(1)
|
||||
static_assert(MOZ_FOR_EACH(HELPER_ONE_PLUS, (), ()) 0 == 0, "");
|
||||
|
||||
#define HELPER_DEFINE_VAR(x) const int test1_##x = x;
|
||||
MOZ_FOR_EACH(HELPER_DEFINE_VAR, (), (10, 20))
|
||||
static_assert(test1_10 == 10 && test1_20 == 20, "");
|
||||
|
|
Загрузка…
Ссылка в новой задаче