二.
1.预处理(preprocessing)->编译(compilation)->汇编(assembly)->链接(linking)
2.预编译 gcc -E hello.c -o hello.i
Hello.i中的内容:
# 1 "Hello.c"
# 1 "<built-in>"
# 1 "<command-line>"
# 1 "Hello.c"
# 1 "/usr/include/stdio.h" 1 3 4
# 28 "/usr/include/stdio.h" 3 4
# 1 "/usr/include/features.h" 1 3 4
# 313 "/usr/include/features.h" 3 4
# 1 "/usr/include/bits/predefs.h" 1 3 4
# 314 "/usr/include/features.h" 2 3 4
# 346 "/usr/include/features.h" 3 4
# 1 "/usr/include/sys/cdefs.h" 1 3 4
# 353 "/usr/include/sys/cdefs.h" 3 4
# 1 "/usr/include/bits/wordsize.h" 1 3 4
# 354 "/usr/include/sys/cdefs.h" 2 3 4
# 347 "/usr/include/features.h" 2 3 4
# 378 "/usr/include/features.h" 3 4
# 1 "/usr/include/gnu/stubs.h" 1 3 4
# 1 "/usr/include/bits/wordsize.h" 1 3 4
# 5 "/usr/include/gnu/stubs.h" 2 3 4
# 1 "/usr/include/gnu/stubs-32.h" 1 3 4
# 8 "/usr/include/gnu/stubs.h" 2 3 4
# 379 "/usr/include/features.h" 2 3 4
# 29 "/usr/include/stdio.h" 2 3 4
# 1 "/usr/lib/gcc/i486-linux-gnu/4.4.3/include/stddef.h" 1 3 4
# 211 "/usr/lib/gcc/i486-linux-gnu/4.4.3/include/stddef.h" 3 4
typedef unsigned int size_t;
# 35 "/usr/include/stdio.h" 2 3 4
# 1 "/usr/include/bits/types.h" 1 3 4
# 28 "/usr/include/bits/types.h" 3 4
# 1 "/usr/include/bits/wordsize.h" 1 3 4
# 29 "/usr/include/bits/types.h" 2 3 4
typedef unsigned char __u_char;
typedef unsigned short int __u_short;
typedef unsigned int __u_int;
typedef unsigned long int __u_long;
typedef signed char __int8_t;
typedef unsigned char __uint8_t;
typedef signed short int __int16_t;
typedef unsigned short int __uint16_t;
typedef signed int __int32_t;
typedef unsigned int __uint32_t;
__extension__ typedef signed long long int __int64_t;
__extension__ typedef unsigned long long int __uint64_t;
__extension__ typedef long long int __quad_t;
__extension__ typedef unsigned long long int __u_quad_t;
# 131 "/usr/include/bits/types.h" 3 4
# 1 "/usr/include/bits/typesizes.h" 1 3 4
# 132 "/usr/include/bits/types.h" 2 3 4
__extension__ typedef __u_quad_t __dev_t;
__extension__ typedef unsigned int __uid_t;
__extension__ typedef unsigned int __gid_t;
__extension__ typedef unsigned long int __ino_t;
__extension__ typedef __u_quad_t __ino64_t;
__extension__ typedef unsigned int __mode_t;
__extension__ typedef unsigned int __nlink_t;
__extension__ typedef long int __off_t;
__extension__ typedef __quad_t __off64_t;
__extension__ typedef int __pid_t;
__extension__ typedef struct { int __val[2]; } __fsid_t;
__extension__ typedef long int __clock_t;
__extension__ typedef unsigned long int __rlim_t;
__extension__ typedef __u_quad_t __rlim64_t;
__extension__ typedef unsigned int __id_t;
__extension__ typedef long int __time_t;
__extension__ typedef unsigned int __useconds_t;
__extension__ typedef long int __suseconds_t;
__extension__ typedef int __daddr_t;
__extension__ typedef long int __swblk_t;
__extension__ typedef int __key_t;
__extension__ typedef int __clockid_t;
__extension__ typedef void * __timer_t;
__extension__ typedef long int __blksize_t;
__extension__ typedef long int __blkcnt_t;
__extension__ typedef __quad_t __blkcnt64_t;
__extension__ typedef unsigned long int __fsblkcnt_t;
__extension__ typedef __u_quad_t __fsblkcnt64_t;
__extension__ typedef unsigned long int __fsfilcnt_t;
__extension__ typedef __u_quad_t __fsfilcnt64_t;
__extension__ typedef int __ssize_t;
typedef __off64_t __loff_t;
typedef __quad_t *__qaddr_t;
typedef char *__caddr_t;
__extension__ typedef int __intptr_t;
__extension__ typedef unsigned int __socklen_t;
# 37 "/usr/include/stdio.h" 2 3 4
# 45 "/usr/include/stdio.h" 3 4
struct _IO_FILE;
typedef struct _IO_FILE FILE;
# 65 "/usr/include/stdio.h" 3 4
typedef struct _IO_FILE __FILE;
# 75 "/usr/include/stdio.h" 3 4
# 1 "/usr/include/libio.h" 1 3 4
# 32 "/usr/include/libio.h" 3 4
# 1 "/usr/include/_G_config.h" 1 3 4
# 15 "/usr/include/_G_config.h" 3 4
# 1 "/usr/lib/gcc/i486-linux-gnu/4.4.3/include/stddef.h" 1 3 4
# 16 "/usr/include/_G_config.h" 2 3 4
# 1 "/usr/include/wchar.h" 1 3 4
# 83 "/usr/include/wchar.h" 3 4
typedef struct
{
int __count;
union
{
unsigned int __wch;
char __wchb[4];
} __value;
} __mbstate_t;
# 21 "/usr/include/_G_config.h" 2 3 4
typedef struct
{
__off_t __pos;
__mbstate_t __state;
} _G_fpos_t;
typedef struct
{
__off64_t __pos;
__mbstate_t __state;
} _G_fpos64_t;
# 53 "/usr/include/_G_config.h" 3 4
typedef int _G_int16_t __attribute__ ((__mode__ (__HI__)));
typedef int _G_int32_t __attribute__ ((__mode__ (__SI__)));
typedef unsigned int _G_uint16_t __attribute__ ((__mode__ (__HI__)));
typedef unsigned int _G_uint32_t __attribute__ ((__mode__ (__SI__)));
# 33 "/usr/include/libio.h" 2 3 4
# 53 "/usr/include/libio.h" 3 4
# 1 "/usr/lib/gcc/i486-linux-gnu/4.4.3/include/stdarg.h" 1 3 4
# 40 "/usr/lib/gcc/i486-linux-gnu/4.4.3/include/stdarg.h" 3 4
typedef __builtin_va_list __gnuc_va_list;
# 54 "/usr/include/libio.h" 2 3 4
# 170 "/usr/include/libio.h" 3 4
struct _IO_jump_t; struct _IO_FILE;
# 180 "/usr/include/libio.h" 3 4
typedef void _IO_lock_t;
struct _IO_marker {
struct _IO_marker *_next;
struct _IO_FILE *_sbuf;
int _pos;
# 203 "/usr/include/libio.h" 3 4
};
enum __codecvt_result
{
__codecvt_ok,
__codecvt_partial,
__codecvt_error,
__codecvt_noconv
};
# 271 "/usr/include/libio.h" 3 4
struct _IO_FILE {
int _flags;
char* _IO_read_ptr;
char* _IO_read_end;
char* _IO_read_base;
char* _IO_write_base;
char* _IO_write_ptr;
char* _IO_write_end;
char* _IO_buf_base;
char* _IO_buf_end;
char *_IO_save_base;
char *_IO_backup_base;
char *_IO_save_end;
struct _IO_marker *_markers;
struct _IO_FILE *_chain;
int _fileno;
int _flags2;
__off_t _old_offset;
unsigned short _cur_column;
signed char _vtable_offset;
char _shortbuf[1];
_IO_lock_t *_lock;
# 319 "/usr/include/libio.h" 3 4
__off64_t _offset;
# 328 "/usr/include/libio.h" 3 4
void *__pad1;
void *__pad2;
void *__pad3;
void *__pad4;
size_t __pad5;
int _mode;
char _unused2[15 * sizeof (int) - 4 * sizeof (void *) - sizeof (size_t)];
};
typedef struct _IO_FILE _IO_FILE;
struct _IO_FILE_plus;
extern struct _IO_FILE_plus _IO_2_1_stdin_;
extern struct _IO_FILE_plus _IO_2_1_stdout_;
extern struct _IO_FILE_plus _IO_2_1_stderr_;
# 364 "/usr/include/libio.h" 3 4
typedef __ssize_t __io_read_fn (void *__cookie, char *__buf, size_t __nbytes);
typedef __ssize_t __io_write_fn (void *__cookie, __const char *__buf,
size_t __n);
typedef int __io_seek_fn (void *__cookie, __off64_t *__pos, int __w);
typedef int __io_close_fn (void *__cookie);
# 416 "/usr/include/libio.h" 3 4
extern int __underflow (_IO_FILE *);
extern int __uflow (_IO_FILE *);
extern int __overflow (_IO_FILE *, int);
# 460 "/usr/include/libio.h" 3 4
extern int _IO_getc (_IO_FILE *__fp);
extern int _IO_putc (int __c, _IO_FILE *__fp);
extern int _IO_feof (_IO_FILE *__fp) __attribute__ ((__nothrow__));
extern int _IO_ferror (_IO_FILE *__fp) __attribute__ ((__nothrow__));
extern int _IO_peekc_locked (_IO_FILE *__fp);
extern void _IO_flockfile (_IO_FILE *) __attribute__ ((__nothrow__));
extern void _IO_funlockfile (_IO_FILE *) __attribute__ ((__nothrow__));
extern int _IO_ftrylockfile (_IO_FILE *) __attribute__ ((__nothrow__));
# 490 "/usr/include/libio.h" 3 4
extern int _IO_vfscanf (_IO_FILE * __restrict, const char * __restrict,
__gnuc_va_list, int *__restrict);
extern int _IO_vfprintf (_IO_FILE *__restrict, const char *__restrict,
__gnuc_va_list);
extern __ssize_t _IO_padn (_IO_FILE *, int, __ssize_t);
extern size_t _IO_sgetn (_IO_FILE *, void *, size_t);
extern __off64_t _IO_seekoff (_IO_FILE *, __off64_t, int, int);
extern __off64_t _IO_seekpos (_IO_FILE *, __off64_t, int);
extern void _IO_free_backup_area (_IO_FILE *) __attribute__ ((__nothrow__));
# 76 "/usr/include/stdio.h" 2 3 4
# 89 "/usr/include/stdio.h" 3 4
typedef _G_fpos_t fpos_t;
# 141 "/usr/include/stdio.h" 3 4
# 1 "/usr/include/bits/stdio_lim.h" 1 3 4
# 142 "/usr/include/stdio.h" 2 3 4
extern struct _IO_FILE *stdin;
extern struct _IO_FILE *stdout;
extern struct _IO_FILE *stderr;
extern int remove (__const char *__filename) __attribute__ ((__nothrow__));
extern int rename (__const char *__old, __const char *__new) __attribute__ ((__nothrow__));
extern int renameat (int __oldfd, __const char *__old, int __newfd,
__const char *__new) __attribute__ ((__nothrow__));
extern FILE *tmpfile (void) ;
# 186 "/usr/include/stdio.h" 3 4
extern char *tmpnam (char *__s) __attribute__ ((__nothrow__)) ;
extern char *tmpnam_r (char *__s) __attribute__ ((__nothrow__)) ;
# 204 "/usr/include/stdio.h" 3 4
extern char *tempnam (__const char *__dir, __const char *__pfx)
__attribute__ ((__nothrow__)) __attribute__ ((__malloc__)) ;
extern int fclose (FILE *__stream);
extern int fflush (FILE *__stream);
# 229 "/usr/include/stdio.h" 3 4
extern int fflush_unlocked (FILE *__stream);
# 243 "/usr/include/stdio.h" 3 4
extern FILE *fopen (__const char *__restrict __filename,
__const char *__restrict __modes) ;
extern FILE *freopen (__const char *__restrict __filename,
__const char *__restrict __modes,
FILE *__restrict __stream) ;
# 272 "/usr/include/stdio.h" 3 4
# 283 "/usr/include/stdio.h" 3 4
extern FILE *fdopen (int __fd, __const char *__modes) __attribute__ ((__nothrow__)) ;
# 296 "/usr/include/stdio.h" 3 4
extern FILE *fmemopen (void *__s, size_t __len, __const char *__modes)
__attribute__ ((__nothrow__)) ;
extern FILE *open_memstream (char **__bufloc, size_t *__sizeloc) __attribute__ ((__nothrow__)) ;
extern void setbuf (FILE *__restrict __stream, char *__restrict __buf) __attribute__ ((__nothrow__));
extern int setvbuf (FILE *__restrict __stream, char *__restrict __buf,
int __modes, size_t __n) __attribute__ ((__nothrow__));
extern void setbuffer (FILE *__restrict __stream, char *__restrict __buf,
size_t __size) __attribute__ ((__nothrow__));
extern void setlinebuf (FILE *__stream) __attribute__ ((__nothrow__));
extern int fprintf (FILE *__restrict __stream,
__const char *__restrict __format, ...);
extern int printf (__const char *__restrict __format, ...);
extern int sprintf (char *__restrict __s,
__const char *__restrict __format, ...) __attribute__ ((__nothrow__));
extern int vfprintf (FILE *__restrict __s, __const char *__restrict __format,
__gnuc_va_list __arg);
extern int vprintf (__const char *__restrict __format, __gnuc_va_list __arg);
extern int vsprintf (char *__restrict __s, __const char *__restrict __format,
__gnuc_va_list __arg) __attribute__ ((__nothrow__));
extern int snprintf (char *__restrict __s, size_t __maxlen,
__const char *__restrict __format, ...)
__attribute__ ((__nothrow__)) __attribute__ ((__format__ (__printf__, 3, 4)));
extern int vsnprintf (char *__restrict __s, size_t __maxlen,
__const char *__restrict __format, __gnuc_va_list __arg)
__attribute__ ((__nothrow__)) __attribute__ ((__format__ (__printf__, 3, 0)));
# 394 "/usr/include/stdio.h" 3 4
extern int vdprintf (int __fd, __const char *__restrict __fmt,
__gnuc_va_list __arg)
__attribute__ ((__format__ (__printf__, 2, 0)));
extern int dprintf (int __fd, __const char *__restrict __fmt, ...)
__attribute__ ((__format__ (__printf__, 2, 3)));
extern int fscanf (FILE *__restrict __stream,
__const char *__restrict __format, ...) ;
extern int scanf (__const char *__restrict __format, ...) ;
extern int sscanf (__const char *__restrict __s,
__const char *__restrict __format, ...) __attribute__ ((__nothrow__));
# 425 "/usr/include/stdio.h" 3 4
extern int fscanf (FILE *__restrict __stream, __const char *__restrict __format, ...) __asm__ ("" "__isoc99_fscanf") ;
extern int scanf (__const char *__restrict __format, ...) __asm__ ("" "__isoc99_scanf") ;
extern int sscanf (__const char *__restrict __s, __const char *__restrict __format, ...) __asm__ ("" "__isoc99_sscanf") __attribute__ ((__nothrow__));
# 445 "/usr/include/stdio.h" 3 4
extern int vfscanf (FILE *__restrict __s, __const char *__restrict __format,
__gnuc_va_list __arg)
__attribute__ ((__format__ (__scanf__, 2, 0))) ;
extern int vscanf (__const char *__restrict __format, __gnuc_va_list __arg)
__attribute__ ((__format__ (__scanf__, 1, 0))) ;
extern int vsscanf (__const char *__restrict __s,
__const char *__restrict __format, __gnuc_va_list __arg)
__attribute__ ((__nothrow__)) __attribute__ ((__format__ (__scanf__, 2, 0)));
# 476 "/usr/include/stdio.h" 3 4
extern int vfscanf (FILE *__restrict __s, __const char *__restrict __format, __gnuc_va_list __arg) __asm__ ("" "__isoc99_vfscanf")
__attribute__ ((__format__ (__scanf__, 2, 0))) ;
extern int vscanf (__const char *__restrict __format, __gnuc_va_list __arg) __asm__ ("" "__isoc99_vscanf")
__attribute__ ((__format__ (__scanf__, 1, 0))) ;
extern int vsscanf (__const char *__restrict __s, __const char *__restrict __format, __gnuc_va_list __arg) __asm__ ("" "__isoc99_vsscanf")
__attribute__ ((__nothrow__)) __attribute__ ((__format__ (__scanf__, 2, 0)));
# 504 "/usr/include/stdio.h" 3 4
extern int fgetc (FILE *__stream);
extern int getc (FILE *__stream);
extern int getchar (void);
# 532 "/usr/include/stdio.h" 3 4
extern int getc_unlocked (FILE *__stream);
extern int getchar_unlocked (void);
# 543 "/usr/include/stdio.h" 3 4
extern int fgetc_unlocked (FILE *__stream);
extern int fputc (int __c, FILE *__stream);
extern int putc (int __c, FILE *__stream);
extern int putchar (int __c);
# 576 "/usr/include/stdio.h" 3 4
extern int fputc_unlocked (int __c, FILE *__stream);
extern int putc_unlocked (int __c, FILE *__stream);
extern int putchar_unlocked (int __c);
extern int getw (FILE *__stream);
extern int putw (int __w, FILE *__stream);
extern char *fgets (char *__restrict __s, int __n, FILE *__restrict __stream)
;
extern char *gets (char *__s) ;
# 638 "/usr/include/stdio.h" 3 4
extern __ssize_t __getdelim (char **__restrict __lineptr,
size_t *__restrict __n, int __delimiter,
FILE *__restrict __stream) ;
extern __ssize_t getdelim (char **__restrict __lineptr,
size_t *__restrict __n, int __delimiter,
FILE *__restrict __stream) ;
extern __ssize_t getline (char **__restrict __lineptr,
size_t *__restrict __n,
FILE *__restrict __stream) ;
extern int fputs (__const char *__restrict __s, FILE *__restrict __stream);
extern int puts (__const char *__s);
extern int ungetc (int __c, FILE *__stream);
extern size_t fread (void *__restrict __ptr, size_t __size,
size_t __n, FILE *__restrict __stream) ;
extern size_t fwrite (__const void *__restrict __ptr, size_t __size,
size_t __n, FILE *__restrict __s);
# 710 "/usr/include/stdio.h" 3 4
extern size_t fread_unlocked (void *__restrict __ptr, size_t __size,
size_t __n, FILE *__restrict __stream) ;
extern size_t fwrite_unlocked (__const void *__restrict __ptr, size_t __size,
size_t __n, FILE *__restrict __stream);
extern int fseek (FILE *__stream, long int __off, int __whence);
extern long int ftell (FILE *__stream) ;
extern void rewind (FILE *__stream);
# 746 "/usr/include/stdio.h" 3 4
extern int fseeko (FILE *__stream, __off_t __off, int __whence);
extern __off_t ftello (FILE *__stream) ;
# 765 "/usr/include/stdio.h" 3 4
extern int fgetpos (FILE *__restrict __stream, fpos_t *__restrict __pos);
extern int fsetpos (FILE *__stream, __const fpos_t *__pos);
# 788 "/usr/include/stdio.h" 3 4
# 797 "/usr/include/stdio.h" 3 4
extern void clearerr (FILE *__stream) __attribute__ ((__nothrow__));
extern int feof (FILE *__stream) __attribute__ ((__nothrow__)) ;
extern int ferror (FILE *__stream) __attribute__ ((__nothrow__)) ;
extern void clearerr_unlocked (FILE *__stream) __attribute__ ((__nothrow__));
extern int feof_unlocked (FILE *__stream) __attribute__ ((__nothrow__)) ;
extern int ferror_unlocked (FILE *__stream) __attribute__ ((__nothrow__)) ;
extern void perror (__const char *__s);
# 1 "/usr/include/bits/sys_errlist.h" 1 3 4
# 27 "/usr/include/bits/sys_errlist.h" 3 4
extern int sys_nerr;
extern __const char *__const sys_errlist[];
# 827 "/usr/include/stdio.h" 2 3 4
extern int fileno (FILE *__stream) __attribute__ ((__nothrow__)) ;
extern int fileno_unlocked (FILE *__stream) __attribute__ ((__nothrow__)) ;
# 846 "/usr/include/stdio.h" 3 4
extern FILE *popen (__const char *__command, __const char *__modes) ;
extern int pclose (FILE *__stream);
extern char *ctermid (char *__s) __attribute__ ((__nothrow__));
# 886 "/usr/include/stdio.h" 3 4
extern void flockfile (FILE *__stream) __attribute__ ((__nothrow__));
extern int ftrylockfile (FILE *__stream) __attribute__ ((__nothrow__)) ;
extern void funlockfile (FILE *__stream) __attribute__ ((__nothrow__));
# 916 "/usr/include/stdio.h" 3 4
# 2 "Hello.c" 2
int main (int argc, char **argv)
{
printf("Hello World!\n");
return 0;
}
3.编译 gcc -S hello.i -o hello.s
Hello.s 中的内容:
.file "Hello.c"
.section .rodata
.LC0:
.string "Hello World!"
.text
.globl main
.type main, @function
main:
pushl %ebp
movl %esp, %ebp
andl $-16, %esp
subl $16, %esp
movl $.LC0, (%esp)
call puts
movl $0, %eax
leave
ret
.size main, .-main
.ident "GCC: (Ubuntu 4.4.3-4ubuntu5) 4.4.3"
.section .note.GNU-stack,"",@progbits
4.汇编
gcc -c hello.s -o hello.o
或 as hello.s -o hello.o
5.链接
编译器:扫描->语法分析->语义分析->源代码优化->代码生成->目标代码优化
*【有限状态机】算法
* lex 词法分析器
语法分析器(Grammar Parser) ->产生语法树(Syntax Tree)就是以表达式为节点的树
*上下文无关语法(Context-free Grammar)
语法分析工具:yacc(Yet Another Compiler Compiler)
语义分析:
语义分析器(Semantic Analyzer)
静态语义
动态语义
中间语言生成:
源代码优化器(Source Code Optimizer)
中间代码可以将编译器分为 前端和 后端:
模块之间的通信:(1)函数调用(2)变量的访问
两者都可以归结为 模块间符号的引用
2.4 静态链接
ELF文件类型:可重定位文件,可执行文件,共享目标文件,核心转储文件
Relocatable File
Executable File
Shared Object File
Core Dump File
SimpleSection.c
int printf(const char* format, ...);
int global_init_var = 84;
int global_uninit_var;
void func1( int i )
{
printf("%d\n", i);
}
int main(void)
{
static int static_var = 85;
static int static_var2;
int a = 1;
int b;
func1( static_var + static_var2 + a + b);
return a;
}
gcc -c SimpleSection.c
objdump -h SimpleSection.o 用来查看object内部的结构
* readelf 是专门针对ELF格式文件的解析器
* size命令可以用来查看ELF文件的代码段、数据段和BSS段的长度
ELF文件格式结构:
***
ELF Header
***
.text
***
.data
***
.bss
***
...other sections
***
Section header table
***
String Tables
Symbol Tables
....
整个链接过程正是基于符号来完成的
* nm 命令 可以用来查看*.o中的符号
$ nm SimpleSection.o
*函数签名(Function Signature)
*c++filt 可以用来解析被修饰过的名称
$ c++filt _ZN1N1C4funcEi
#ifdef __cplusplus
extern “C”{
#endif
void *memset (void *, int, size_t);
#ifdef __cplusplus
}
#endif
*弱符号 弱引用 强符号 强引用
如果一个程序被设计成可以支持单线程或多线程的模式,就可以通过弱引用的方法来判断当前的程序是链接到了单线程的Glibc库还是多线程的Glibc库(是否在编译时有-lpthread选项)
#include <stdio.h>
#include <pthread.h>
int pthread_create(
pthread_t *,
const pthread_attr_t*,
void * (*)(void*),
void *)
__attribute__ ((weak));
int main(){
if(pthread_create){
printf("This is multi-thread version\n");
// run the multi-thread version
// main_multi_thread()
}else{
printf("This is single-thread version\n");
}
}
$ gcc pthread.c -o pt
$ gcc pthread.c -lpthread -o pt
*在linux下我们可以用 strip 命令去掉调试信息!
四.静态链接
4.1 空间与地址分配
4.2符号解析与重定位
* objdump -d a.o 可以查看a.o的代码反汇编结果
* objdump -r a.o 可以查看a.o的重定位表
4.3COMMON块
4.4C++相关问题
*函数级别链接
4.5静态库链接
* ar -t libc.a ar工具查看静态库
e.g:
$ gcc -c -fno-builtin hello.c
//默认情况GCC会自作聪明的将”printf”替换成“puts”以提高运行速度
$ ar -x libc.a
//将libc.a 中的所有目标文件解压到当前目录
$ld hello.o printf.o
//链接却失败了!!!!!原因是缺少2个外部符号的定义
$ gcc -static --verbose -fno-builtin hello.c
--verbose 可以将编译链接的中间所有过程打印出来
cc1 GCC的c语言编译器
as GNU的汇编器
collect2 ld连接器的一个包装
4.6链接库控制
*.c
char * str = "Hello World!\n";
void print()
{
asm( "movl $13, %%edx \n\t"
"movl %0, %%ecx \n\t"
"movl $0, %%ebx \n\t"
"movl $4, %%eax \n\t"
"int $0x80 \n\t"
::"r"(str):"edx","ecx","ebx");
}
void exit()
{
asm("movl $42, %ebx \n\t"
"movl $1, %eax \n\t"
"int $0x80 \n\t");
}
void nomain()
{
print();
exit();
}
*.lds
ENTRY(nomain)
SECTIONS
{
. = 0x08048000 + SIZEOF_HEADERS;
tinytext : { *(.text) *(.data) *(.rodata)}
/DISCARD/ : {*(.comment)}
}
gcc -c -fno-builtin *.c
ld -static -T *.lds -o *.c *.o
4.7BFD库
本章小结
六.可执行文件的装载和执行
*页映射
6.3.1 进程的建立
(1)创建虚拟地址空间
(2)读取可执行文件头,并且建立虚拟空间与可执行文件的映射关系
*虚拟内存区域 (VMA Virtual Memory Area)
(3)将CPU指令寄存器设置成可执行文件的入口,启动执行
段的权限:
(1)以代码段为代表的权限为可读可执行的段
(2)以数据段和BSS段为代表的权限为可读可写的段
(3)以只读数据段为代表的权限为只读的段
对于权限相同的段,我们可以合并映射(可以减少页面的碎片,节省内存空间)
从section角度看ELF文件就是链接视图(Linking View)
从segment的角度看ELF文件就是执行视图(Execution View)
fly@Baby-o_o:~/Code/HelloWorld$ readelf -S SectionMapping.elf
There are 27 section headers, starting at offset 0x7d84c:
Section Headers:
[Nr] Name Type Addr Off Size ES Flg Lk Inf Al
[ 0] NULL 00000000 000000 000000 00 0 0 0
[ 1] .note.ABI-tag NOTE 080480f4 0000f4 000020 00 A 0 0 4
[ 2] .note.gnu.build-i NOTE 08048114 000114 000024 00 A 0 0 4
[ 3] .init PROGBITS 08048138 000138 000030 00 AX 0 0 4
[ 4] .text PROGBITS 08048170 000170 05eb3c 00 AX 0 0 16
[ 5] __libc_freeres_fn PROGBITS 080a6cb0 05ecb0 000b57 00 AX 0 0 16
[ 6] .fini PROGBITS 080a7808 05f808 00001c 00 AX 0 0 4
[ 7] .rodata PROGBITS 080a7840 05f840 018650 00 A 0 0 32
[ 8] __libc_subfreeres PROGBITS 080bfe90 077e90 000030 00 A 0 0 4
[ 9] __libc_atexit PROGBITS 080bfec0 077ec0 000004 00 A 0 0 4
[10] .eh_frame PROGBITS 080bfec4 077ec4 004dc0 00 A 0 0 4
[11] .gcc_except_table PROGBITS 080c4c84 07cc84 00011a 00 A 0 0 1
[12] .tdata PROGBITS 080c5f98 07cf98 000010 00 WAT 0 0 4
[13] .tbss NOBITS 080c5fa8 07cfa8 000018 00 WAT 0 0 4
[14] .ctors PROGBITS 080c5fa8 07cfa8 000008 00 WA 0 0 4
[15] .dtors PROGBITS 080c5fb0 07cfb0 00000c 00 WA 0 0 4
[16] .jcr PROGBITS 080c5fbc 07cfbc 000004 00 WA 0 0 4
[17] .data.rel.ro PROGBITS 080c5fc0 07cfc0 00002c 00 WA 0 0 4
[18] .got PROGBITS 080c5fec 07cfec 000008 04 WA 0 0 4
[19] .got.plt PROGBITS 080c5ff4 07cff4 00000c 04 WA 0 0 4
[20] .data PROGBITS 080c6000 07d000 000720 00 WA 0 0 32
[21] .bss NOBITS 080c6720 07d720 001b5c 00 WA 0 0 32
[22] __libc_freeres_pt NOBITS 080c827c 07d720 000018 00 WA 0 0 4
[23] .comment PROGBITS 00000000 07d720 000023 01 MS 0 0 1
[24] .shstrtab STRTAB 00000000 07d743 000107 00 0 0 1
[25] .symtab SYMTAB 00000000 07dc84 008200 10 26 967 4
[26] .strtab STRTAB 00000000 085e84 007397 00 0 0 1
Key to Flags:
W (write), A (alloc), X (execute), M (merge), S (strings)
I (info), L (link order), G (group), x (unknown)
O (extra OS processing required) o (OS specific), p (processor specific)
fly@Baby-o_o:~/Code/HelloWorld$ readelf -l SectionMapping.elf
Elf file type is EXEC (Executable file)
Entry point 0x8048170
There are 6 program headers, starting at offset 52
Program Headers:
Type Offset VirtAddr PhysAddr FileSiz MemSiz Flg Align
LOAD 0x000000 0x08048000 0x08048000 0x7cd9e 0x7cd9e R E 0x1000
LOAD 0x07cf98 0x080c5f98 0x080c5f98 0x00788 0x022fc RW 0x1000
NOTE 0x0000f4 0x080480f4 0x080480f4 0x00044 0x00044 R 0x4
TLS 0x07cf98 0x080c5f98 0x080c5f98 0x00010 0x00028 R 0x4
GNU_STACK 0x000000 0x00000000 0x00000000 0x00000 0x00000 RW 0x4
GNU_RELRO 0x07cf98 0x080c5f98 0x080c5f98 0x00068 0x00068 R 0x1
Section to Segment mapping:
Segment Sections...
00 .note.ABI-tag .note.gnu.build-id .init .text __libc_freeres_fn .fini .rodata __libc_subfreeres __libc_atexit .eh_frame .gcc_except_table
01 .tdata .ctors .dtors .jcr .data.rel.ro .got .got.plt .data .bss __libc_freeres_ptrs
02 .note.ABI-tag .note.gnu.build-id
03 .tdata .tbss
04
05 .tdata .ctors .dtors .jcr .data.rel.ro .got .got.plt
我们可以通过查看 “/proc”来查看进程的虚拟空间分布:
$ ./SectionMapping.elf &
[1]21963
$cat /proc/21963/maps
fly@Baby-o_o:~/Code/HelloWorld$ ./SectionMapping.elf &
[1] 3869
fly@Baby-o_o:~/Code/HelloWorld$ cat /proc/3869/maps
08048000-080c5000 r-xp 00000000 08:01 18874525 /home/fly/Code/HelloWorld/SectionMapping.elf
080c5000-080c7000 rwxp 0007c000 08:01 18874525 /home/fly/Code/HelloWorld/SectionMapping.elf
080c7000-080c9000 rwxp 00000000 00:00 0
08ee4000-08f06000 rwxp 00000000 00:00 0 [heap]
b7746000-b7747000 r-xp 00000000 00:00 0 [vdso]
bfc5b000-bfc70000 rw-p 00000000 00:00 0 [stack]
*匿名虚拟内存区域(Anonymous Virtual Memory Area)
p192
6.5 linux内核装载ELF过程简介
(1)用户层面:bash进程->fork()->execve()->执行elf->bash进程等待其结束
execve()被定义在unistd.h 原型:
int execve(const char *filename, char *const argv[], char *const envp[]);
*p174
七.动态链接
静态库的缺点:
(1)浪费内存空间,不同的程序调用相同的静态库,会在内存中存在两个相同的静态库的副本
(2)影响程序的更新、部署、和发布
一旦有一个模块需要更新,整个程序就要重新链接,发布给客户
为了解决上述静态库的缺点,动态库应运而生,把链接这个工程推迟到了运行时进行(动态链接 Dynamic Linking 的基本思想)
这样做的好处不仅仅是节省内存,还可以减少物理页面的换入换出,增加CPU缓存的命中率(因为不同进程间的数据和指令访问都集中在了同一个共享模块上)
缺点:当新旧模块接口不兼容的时候,导致原有的程序无法运行!!!
*延迟绑定
Example: SimpleDynamicalLinking
a.c
#include "Lib.h"
int main(){
foobar(1);
return 0;
}
b.c
#include "Lib.h"
int main(){
foobar(2);
return 0;
}
Lib.c
#include <stdio.h>
void foobar(int i){
printf("Printing from Lib.so %d\n", i);
}
Lib.h
#ifndef LIB_H
#define LIB_H
void foobar(int i);
#endif
$ gcc -fPIC -shared -o Lib.so Lib.c
$ gcc -o a a.c ./Lib.so
Libc -> [Compiler] -> Lib.o -> [Linker] <- [C Runtime Library]
|
\/
Lib.so
|
\/
a.c -> [Compiler] -> a.o -> [Linker] -> a
7.3 地址无关代码
静态库 使用编译时重定位
动态库 使用装载时重定位
如果上面只使用 -shared 参数, 则输出的共享对象 就是使用的 装载时重定位的方法
但是 这样虽然可以解决动态模块中有绝对地址引用,但是缺点是:指令部分无法在多个进程之间共享
这样就失去了动态链接节省内存的一大优势!!!
那么 -fPIC 参数有什么作用呢?
解决 共享对象指令中绝对地址重定位问题基本思路:
把指令中那些需要修改的部分分离出来,和数据部分放在一起,这样指令部分可以保持不变,而数据部分可以个在每个进程中拥有一个副本
这种方案目前被称为:地址无关代码(PIC, Position-independent Code)
共享对象模块中的地址引用按照是否为跨模块分为
(1)模块内部引用
(2)模块外部引用
按照引用方式:
(1)指令引用
(2)数据访问
于是我们得到4中情况:
(1)模块内部 的 函数调用、跳转。
(2)模块内部 的 数据访问,比如模块中定义的全局变量、静态变量。
(3)模块外部 的 函数调用、跳转。
(4)模块外部 的 数据访问,比如其他模块中定义的全局变量。
p217