首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >我对gnu gcc ld语法有个问题

我对gnu gcc ld语法有个问题
EN

Stack Overflow用户
提问于 2019-10-18 17:26:43
回答 1查看 227关注 0票数 0

好吧,我对gcc和ld很陌生,我显然遗漏了一些东西,但是我似乎在这里,在“使用ld”或者在网上其他地方找不到具体的答案。

我使用的是一个简单的源代码文件,我把它从网络中删除了(我已经将它保存为pumy.c):

代码语言:javascript
运行
复制
 #include <stdio.h>

 int main () {
    int ch;

    for( ch = 75 ; ch <= 100; ch++ ) {
       printf("ASCII value = %d, Character = %c\n", ch , ch );
    }

    return(0);
 }

代码语言:javascript
运行
复制
 gcc dummy.c 

工作,然后

代码语言:javascript
运行
复制
 ./a.out 

也很管用。

更具体的是:

代码语言:javascript
运行
复制
 gcc -o dummy dummy.c 

工作,然后

代码语言:javascript
运行
复制
 ./dummy 

也很管用。

但是,我删除了a.out和虚拟,然后尝试:

代码语言:javascript
运行
复制
 gcc -c dummy.c 

产生假人.o

然后我做了:

代码语言:javascript
运行
复制
 ld -o dummy dummy.o -lc

但结果是:

代码语言:javascript
运行
复制
 ld: warning: cannot find entry symbol _start; defaulting to 0000000008049020

尽管它会产生假人,但尝试./dummy的结果是:

代码语言:javascript
运行
复制
 -bash: ./dummy: No such file or directory

尽管虚拟文件显然在目录中(具有755权限)。

我做错了什么?

=====

按照ssbssa下面的建议,我尝试了gcc -v dumy.c->,结果是:

代码语言:javascript
运行
复制
 Using built-in specs.
 COLLECT_GCC=gcc
 COLLECT_LTO_WRAPPER=/usr/lib/gcc/i686-linux-gnu/8/lto-wrapper
 Target: i686-linux-gnu
 Configured with: ../src/configure -v --with-pkgversion='Debian 8.3.0-6' --with-bugurl=file:///usr/share/doc/gcc-8/README.Bugs --enable-languages=c,ada,c++,go,brig,d,fortran,objc,obj-c++ --prefix=/usr --with-gcc-major-version-only --program-suffix=-8 --program-prefix=i686-linux-gnu- --enable-shared --enable-linker-build-id --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --libdir=/usr/lib --enable-nls --enable-bootstrap --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --with-default-libstdcxx-abi=new --enable-gnu-unique-object --disable-vtable-verify --enable-libmpx --enable-plugin --enable-default-pie --with-system-zlib --with-target-system-zlib --enable-objc-gc=auto --enable-targets=all --enable-multiarch --disable-werror --with-arch-32=i686 --with-multilib-list=m32,m64,mx32 --enable-multilib --with-tune=generic --enable-checking=release --build=i686-linux-gnu --host=i686-linux-gnu --target=i686-linux-gnu
 Thread model: posix
 gcc version 8.3.0 (Debian 8.3.0-6)
 COLLECT_GCC_OPTIONS='-v' '-mtune=generic' '-march=i686'
  /usr/lib/gcc/i686-linux-gnu/8/cc1 -quiet -v -imultiarch i386-linux-gnu dummy.c -quiet -dumpbase dummy.c -mtune=generic -march=i686 -auxbase dummy -version -o /tmp/ccbFKhxi.s
 GNU C17 (Debian 8.3.0-6) version 8.3.0 (i686-linux-gnu)
         compiled by GNU C version 8.3.0, GMP version 6.1.2, MPFR version 4.0.2, MPC version 1.1.0, isl version isl-0.20-GMP

 GGC heuristics: --param ggc-min-expand=100 --param ggc-min-heapsize=131072
 ignoring nonexistent directory "/usr/local/include/i386-linux-gnu"
 ignoring nonexistent directory "/usr/lib/gcc/i686-linux-gnu/8/../../../../i686-linux-gnu/include"
 #include "..." search starts here:
 #include <...> search starts here:
  /usr/lib/gcc/i686-linux-gnu/8/include
  /usr/local/include
  /usr/lib/gcc/i686-linux-gnu/8/include-fixed
  /usr/include/i386-linux-gnu
  /usr/include
 End of search list.
 GNU C17 (Debian 8.3.0-6) version 8.3.0 (i686-linux-gnu)
         compiled by GNU C version 8.3.0, GMP version 6.1.2, MPFR version 4.0.2, MPC version 1.1.0, isl version isl-0.20-GMP

 GGC heuristics: --param ggc-min-expand=100 --param ggc-min-heapsize=131072
 Compiler executable checksum: c52019c44a1e362eeb6021f108314d7e
 COLLECT_GCC_OPTIONS='-v' '-mtune=generic' '-march=i686'
  as -v --32 -o /tmp/ccORYUuk.o /tmp/ccbFKhxi.s
 GNU assembler version 2.31.1 (i686-linux-gnu) using BFD version (GNU Binutils for Debian) 2.31.1
 COMPILER_PATH=/usr/lib/gcc/i686-linux-gnu/8/:/usr/lib/gcc/i686-linux-gnu/8/:/usr/lib/gcc/i686-linux-gnu/:/usr/lib/gcc/i686-linux-gnu/8/:/usr/lib/gcc/i686-linux-gnu/
 LIBRARY_PATH=/usr/lib/gcc/i686-linux-gnu/8/:/usr/lib/gcc/i686-linux-gnu/8/../../../i386-linux-gnu/:/usr/lib/gcc/i686-linux-gnu/8/../../../../lib/:/lib/i386-linux-gnu/:/lib/../lib/:/usr/lib/i386-linux-gnu/:/usr/lib/../lib/:/usr/lib/gcc/i686-linux-gnu/8/../../../:/lib/:/usr/lib/
 COLLECT_GCC_OPTIONS='-v' '-mtune=generic' '-march=i686'
  /usr/lib/gcc/i686-linux-gnu/8/collect2 -plugin /usr/lib/gcc/i686-linux-gnu/8/liblto_plugin.so -plugin-opt=/usr/lib/gcc/i686-linux-gnu/8/lto-wrapper -plugin-opt=-fresolution=/tmp/cci9u5um.res -plugin-opt=-pass-through=-lgcc -plugin-opt=-pass-through=-lgcc_s -plugin-opt=-pass-through=-lc -plugin-opt=-pass-through=-lgcc -plugin-opt=-pass-through=-lgcc_s --build-id --eh-frame-hdr -m elf_i386 --hash-style=gnu -dynamic-linker /lib/ld-linux.so.2 -pie /usr/lib/gcc/i686-linux-gnu/8/../../../i386-linux-gnu/Scrt1.o /usr/lib/gcc/i686-linux-gnu/8/../../../i386-linux-gnu/crti.o /usr/lib/gcc/i686-linux-gnu/8/crtbeginS.o -L/usr/lib/gcc/i686-linux-gnu/8 -L/usr/lib/gcc/i686-linux-gnu/8/../../../i386-linux-gnu -L/usr/lib/gcc/i686-linux-gnu/8/../../../../lib -L/lib/i386-linux-gnu -L/lib/../lib -L/usr/lib/i386-linux-gnu -L/usr/lib/../lib -L/usr/lib/gcc/i686-linux-gnu/8/../../.. /tmp/ccORYUuk.o -lgcc --push-state --as-needed -lgcc_s --pop-state -lc -lgcc --push-state --as-needed -lgcc_s --pop-state /usr/lib/gcc/i686-linux-gnu/8/crtendS.o /usr/lib/gcc/i686-linux-gnu/8/../../../i386-linux-gnu/crtn.o
 COLLECT_GCC_OPTIONS='-v' '-mtune=generic' '-march=i686' 

