Ao tentar compilar esse código
#include <stdarg.h>
void bar_ptr(int n, va_list *pvl) {
// do va_arg stuff here
}
void bar(int n, va_list vl) {
va_list *pvl = &vl; // error here
bar_ptr(n, pvl);
}
void foo(int n, ...) {
va_list vl;
va_list *pvl = &vl; // fine here
va_start(vl, n);
bar(n, vl);
va_end(vl);
}
int main() {
foo(3, 1, 2, 3);
return 0;
}
o compilador GCC imprime um aviso sobre initialization from incompatible pointer typena barfunção. A afirmação idêntica está correta foo.
Parece que o tipo de um instrumento do tipo va_listnão é a va_list. Isso pode ser testado facilmente com uma asserção estática, como
_Static_assert(sizeof(vl) == sizeof(va_list), "invalid type");
na barfunção. Com o GCC, a _Static_assertfalha. O mesmo pode ser testado também em C ++ com declytpee std::is_same.
Gostaria de pegar o endereço do va_list vlargumento de bar, e passá-lo como argumento de bar_ptr, para fazer pensamentos como o descrito neste tópico . Por outro lado, não há problema em ligar bar_ptr(n, pvl)diretamente de main, substituindo bar(n, vl).
De acordo com a nota 253 da minuta final do C11 ,
É permitido criar um ponteiro para um
va_liste passar esse ponteiro para outra função
Por que isso não pode ser feito se va_listdefinido como argumento da função, e não no corpo da função?
Gambiarra:
Mesmo que isso não responda à pergunta, uma possível solução alternativa é alterar o conteúdo barusando uma cópia local do argumento criado com va_copy:
void bar(int n, va_list vl) {
va_list vl_copy;
va_copy(vl_copy, vl);
va_list *pvl = &vl_copy; // now fine here
bar_ptr(n, pvl);
va_end(va_copy);
}
va_list, está passando um real va_list.
va_list. Suponha que eu tenho uma terceira função void bar_ptr(va_list *pvl);, quero passar um ponteiro va_list vlpara essa função. Vou editar a pergunta para especificá-la.
va_list
va_copyabordagem que você tem é a solução canônica para esse problema. Eu tenho algumas perguntas anteriores que basicamente concluíram isso.
va_list.