- 浏览: 261345 次
- 性别:
- 来自: 济南
文章分类
- 全部博客 (303)
- c (31)
- c++ (16)
- java (18)
- c# (1)
- python (3)
- java web (6)
- oracle (7)
- sqlserver (2)
- mysql (2)
- android (24)
- android系统 (15)
- android多媒体部分 (15)
- android游戏 (12)
- linux (26)
- javaScript (1)
- ajax (1)
- node JS (2)
- html (5)
- apache (3)
- jboss (1)
- weblogic (0)
- 通信协议 (10)
- 云计算 (1)
- 分布式 (5)
- ejb (1)
- webservice (5)
- 设计模式 (16)
- JNI (6)
- swing (13)
- 版本控制 (1)
- UML (1)
- xml (4)
- spring (5)
- hibernate (5)
- struts1 (3)
- struts2 (4)
- ibatis (0)
- tomcat (2)
- 心得体会 (1)
- css (1)
- 嵌入式 (41)
- arm体系结构 (10)
ELF 文件
三种格式:
1 可重定位的目标文件
可用readelf工具读取
readelf -a 文件名
ELF header
program header table
section header table
目标文件的布局
起始文件地址 Section或Header
0 ELF Header
0x34 .text
0x60 .data
0x98 .bss (此段为空)
0x98 .shstrtab
0xc8 Section Header Table
0x208 .symtab
0x288 .strtab
0x2b0 .rel.text
2 可执行文件
3 共享库
函数调用
示例代码:
int bar(int c,int d){
int e=c+d;
reutrn e;
}
int foo(int a,int b){
return bar(a,b);
}
int main(void){
foo(2,3);
return 0;
}
编译:gcc test.c -g (添加-g选项,在反汇编时可以把c代码和汇编代码穿插显示)
反汇编:objdump -dS test
示例代码如下:080483b4 <bar>:
//将foo函数的ebp压栈,
80483b4: 55
push %ebp
//给epb赋新值指向bar 函数的栈底,
80483b5: 89 e5
mov %esp,%ebp
80483b7: 83 ec 10
sub $0x10,%esp
//访问两个参数
80483ba: 8b 45 0c
mov 0xc(%ebp),%eax
80483bd: 8b 55 08
mov 0x8(%ebp),%edx
80483c0: 8d 04 02
lea (%edx,%eax,1),%eax
80483c3: 89 45 fc
mov %eax,-0x4(%ebp)
//把e的值读到eax寄存器中,
80483c6: 8b 45 fc
mov -0x4(%ebp),%eax
//push %ebp和mov %esp,%ebp的逆操作,把ebp的值赋值给esp,
80483c9: c9
leave
//call指令的逆操作,
80483ca: c3
ret
080483cb <foo>:
//先将epb寄存器的值压栈,esp的值-4送给ebp,(esp指向栈顶,ebp指向栈底)
80483cb: 55
push %ebp
80483cc: 89 e5
mov %esp,%ebp
80483ce: 83 ec 08
sub $0x8,%esp
80483d1: 8b 45 0c
mov 0xc(%ebp),%eax
80483d4: 89 44 24 04
mov %eax,0x4(%esp)
80483d8: 8b 45 08
mov 0x8(%ebp),%eax
80483db: 89 04 24
mov %eax,(%esp)
//把返回地址压栈调用bar函数
80483de: e8 d1 ff ff ff
call 80483b4 <bar>
80483e3: c9
leave
80483e4: c3
ret
080483e5 <main>:
80483e5: 55
push %ebp
80483e6: 89 e5
mov %esp,%ebp
80483e8: 83 ec 08
sub $0x8,%esp
//foo(2,3); 参数是从右向左依次压栈
80483eb: c7 44 24 04 03 00 00
movl $0x3,0x4(%esp)
80483f2: 00
80483f3: c7 04 24 02 00 00 00
movl $0x2,(%esp)
//调用call指令:
// 1 把call的下一条指令 0x80483ff 压栈,同时esp-4,现值为
// 2 修改程序计数器eip,跳转到foo函数的开头执行 (80483cb <foo>)
80483fa: e8 cc ff ff ff
call 80483cb <foo>
80483ff: b8 00 00 00 00
mov $0x0,%eax
8048404: c9
leave
8048405: c3
ret
8048406: 90
nop
8048407: 90
nop
8048408: 90
nop
8048409: 90
nop
804840a: 90
nop
804840b: 90
nop
804840c: 90
nop
804840d: 90
nop
804840e: 90
nop
804840f: 90
nop
main函数和启动例程
汇编程序的入口是_start
汇编的链接步骤是:
as hello.s -o hello.o
ld hello.o -o hello
c程序的入口是main()
c程序的编译:(-o 给文件重新命名,-v查看编译过程)
gcc -s test.c //生成汇编代码
gcc -c test.s //生成目标文件
gcc test.o //生成可执行文件
readelf/nm命令查看.o文件
变量的存储布局
示例代码:
#include<stdio.h>
/*全局变量*/
const int A=10;
int a=20;
static int b=30;
int c;
int main(void){
/*局部变量*/
static int a=40;
char b[]="Hello World";
register int c=50;
printf("Hello World %d\n",c)
return 0;
}
编译: gcc test.c -g
查看符号表:readelf -a a.out
//local 局部变量
49: 0804a01c 4 OBJECT LOCAL DEFAULT 24 b
50: 0804a020 4 OBJECT LOCAL DEFAULT 24 a.1709
//global全局变量
69: 08048570 4 OBJECT GLOBAL DEFAULT 15 A
73: 0804a02c 4 OBJECT GLOBAL DEFAULT 25 c
74: 0804a018 4 OBJECT GLOBAL DEFAULT 24 a
示例代码如下:
Section Headers:
[Nr] Name Type Addr Off Size ES Flg Lk
Inf Al
[ 0] NULL 00000000 000000 000000 00 0
0 0
[ 1] .interp PROGBITS 08048154 000154 000013 00 A 0
0 1
[ 2] .note.ABI-tag NOTE 08048168 000168 000020 00 A 0
0 4
[ 3] .note.gnu.build-i NOTE 08048188 000188 000024 00 A 0
0 4
[ 4] .gnu.hash GNU_HASH 080481ac 0001ac 000020 04 A 5
0 4
[ 5] .dynsym DYNSYM 080481cc 0001cc 000060 10 A 6
1 4
[ 6] .dynstr STRTAB 0804822c 00022c 000067 00 A 0
0 1
[ 7] .gnu.version VERSYM 08048294 000294 00000c 02 A 5
0 2
[ 8] .gnu.version_r VERNEED 080482a0 0002a0 000030 00 A 6
1 4
[ 9] .rel.dyn REL 080482d0 0002d0 000008 08 A 5
0 4
[10] .rel.plt REL 080482d8 0002d8 000020 08 A 5
12 4
[11] .init PROGBITS 080482f8 0002f8 00002e 00 AX 0
0 4
[12] .plt PROGBITS 08048330 000330 000050 04 AX 0
0 16
[13] .text PROGBITS 08048380 000380 0001cc 00 AX 0
0 16
[14] .fini PROGBITS 0804854c 00054c 00001a 00 AX 0
0 4
//
[15] .rodata PROGBITS 08048568 000568 00001c 00 A 0
0 4
[16] .eh_frame_hdr PROGBITS 08048584 000584 00002c 00 A 0
0 4
[17] .eh_frame PROGBITS 080485b0 0005b0 0000a4 00 A 0
0 4
[18] .ctors PROGBITS 08049f14 000f14 000008 00 WA 0
0 4
[19] .dtors PROGBITS 08049f1c 000f1c 000008 00 WA 0
0 4
[20] .jcr PROGBITS 08049f24 000f24 000004 00 WA 0
0 4
[21] .dynamic DYNAMIC 08049f28 000f28 0000c8 08 WA 6
0 4
[22] .got PROGBITS 08049ff0 000ff0 000004 04 WA 0
0 4
[23] .got.plt PROGBITS 08049ff4 000ff4 00001c 04 WA 0
0 4
[24] .data PROGBITS 0804a010 001010 000014 00 WA 0
0 4
[25] .bss NOBITS 0804a024 001024 00000c 00 WA 0
0 4
[26] .comment PROGBITS 00000000 001024 000055 01 MS 0
0 1
[27] .debug_aranges PROGBITS 00000000 001079 000020 00 0
0 1
[28] .debug_pubnames PROGBITS 00000000 001099 00002d 00 0
0 1
[29] .debug_info PROGBITS 00000000 0010c6 00010e 00 0
0 1
[30] .debug_abbrev PROGBITS 00000000 0011d4 000085 00 0
0 1
[31] .debug_line PROGBITS 00000000 001259 00003e 00 0
0 1
[32] .debug_frame PROGBITS 00000000 001298 000038 00 0
0 4
[33] .debug_str PROGBITS 00000000 0012d0 000076 01 MS 0
0 1
[34] .debug_loc PROGBITS 00000000 001346 00002c 00 0
0 1
[35] .shstrtab STRTAB 00000000 001372 000164 00 0
0 1
[36] .symtab SYMTAB 00000000 001ac8 0004f0 10 37
55 4
[37] .strtab STRTAB 00000000 001fb8 000229 00 0
查看内容:hexdump -c a.out
示例代码如下:
0000560 � 203 � \b [ � \0 \0 003 \0 \0 \0 001 \0 002 \0
0000570 \n \0 \0 \0 H e l l o W o r l
c 语言的作用域:
函数作用域: function scope 整个函数中有效
文件作用域: file scope 文件的开始到末尾都有效
块作用域: block scope {}之间有效
函数原型作用域: function protorype scope 在函数原型内有效
同一命名空间的重名标示符:
语名标号单独属于一个命名空间
struct/enum/union属于一个命名空间
其他标识符为一命名空间,宏定义覆盖所有,内层作用域覆盖处层作用域
标识符的链接属性:
外部链接:external linkage 多个文件中声明多次也都代表同一标识符
内部链接:internal linkage 某个文件中声明多次也都代表同一标识符
无链接:no linkage
存储类修饰符:
static 静态分配,内部链接
auto 自动在栈上分配空间,返回时自动释放,不能作用文件作用域
register 分配专问寄存器存储,如果分配不开,则作为auto处理,不能作用文件作用
域
extern
typedef
变量的生命周期:
静态生存期: 内部外部链接,static ,直到程序结束
自动生存期: 无链接,块作用域,退出块作用域时释放
动态分配生存期:malloc函数在堆空间中分配内存,调用free释放内存
发表评论
-
linux 系统编程学习笔记四
2012-12-13 21:41 851文件系统 ext2文件系统 分区的格式化工具mkfs 文 ... -
linux 系统编程学习笔记二
2012-12-06 22:15 762lseek 文件在打开时读写位置是0,如果是以O_APP ... -
linux 系统编程学习笔记三
2012-12-10 20:54 822mmap 可以把磁盘文件的一部分直接映射到内存,这样文件中的 ... -
linux 系统编程学习笔记一
2012-12-05 21:29 796文件与i/o 汇编程序的he ... -
c语言学习笔记二十五
2012-12-03 20:22 693c标准库 1 字符串操作函数 初始化字符串#includ ... -
c语言学习笔记二十六
2012-12-03 21:42 7272 标准I/O库函数 文件的基本概念 文本文件(源文件) ... -
c语言学习笔记二十四
2012-11-19 20:57 591函数接口 示例代码如下: man page ht ... -
c语言学习笔记二十三
2012-11-11 16:56 701指针 占4个字节的存储空间 概念 把一个变量所在的内存单元 ... -
c语言学习笔记二十二
2012-11-09 00:17 621自动处理头文件的依赖关系 all:test34 test3 ... -
c语言学习笔记一
2012-09-16 01:45 601数据类型 基本类型 : 整型 整数 ... -
c语言学习笔记二
2012-09-16 12:13 593函数 注: linux下运行.c执行以下命令: gc ... -
c语言学习笔记三
2012-09-16 13:01 546return 语句 如果没有返回结果,结束当前函数的 ... -
c语言学习笔记四
2012-09-16 13:59 632结构体 复合类型和结构体 复合类型: 示例代码: ... -
c语言学习笔记五
2012-09-16 16:13 547gdb调试工具 sudo su 切换到root用户 m ... -
c语方学习笔记六
2012-09-16 22:18 745排序 sb东西 算法(alorithm) 解决一类计算问题 ... -
c 语方学习笔记七
2012-09-16 22:39 733线性查找 示例代码: #include<stdio ... -
c语言学习笔记八
2012-09-16 23:48 491数据结构的概念 学这东西感觉一下又回到了十年前 数据结构( ... -
c语言学习笔记九
2012-09-17 12:40 578运算符 逻辑运算 XOR 的真值表 A B A ... -
c语言学习笔记十
2012-09-17 13:51 557内存与地址 32位x86 ,从0x0000 0000到0xf ... -
c语言学习笔记十一
2012-09-17 15:36 561文件I/O .data msg : .string ...
相关推荐
综上所述,Go语言学习笔记通常涵盖了语言的起源、设计哲学、语法特点、环境配置、核心功能使用等多方面知识。通过理解这些知识点,开发者能够有效地掌握Go语言,为开发高性能的应用程序打下坚实的基础。随着对Go语言...
"Go语言学习笔记" Go语言学习笔记是关于Go语言基础入门篇的笔记,主要介绍Go语言基础语法、数据类型、逻辑语句等相关知识点。 语言概述 Go语言是一种开源的编程语言,能让构造简单、可靠且高效的软件变得容易。Go...
本压缩包包含的“C语言学习笔记”是一份详尽的C语言学习资料,旨在帮助读者深入理解和掌握C语言的基本概念、语法结构以及实际应用技巧。 一、C语言基础 C语言的基础包括变量、数据类型、运算符、流程控制等。变量...
C语言是一种广泛使用的编程语言,尤其适合底层系统开发和嵌入式系统。本文将深入探讨C语言的基础知识,包括数据类型和表达式、变量与常量、输出输入以及运算符号。 首先,我们要理解数据类型和表达式。在C语言中,...
根据提供的信息,我们可以总结出这份文档是关于Go语言学习笔记的部分内容,主要涵盖了Go语言的基础概念、语法结构、数据类型以及并发模型等关键知识点。以下是对这些知识点的详细解析: ### Go语言概述 Go(也称作...
根据提供的文件内容,以下是对Go语言学习笔记的详细知识点阐述。 Go语言是Google开发的一种静态类型、编译型、并发型,并具有垃圾回收功能的编程语言。它由Robert Griesemer、Rob Pike和Ken Thompson于2007年9月...
本压缩包“C语言学习笔记.zip”包含了丰富的C语言学习资料,主要集中在“c-notes-master”这个子目录下,适合初学者和有一定基础的程序员进行深入学习。 在C语言的学习中,首先要掌握的基本概念包括变量、数据类型...
C语言是一种强大的编程语言,它的学习笔记涵盖了从基础知识到高级特性的全方位内容。在学习C语言时,首先需要了解文件类型,如C语言的源文件通常以`.c`为扩展名,而C++源文件是`.cpp`。头文件`.h`用于包含函数声明和...
本压缩包包含了一份精心整理的C语言学习笔记,旨在帮助读者深入理解和掌握C语言的核心概念与技术。 笔记首先从基础语法开始,包括变量、数据类型、运算符以及表达式。C语言中的数据类型有基本类型如int、char、...
本学习笔记旨在全面覆盖C语言的基础部分,帮助初学者掌握其核心概念和语法。 一、C语言概述 C语言由贝尔实验室的Dennis Ritchie在1972年创造,它的特点是运行效率高,程序控制性强,语法简洁。C语言是许多现代编程...
本学习笔记主要涵盖了Go语言的基础知识,包括类型、表达式、函数、数据结构、方法、接口、并发以及包管理等多个方面,并附带了源码分析和附录中的工具、调试和测试等内容。 一、类型 1. 变量:Go语言中的变量需要...
本资源包"**C语言学习笔记——基础知识篇**"是为C语言初学者或需要复习C语言概念的人准备的宝贵资料。 首先,我们要了解C语言的基础知识。C语言的语法结构基于ALGOL60和BCPL,由Dennis Ritchie在贝尔实验室设计和...
这份学习笔记主要涵盖C语言的基础知识,包括数据类型、输入输出以及一些关键的运算规则。 首先,C语言的数据类型是编程的基础。它包括基本类型、构造类型、指针类型和空类型。基本类型包括整型(如int)、字符型...
这份“C语言学习笔记”涵盖了从入门到进阶的各种知识点,旨在帮助学习者掌握C语言的核心概念和技术。 首先,C语言的基础部分包括语法结构,如变量声明、数据类型(整型、浮点型、字符型等)、运算符(算术、比较、...
【C语言和C++Builder学习笔记】 C语言和C++是两种广泛使用的编程语言,它们在软件开发领域占据着重要地位。C语言以其简洁、高效和底层操作能力著称,而C++则在C的基础上引入了面向对象编程,极大地扩展了其应用范围...
"C语言基础学习笔记"是一个关于C语言入门的学习资源,旨在帮助初学者掌握C语言的基本概念、语法和编程技巧。 在C语言中,我们首先会接触到基本的数据类型,如整型(int)、浮点型(float、double)、字符型(char)...
学习C语言,首先要明确地告诉自己:C语言是世界上最值得学习的语言。如今决定学习任何一门其他语言之前,都要先考察这个语言有什么成功项目吗?唯独C语言没有必要问,因为世界上所有最最重要的的系统中,都必然有...
这份“C语言个人学习笔记汇总”旨在帮助初学者和进阶者巩固和深化对C语言的理解。 在学习C语言的过程中,首先需要掌握的是基本语法,包括变量定义、数据类型(如整型、浮点型、字符型等)、运算符(算术、比较、...
**C语言基础学习笔记概述** C语言是一种强大的、低级的编程语言,广泛应用于系统编程、软件开发、设备驱动和嵌入式系统等。它的设计目标是提供一种简洁、高效且可移植的语言,使得程序员能够对计算机硬件进行直接...