Por que essa substituição de macro aninhada está falhando?

Estou tentando aplicar oX Macro conceito, para ter a possibilidade de inicializar todos os membros da estrutura para um valor padrão (inválido) personalizado. Eu escrevo o seguinte código:

#define LIST_OF_STRUCT_MEMBERS_foo \
    X(a) \
    X(b) \
    X(c)

#define X(name) int name;
struct foo {
     LIST_OF_STRUCT_MEMBERS_foo
};
#undef X


#define X(name) -1,
static inline void foo_invalidate(struct foo* in) {
     *in = (struct foo){
     LIST_OF_STRUCT_MEMBERS_foo
     };
}
#undef X

#define X(name) -1,
#define foo_DEFAULT_VALUE  { LIST_OF_STRUCT_MEMBERS_foo }
#undef X

static struct foo test = foo_DEFAULT_VALUE;

No entanto, quando executo o pré-processador, a definição defoo_DEFAULT_VALUE falha ao substituir oX(name) liga com-1,

Saída do pré-processador:

struct foo {
     int a; int b; int c;
};

static inline void foo_invalidate(struct foo* in) {
     *in = (struct foo){
     -1, -1, -1, /*Here the substitution worked nicely*/
     };
}

static struct foo test = { X(a) X(b) X(c) }; /*Why this substitution failed?*/

eu penseiAs macros C podem se referir a outras macros. Você sabe por que essa substituição falha? Existe alguma solução alternativa?

Eu poderia viver comfoo_invalidate, mas reluto em desistir de uma etapa de ter um valor a ser usado diretamente na inicialização.