- startup code
- HeapCreate
- 总结
- main执行之前
- main执行之后
生前:CRT startup code
看完课程,能够回答一下问题:
- C++进入点是main()嘛?
- 什么代码比main更早执行?
- 什么代码在main结束后执行?
- 为什么上述代码可以如此行为?
- Heap的结构如何?
- I/O的结构如何?
有一个启动代码:就是指定/ENTRY:function (Entry Point Symbol),他是一个function,函数的调用形式,返回值都有规定。需要完成的事:初始化CRT library,初始化静态对象。
main应该由启动代码来调用。
在使用gcc编译,-e
参数用于指定启动函数。
还记得内存管理中的内存分配的具体步骤嘛?
在内存管理部分,我们只关注前面两个函数,他们完成指定的内存分配策略。
- _heap_init(): 内存heap初始化。内存管理那一节我们已经讲过,是做16个header初始化工作。
- _ioinit():IO初始化,malloc分配256k大小(过程很复杂,见内存管理那个部分),用于IO初始化。比如初始化stdin(0)、stdout(1)、stderr(2)等东西(最多分配2048个文件描述符),不太感兴趣哦。
environ是pointer to pointer table,table中的每个entry都是代表环境变量的pointer to string,如下图:
也就是要得到main函数的参数,包括:__argc,__argv,_environ。其中的_environ中的内容就是startup code想要得到的,一般由 *** 作系统持有,程序需要把这个信息从 *** 作系统copy过来,所以分配内存来存放这些信息。环境变量具体的内容如下:
- GetCommandLineA(): 处理一个字符串。
- __crtGetEnvironmentStringsA(): 处理一堆字符串
- _setargv(): 命令行的参数
- _setenvp(): 引发11次内存分配,也就是上面说的,将 *** 作系统持有的环境变量复制到程序中来,需要内存来存放。
- _cinit():
上面的3-7步骤的详细内存分配计算图如下,每个分配的内存大小都需要膨胀,加debug信息32bytes,加cookie 8bytes,都是在由ioinit申请的内存中分配的:
HeapCreate前面说到,小于1k的内存交给SBH来分配,其余的内存交由 *** 作系统来分配,也就是HeapCreate来处理。
有128条链表用于管理内存分配,设计思想和SBH内存分配是一样的,只不过能分配的内存更大,对应的cookie只有上cookie,上cookie中存放当前块大小,前一个区块大小,单位为单元(一个单元8byte),而不是字节。
总结 main执行之前- 设置栈指针:为栈分配相关的位置,用来放一些局部变量和其他数据
- 初始化static静态和global全局变量,即data段的内容:把全局和静态变量初始化,放在相应的位置
- 将未初始化部分的全局变量赋初值:数值型short,int,long等为0,bool为FALSE,指针为NULL,等等,即.bss段的内容:将未设置初值的全局变量赋初值
- 全局对象初始化,在main之前调用构造函数
- 将main函数的参数,argc,argv等传递给main函数,然后才真正运行main函数:argc为整数,argv为指针的指针
执行全局的析构函数,可以用_onexit 注册一个函数,它会在main 之后执行; 。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)