基本GCC命令的使用

GCC是一套由GNU项目开发的编程语言编译器,可处理C语言、

C++、Fortran、Pascal、Objective-C、Java等等。GCC通常是 跨平台软件的编译器首选。gcc是GCC套件中的编译驱动程序名。

若计算机是x86-64位系统,为了编译成IA-32指令集,

则请先运行下列命令:

sudo apt-get install build-essential module-assistant
sudo apt-get install gcc-multilib g++-multilib

接下来就以输出hello world这样一个简单的C语言程序hello.c来演示这个过程。

#include <stdio.h>
int main()
{
printf("Hello,world\n");
return 0;
}

hello.c这个源文件要经过预处理,编译,汇编,链接四个过程,最终生成可执行目标文件。示意图如下所示:



Linux中的gcc编译驱动程序可以实现上图过程的每一步,下面分别对其进行解释:

gcc –E hello.c –o hello.i

对hello.c的程序进行预编译,预编译是对源程序以字符#开头的命令进行处理,对于这里来说就是将include后的.h文件内容嵌入到源程序文件中,预处理后的文件还是文本文件,用.i作扩展名,如下图所示:

gcc –S hello.i –o hello.s

对hello.i进行编译,生成一个汇编语言源程序,用.s作为扩展名,编译后的文件还是文本文件。

gcc –c hello.s –o hello.o

对hello.s文件进行汇编,生成一个可重定位目标文件,以.o作为扩展名,汇编后的文件是二进制文件。内容为0,1表示的机器指令,数据和其他信息。

gcc hello.o –o hello

此命令将多个可重定位的目标文件和标准库函数链接合成一个可执行文件。在这个例子中,链接将hello.o文件和标准库函数printf所在的可重定位目标模块printf.o进行链接,生成可执行目标文件hello。

运行可执行文件hello输入指令

./hello

上面是分步骤转换C语言程序到可执行目标文件,也可以用命令

gcc hello.c –o hello

将hello.c直接编译成可执行目标文件hello。

在用gcc命令编译c程序时,会加入各种选择,例如以下命令:

gcc -o0 -m32 -g hello.c -o hello

比如加入-O0,塔表示编译时采用的优化级别,0表示不用编译优化,-m32这个选项表示编译成x86-32位的指令。如果计算机是64位架构的处理器,不加这个选项则会编译成x86-64位的指令集。-g表示带调试信息(单步调试必须加入)。

objdump命令的使用

目标文件都是由01序列的机器指令构成,数据和其他信息构成,用文本编辑器打不开,怎样才能看到目标文件的内容呢?答案是可以利用objdump工具来反汇编二进制的目标文件,对可重定位目标文件和可执行的目标文件都可以反汇编。

我们利用一个c程序来进行举例,程序名gdbtest.c。

#include "stdio.h"
int main()
{
int x=3,y=5,z;
z=x+y;
printf("z=%d\n",z);
return 0;
}

利用gcc命令可以分别编译为gdbtest.o的可重定位目标文件和gdbtest可执行目标文件。

gcc -E -g -m32 gdbtest.c -o gdbtest.i
gcc -S -g -m32 gdbtest.i -o gdbtest.s
gcc -c -g -m32 gdbtest.s -o gdbtest.o
gcc -o0 -m32 -g gdbtest.c -o gdbtest

建议在objdump命令中使用-S选项,并与gcc命令中的-g选项一起配合使用。

利用以下命令:

objdump –S gdbtest.o>gdbtesto.txt
objdump –S gdbtest>gdbtest.txt

对这两个文件来进行反汇编,-S表示在反汇编后的内容中添加源代码,方便理解C语言源程序与IA-32机器级指令之间的对应关系。'>'这个符号表示将反汇编后的内容保存在文件中,在这里是保存为文本文件,为了防止内容太多输出到屏幕上不方便阅读。

gdbtest.o的可重定位目标文件的反汇编文件内容如下:

可执行目标文件的反汇编内容:

