网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
extern int create_process (char* program, char** arg_list);
int create_process (char* program, char** arg_list) {
pid_t child_pid;
child_pid = fork (); //fork 系统调用
if (child_pid != 0) return child_pid;
else {
execvp (program, arg_list); // execvp 运行一个新的程
abort ();
}
return 0;
}
`createprocess.c`
#include <stdlib.h>
#include <sys/types.h>
#include <unistd.h>
extern int create_process (char* program, char** arg_list); //创建子进程运行 命令 ls
int main () {
char* arg_list[] = {
“ls”,
“-l”,
“/etc/yum.repos.d/”,
NULL
};
create_process (“ls”, arg_list); return 0;
}
**进行编译:程序的二进制格式**
## 静态链接库
* 在 Linux 下面,二进制的程序也要有严格的格式,这个格式我们称为ELF(Executeable and Linkable Format,可执行与可链接格式)。这个格式可以根据编译的结果不同,分为不同的格式。
![在这里插入图片描述](https://img-blog.csdnimg.cn/571e93fdc8304524a064ffe0eba58ebd.png)
gcc -c -fPIC process.c
gcc -c -fPIC createprocess.c
* 在编译的时候,先做预处理工作,例如将头文件嵌入到正文中,将定义的宏展开,然后就是 真正的编译过程,最终编译成为.o 文件,这就是 ELF 的第一种类型,可重定位文件 (Relocatable File)还是.o 文件,不是一个可以直接运行的程序,这里面只是 部分代码片段。
* 可重定位文件的格式 每一个格的格式叫做节section![在这里插入图片描述](https://img-blog.csdnimg.cn/84d3b3a9464241d1b9ee458542f438e9.png)
* .text:放编译好的二进制可执行代码
* .data:已经初始化好的全局变量
* .rodata:只读数据,例如字符串常量、const 的变量
* .bss:未初始化全局变量,运行时会置 0
* .symtab:符号表,记录的则是函数和变量
* .strtab:字符串表、字符串常量和变量名
* .rel.text, .rel.data 就与重定位有关 , 存放被调用函数的位置
* `create_process` 这个函数作为库文件被重用,不能以.o 的形式存在,而是要形成库 文件,最简单的类型是静态链接库.a 文件(Archives),仅仅将一系列对象文件(.o)归档 为一个文件,使用命令 ar 创建。
`ar cr libstaticprocess.a process.o # libstaticprocess.a 里面只有一个.o,实际可以有多个`
* `gcc -o staticcreateprocess createprocess.o -L. -lstaticprocess`
-L 表示在当前目录下找.a 文件,-lstaticprocess 会自动补全文件名,比如 加前缀 lib,后缀.a,变成 libstaticprocess.a,找到这个.a 文件后,将里面的 process.o 取 出来,和 createprocess.o 做一个链接,形成二进制执行文件 staticcreateprocess。
`运行 # ./staticcreateprocess`
二进制文件格式
![在这里插入图片描述](https://img-blog.csdnimg.cn/2f6b6f9ff6de485cb63df635616afff6.png)
>
> 这个section 是多个.o 文件合并过的。
> 因而这些 section 被分成了需要加载到内存里面的代码段、数据段和不 需要加载到内存里面的部分,将小的 section 合成了大的段 segment,并且在最前面加一 个段头表(Segment Header Table)。在代码里面的定义为 struct elf32\_phdr 和 struct elf64\_phdr,(下面有)这里面除了有对于段的描述之外,最重要的是 p\_vaddr,这个是这个段加载 到内存的虚拟地址。
>
>
>
## 动态链接库
对于静态链接库, 缺点就是相同的代码段,如果被多个程序使用的 话,在内存里面就有多份,而且一旦静态链接库更新了,如果二进制执行文件不重新编译, 也不随着更新。
gcc -shared -fPIC -o libdynamicprocess.so process.o # 多了 -shared -fPIC
gcc -o dynamiccreateprocess createprocess.o -L. -ldynamicprocess
export LD_LIBRARY_PATH=. # 首先寻找动态链接库,然后加载它
./dynamiccreateprocess
* 动态链接库,就是 ELF 的第三种类型,共享对象文件(Shared Object)。
## 运行程序为进程
内核中,有这样一个数据结构,用来定义加载二进制文件的方法。
struct linux_binfmt {
struct list_head lh; struct module *module;
int (*load_binary)(struct linux_binprm *); int (*load_shlib)(struct file *); int (*core_dump)(struct coredump_params *cprm); unsigned long min_coredump;
/* minimal dump size */ } __randomize_layout;
![img](https://img-blog.csdnimg.cn/img_convert/05051ecb909dff68febada2132fbbc74.png)
![img](https://img-blog.csdnimg.cn/img_convert/82cb2913153dc7c9916e43e475df1a99.png)
**网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。**
**[需要这份系统化的资料的朋友,可以添加戳这里获取](https://bbs.csdn.net/topics/618668825)**
**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**
dn.net/topics/618668825)**
**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**