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 type
na bar
função. A afirmação idêntica está correta foo
.
Parece que o tipo de um instrumento do tipo va_list
nã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 bar
função. Com o GCC, a _Static_assert
falha. O mesmo pode ser testado também em C ++ com declytpe
e std::is_same
.
Gostaria de pegar o endereço do va_list vl
argumento 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_list
e passar esse ponteiro para outra função
Por que isso não pode ser feito se va_list
definido 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 bar
usando 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);
}
fonte
va_list
.va_list
, está passando um realva_list
.va_list
. Suponha que eu tenho uma terceira funçãovoid bar_ptr(va_list *pvl);
, quero passar um ponteirova_list vl
para essa função. Vou editar a pergunta para especificá-la.va_list
va_copy
abordagem que você tem é a solução canônica para esse problema. Eu tenho algumas perguntas anteriores que basicamente concluíram isso.Respostas:
va_list
é permitido pelo padrão ser uma matriz, e geralmente é. Isso significa que,va_list
em uma função, o argumento é ajustado a um ponteiro para ova_list
primeiro elemento interno de qualquer um .A regra estranha ( 7.16p3 ) sobre como
va_list
é aprovada, basicamente acomoda a possibilidade de queva_list
possa ser de um tipo de matriz ou de um tipo regular.Eu envolvo pessoalmente
va_list
um,struct
então não tenho que lidar com isso.Quando você passa os ponteiros para tal
struct va_list_wrapper
, é basicamente como se você tivesse passado os ponteiros parava_list
, e então a nota de rodapé 253 se aplica, o que lhe dá a permissão para que um chamador e um chamador manipulem o mesmova_list
através desse ponteiro.(A mesma coisa se aplica ao
jmp_buf
esigjmp_buf
a partirsetjmp.h
. Em geral, este tipo de matriz para ajuste de ponteiro é uma das razões pelas quais digitado de matriztypedef
s devem ser evitados. Ele só cria confusão, IMO).fonte
va_list
pode ser uma matriz, é enganoso para o padrão dizer "O objetoap
pode ser passado como argumento para outra função ..." (C 2018 7.16 3,ap
é um objeto do tipova_list
).Outra solução (apenas C11 +):
Explicação: se
vl
tem tipova_list
, entãova_list
não é um tipo de matriz e apenas pegar o endereço é bom parava_list *
apontar para ele. Caso contrário, ele deve ter o tipo de matriz e, em seguida, você poderá converter um ponteiro para o primeiro elemento da matriz (qualquer que seja o tipo) em um ponteiro para a matriz.fonte