可重定位目标文件和可执行目标文件的一个很重要的区别就是指令的地址是不是从0地址开始。可重定位目标文件是完成一个子任务的独立模块,所以每个模块的地址都是从0地址开始,然而可执行目标文件中的的指令和数据都有一个确定的地址,是安装操作系统给定的储存器地址映射分配,也是在调试步骤中可以看到的地址。不是内存的物理地址,是虚拟地址。

GDB调试工具的使用

启动GDB调试工具

启动GDB调试工具,加载被调试的可执行文件。

命令 作用
1 gdb [可执行文件名] 启动GDB调试工具,并加载可执行文件
2 1.gdb
2. file [可执行文件名]
启动GDB调试工具
加载可执行文件

设置断点

设置断点,使程序运行到断点处停下来,方便查看程序运行的状态。

命令 作用
break main 在main函数的入口处设置断点
break gdbtest.c:3 在源程序gdbtest.c的di sa第3行处设置断点

启动程序运行

启动并运行已经加载的程序,程序在执行的设置的第一个断点会停下来。

命令 作用
run 启动程序运行,程序在断点处停下

查看程序运行时的当前状态

  1. 程序的当前断点位置

    含义:反映程序已经执行了哪些指令,下一步要执行哪一条指令。

    eip寄存器:保存了下一条将要执行的指令的地址。

    ir 显示所有寄存器的内容
    ir eip 只显示寄存器eip的内容
  2. 通用寄存器的内容:ir eax ebx ecx edx(或者ir)

  3. 存储器的单元内容:x/8xb Oxffd2bc

    x命令用于查看储存单元的内容,后跟一些参数选项。

    数据表示要显示的数据单元的个数。

    x表示存储单元的内容,十六进制形式。

    b表示要显示的储存单元的宽度,按字节显示(w:按4字节显示)

    后面的数据表示要显示的存储单元的起始地址。

    x/8xb Oxffd2bc这条内容就是表示从Oxffd2bc地址单元开始,显示8个字节的存储单元内容,并用十六进制表示。

  4. 查看运行时的当前状态

    说明: IA-32用栈来支持过程的嵌套调用,过程的入口参数,返回地址,被保存寄存器的值,被调用过程中的非静态局部变量等都会被保存在栈中。

    栈帧信息:

    当前栈帧范围: i r esp ebp (esp栈顶指针和ebp栈底指针 )

    当前栈帧字节数:y=R[ebp]-R[esp]+4 (不是命令,是计算方法)

    显示当前栈帧内容:

    x/yxb $esp
    x/zxw $esp //z=y/4

继续执行下一条指令或语句

命令 作用
si 执行一条机器指令
s 执行一条c语句

退出调试

命令 作用
quit 退出GDB调试过程

本次给大家分享的内容就到这里啦,觉得还不错的点个赞支持一下小编,你的肯定就是小编前进的动力。另外如果想了解更多计算机专业的知识和技巧的,献上我的个人博客北徯,另外需要各种资料的童鞋,可以关注我的微信公众号北徯,免费的PPT模板,各种资料等你来领。