唯一看上去有点像"ld“的地方似乎是在最后一行,比滚动屏幕上的屏幕稍小一点:

代码语言:javascript
运行
复制
 /lib/ld-linux.so.2 -pie 

谢谢ssbssa,我会玩一玩,看看会发生什么。我还将浏览您所指向的特定文档页。

在研究了ssbssa推荐的引用和其他几个引用之后,我开始怀疑ld不应该像那些引用看起来那样复杂。因此,我尝试了一个简单的直接汇编源程序:

代码语言:javascript
运行
复制
 # From GASProfessional Book - location 108 ff
 # cpuid.s Sample program to extract the processor Vendor ID

 .section .data

 output:
   .ascii "The processor Vendor ID is 'xxxxxxxxxxxx'\n"

 .section .text

 .globl _start

 _start:

   movl $0, %eax           # Get the CPU's Vendor ID
   cpuid

   movl $output, %edi      # Setup the output string
   movl %ebx, 28(%edi)
   movl %edx, 32(%edi)
   movl %ecx, 36(%edi)

   movl $4, %eax           # Display the output string
   movl $1, %ebx
   movl $output, %ecx
   movl $42, %edx
   int $0x80

   movl $1, %eax           # Exit 0
   movl $0, %ebx
   int $0x80

然后我做了:

代码语言:javascript
运行
复制
 as -o cpuid.o cpuid.s

以及:

代码语言:javascript
运行
复制
 ld -o cpuid cpuid.s

而命令:

代码语言:javascript
运行
复制
 ./cpuid

然后制作:

代码语言:javascript
运行
复制
 The processor Vendor ID is 'GenuineIntel'

如预期的那样。

所以,在c++程序中,也许我需要玩main vs. _start?往前冲,冲进缺口!!

EN

回答 1

Stack Overflow用户

发布于 2020-01-23 15:15:09

ld -o虚拟假人.o -lc

通常,您不想直接调用ld。相反,使用gcc链接。gcc实际上只是一个驱动程序,它调用所需的sup进程(编译器专用程序(cc1 for C,cc1plus for C++)、汇编程序as、linker ld),并将相当多的选项传递给这些子进程,例如路径、库、多库信息、启动代码等等。

gcc -o虚拟假人

由于只有一个编译单元,所以可以编译、组装和链接到

gcc -o虚拟假人

为了查看gcc正在调用的命令,添加选项-v

类似的情况也适用于运行汇编程序。让gcc运行汇编程序可以简化命令,例如C预处理器应该在asm源上运行(扩展.S.sx,或者使用gcc -x assembler-with-cpp module.asm进行其他扩展)。

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/58455582

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档