Eu incluí sys/ptrace.h
no meu programa C.
A saída de /usr/lib/gcc/x86_64-linux-gnu/4.8/cc1 -v
fornece os seguintes caminhos em que o gcc procura por arquivos de cabeçalho
#include "..." search starts here:
#include <...> search starts here:
/usr/lib/gcc/x86_64-linux-gnu/4.8/include
/usr/local/include
/usr/lib/gcc/x86_64-linux-gnu/4.8/include-fixed
/usr/include
End of search list.
saída de gcc -M
para o meu programa fornece os seguintes locais de arquivo de cabeçalho
pt.o: pt.c /usr/include/stdc-predef.h /usr/include/stdio.h \
/usr/include/features.h /usr/include/x86_64-linux-gnu/sys/cdefs.h \
/usr/include/x86_64-linux-gnu/bits/wordsize.h \
/usr/include/x86_64-linux-gnu/gnu/stubs.h \
/usr/include/x86_64-linux-gnu/gnu/stubs-64.h \
/usr/lib/gcc/x86_64-linux-gnu/4.8/include/stddef.h \
/usr/include/x86_64-linux-gnu/bits/types.h \
/usr/include/x86_64-linux-gnu/bits/typesizes.h /usr/include/libio.h \
/usr/include/_G_config.h /usr/include/wchar.h \
/usr/lib/gcc/x86_64-linux-gnu/4.8/include/stdarg.h \
/usr/include/x86_64-linux-gnu/bits/stdio_lim.h \
/usr/include/x86_64-linux-gnu/bits/sys_errlist.h \
/usr/include/x86_64-linux-gnu/sys/ptrace.h
Como /usr/include/x86_64-linux-gnu/
não está contido na primeira saída, como o gcc encontra sys/ptrace.h
?
EDITAR:
A saída dos echo '#include <sys/ptrace.h>' | gcc -fsyntax-only -xc -v -H -
resultados em
Configured with: ../src/configure -v --with-pkgversion='Ubuntu 4.8.4-2ubuntu1~14.04' --with-bugurl=file:///usr/share/doc/gcc-4.8/README.Bugs --enable-languages=c,c++,java,go,d,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-4.8 --enable-shared --enable-linker-build-id --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.8 --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --enable-gnu-unique-object --disable-libmudflap --enable-plugin --with-system-zlib --disable-browser-plugin --enable-java-awt=gtk --enable-gtk-cairo --with-java-home=/usr/lib/jvm/java-1.5.0-gcj-4.8-amd64/jre --enable-java-home --with-jvm-root-dir=/usr/lib/jvm/java-1.5.0-gcj-4.8-amd64 --with-jvm-jar-dir=/usr/lib/jvm-exports/java-1.5.0-gcj-4.8-amd64 --with-arch-directory=amd64 --with-ecj-jar=/usr/share/java/eclipse-ecj.jar --enable-objc-gc --enable-multiarch --disable-werror --with-arch-32=i686 --with-abi=m64 --with-multilib-list=m32,m64,mx32 --with-tune=generic --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 4.8.4 (Ubuntu 4.8.4-2ubuntu1~14.04)
/usr/include
.. Que problema você está tentando resolver?/sys/ptrace.h
massys/ptrace.h
, certo?/usr/include/x86_64-linux-gnu
está sendo tratado como um diretório de inclusão do sistema e deve ser incluído na lista de caminhos de pesquisa impressa porgcc -v
. Não sei como alguém conseguiu alcançar esse bug; se bem me lembro, a maneira mais óbvia de adicionar diretórios de inclusão do sistema os adiciona ao que é impresso-v
. (Eu escrevi ~ 50% do pré-processador do GCC, mas isso foi há 15 anos, para que eu possa ser misremembering alguma coisa.)/usr/include
. Isso quebraria quase todas as bibliotecas C do mundo.Respostas:
Resposta mais curta.
Sua pergunta é sobre a saída de
cc1 -v
, mas isso não afeta o CPP (pré-processador C) e inclui os que são misturados em toda a cadeia de compilação. Se você rodarcpp -v
em seu sistema, verá uma mistura de inclusões semelhante à saída de,cc1 -v
mas com pelo menos o/usr/include/x86_64-linux-gnu
caminho adicionado.Resposta mais longa.
Tecnicamente,
/usr/include/x86_64-linux-gnu/
não é explicitamente definido na primeira saída, mas/usr/include/
definitivamente é. E esse é um caminho de pesquisa padrão, conforme explicado na documentação oficial do GNU GCC :E mais explicado aqui:
Portanto, isso implica que o
x86_64-linux-gnu/
caminho é simplesmente inserido/usr/include/*/sys/
assim:Pelo menos foi o que pensei inicialmente em uma versão anterior desta questão . Mas, depois de verificar este site, a explicação do que está acontecendo é um pouco mais detalhada e a resposta direta desse site ao conteúdo equivalente ao que eu publiquei acima é postada abaixo; ênfase em negrito é minha:
Saiba que o CPP (pré-processador C) é o primeiro passo no processo do compilador, vamos dar uma olhada na saída "incluir" do
cpp -v
meu sistema de teste Ubuntu 12.04.5:Lá você pode ver claramente
/usr/include/x86_64-linux-gnu
. E para comparar, aqui está a saída "include" semelhante/usr/lib/gcc/x86_64-linux-gnu/4.6/cc1 -v
no mesmo sistema de teste Ubuntu 12.04.5:Observe como
/usr/include/x86_64-linux-gnu
é claramente inserido no mix pela ação inicial do CPP (pré-processador C). E a publicação nesse site continua explicando de onde vêm esses caminhos; novamente a ênfase ousada é minha:Portanto, tudo se resume ao CPP (pré-processador C) sendo chamado como a primeira parte de uma cadeia de compilação C.
fonte
$TARGET
parte que mencionei na minha resposta e comentário. É o resultado deconfig.guess
quando o GCC foi compilado ou que foi fornecido ao seuconfigure
script com o--target
sinalizador. A verdadeira questão é: como esse caminho é montado? Ele volta à mesma lista, acrescentando$TARGET
a cada uma, depois de não encontrar o cabeçalho pela primeira vez?Antes de me aprofundar no código-fonte do GCC, não posso lhe dar um "porquê", mas posso lhe dizer que a versão do GCC que tenho aqui recai
/usr/include/$TARGET
depois de esgotar as opções que você e JakeGould encontraram . Você pode vê-lo assim:onde
foo.c
contém um#include <sys/ptrace.h>
.Você precisa do
-f
argumento aqui porquegcc
gera filhos para fazer o trabalho de compilação real. Você precisa do2>&1
porquestrace
está gravando seus resultados no stderr, não no stdout.Observe que você obtém
ENOENT
erros para todos os diretórios documentados antes de finalmente tentar o que é bem-sucedido.fonte