计算机系统基础学习笔记(1)-基本GCC,objdump,GBD命令的使用的更多相关文章

  1. 【C#编程基础学习笔记】4---Convert类型转换

    2013/7/24 技术qq交流群:JavaDream:251572072  教程下载,在线交流:创梦IT社区:www.credream.com [C#编程基础学习笔记]4---Convert类型转换 ...

  2. 【C#编程基础学习笔记】6---变量的命名

    2013/7/24 技术qq交流群:JavaDream:251572072  教程下载,在线交流:创梦IT社区:www.credream.com [C#编程基础学习笔记]6---变量的命名 ----- ...

  3. 1.C#基础学习笔记3---C#字符串(转义符和内存存储无关)

    技术qq交流群:JavaDream:251572072  教程下载,在线交流:创梦IT社区:www.credream.com ------------------------------------- ...

  4. Java基础学习笔记总结

    Java基础学习笔记一 Java介绍 Java基础学习笔记二 Java基础语法之变量.数据类型 Java基础学习笔记三 Java基础语法之流程控制语句.循环 Java基础学习笔记四 Java基础语法之 ...

  5. Mysql数据库基础学习笔记

    Mysql数据库基础学习笔记 1.mysql查看当前登录的账户名以及数据库 一.单表查询 1.创建数据库yuzly,创建表fruits 创建表 ) ) ,) NOT NULL,PRIMARY KEY( ...

  6. 0003.5-20180422-自动化第四章-python基础学习笔记--脚本

    0003.5-20180422-自动化第四章-python基础学习笔记--脚本 1-shopping """ v = [ {"name": " ...

  7. Java基础学习笔记(一)

    Java基础学习笔记(一) Hello World 基础代码学习 代码编写基础结构 class :类,一个类即一个java代码,形成一个class文件,写于每个代码的前端(注意无大写字母) XxxYy ...

  8. C#RabbitMQ基础学习笔记

    RabbitMQ基础学习笔记(C#代码示例) 一.定义: MQ是MessageQueue,消息队列的简称(是流行的开源消息队列系统,利用erlang语言开发).MQ是一种应用程序对应用程序的通信方法. ...

  9. handlebars.js基础学习笔记

    最近在帮学校做个课程网站,就有人推荐用jquery+ajax+handlebars做网站前端,刚接触发现挺高大上的,于是就把一些基础学习笔记记录下来啦. 1.引用文件: jquery.js文件下载:h ...

随机推荐

  1. wr720n v4 折腾笔记(二):刷入不死Uboot

    0x01 前言 接着上节刷入Openwrt开始说起,此次开始刷入不死Uboot,刷入之后就可以在Uboot里面随便刷机,再也不怕成砖了. 固件附件地址: 下载地址1(还是之前一的包) flash文件地 ...

  2. 深入浅出C#结构体

    目录 1.应用背景 2.结构体解析 2.1.结构体存在栈中 2.2.结构体不需要手动释放 3.封装心跳包结构体 4.结构体静态帮助类 5.New出来的结构体是存在堆中还是栈中? 5.1.不带形参的结构 ...

  3. Python数据处理(持续更新)

    #打开txt文件 #打开txt文件 with open('day02.txt') as f: for line in f.readlines(): aline=line.strip() bline=a ...

  4. msys2 mingw64安装

    (1)安装msys2 (2)更新\etc\pacman.d\下的源文件 mirrorlist.msys Server = http://repo.msys2.org/msys/$arch/ Serve ...

  5. WEB应用之httpd基础入门(五)

    前文我们聊到了httpd的启动用户和相关权限的说明,资源压缩配置.https的实现,回顾请参考https://www.cnblogs.com/qiuhom-1874/p/12593675.html:今 ...

  6. stm32CubeMx+TrueSTUDIO+uc/os-III移植开发(一)

    自从接触到stm32cubemx后,发现使用起来很方便,用来做项目开发的话,由于是图形化操作界面,工作效率比较快.如果要学习操作系统,以前的IDE如keil,IAR等IDE对操作系统的代码调试起来不够 ...

  7. TensorFlow-Bitcoin-Robot:一个基于 TensorFlow LSTM 模型的 Bitcoin 价格预测机器人。

    简介 TensorFlow-Bitcoin-Robot:一个基于 TensorFlow LSTM 模型的 Bitcoin 价格预测机器人. 文章包括一下几个部分: 1.为什么要尝试做这个项目? 2.为 ...

  8. SpringBoot使用RedisTemplate操作Redis时,key值出现 \xac\xed\x00\x05t\x00\tb

    原因分析 原因与RedisTemplate源码中的默认序列化方式有关 defaultSerializer = new JdkSerializationRedisSerializer( classLoa ...

  9. Java static修饰符

    Java static修饰符 static修饰符是静态修饰符,大家写Java第一次接触static就是主函数中的pubic static void main的声明了. 那么static有什么用?本文来 ...

  10. 用SQL*Plus命令启动和关闭数据库

    用SQL*Plus命令启动和关闭数据库 1.启动方式 starup或startup open startup nomount startup mount startup read only [x] s ...