基本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. 洛谷 P1438 无聊的数列 题解

    原题链接 首先,我们考虑用差分解决问题. 用 \(x_i\) 表示原数列,\(a_i = x_i - x_{i-1}\) 那么,先普及一下差分: 如果我们只需要维护区间加值,单点求值的话,你会发现两个 ...

  2. HTTP 错误 500.21 模块 IIS Web Core

    如果出现如下图错误 就是iis没有安装Web Core模块 下载并安装DotNetCore.1.0.4_1.1.1-WindowsHosting.exe 安装完成之后会出现如下 证明安装成功.网站也可 ...

  3. ICLR 2020 | 抛开卷积,multi-head self-attention能够表达任何卷积操作

    近年来很多研究将nlp中的attention机制融入到视觉的研究中,得到很不错的结果,于是,论文侧重于从理论和实验去验证self-attention可以代替卷积网络独立进行类似卷积的操作,给self- ...

  4. Swagger2在DBA Service中生成RESTful API的实践

    目的与背景: 目的:对外暴露DBA Service必要的RESTful API,形成规整的API文档 背景:DBA Service后端采用Spring-boot,属于Spring家族,故生成API的工 ...

  5. 决战Leetcode: easy part(1-50)

    本博客是个人原创的针对leetcode上的problem的解法,所有solution都基本通过了leetcode的官方Judging,个别未通过的例外情况会在相应部分作特别说明. 欢迎互相交流! em ...

  6. redis处理高并发

    参考: https://www.cnblogs.com/wanlei/p/10464517.html 关于Redis处理高并发 Redis的高并发和快速原因 1.Redis是基于内存的,内存的读写速度 ...

  7. OpenCV-Python 轮廓:更多属性 | 二十四

    目标 在本章中,我们将学习 凸性缺陷以及如何找到它们 查找点到多边形的最短距离 匹配不同的形状 理论和代码 1. 凸性缺陷 我们看到了关于轮廓的第二章的凸包.从这个凸包上的任何偏差都可以被认为是凸性缺 ...

  8. ​知识图谱里的知识存储:neo4j的介绍和使用

      一般情况下,我们使用数据库查找事物间的联系的时候,只需要短程关系的查询(两层以内的关联).当需要进行更长程的,更广范围的关系查询时,就需要图数据库的功能. 而随着社交.电商.金融.零售.物联网等行 ...

  9. ASP.NET Core MVC通过IViewLocationExpander扩展视图搜索路径

    IViewLocationExpander API ExpandViewLocations Razor视图路径,视图引擎会搜索该路径. PopulateValues 每次调用都会填充路由 项目目录如下 ...

  10. 作为 attribute 和 property 的 value 及 Vue.js 的相关处理

    attribute 和 property 是 Web 开发中,比较容易混淆的概念,而对于 value,因其特殊性,更易困惑,本文尝试做一下梳理和例证 attribute 和 property 的概念 ...