DB Version:9.5.3

环境:CentOS7.x

调试工具:GDB

source:src/backend/main/main.c

  1. 56 /*
  2. 57 * Any Postgres server process begins execution here.
  3. 58 */
  4. 59 int
  5. 60 main(int argc, char *argv[])
  6. 61 {
  7. 62 bool do_check_root = true;
  8. 63 sleep(30);
  9. 64 progname = get_progname(argv[0]);

修改一下代码,睡它30s。或者执行postgres可执行文件,set args 也OK。

启动数据库:

  1. /usr/local/psql-9.5.3/bin/pg_ctl -D db2/ -l logfile start -m fast

查看后台进程PID:

  1. [postgres@localhost ~]$ ps -ef |grep postgres
  2. root 57843 57805 0 10:58 pts/1 00:00:00 su - postgres
  3. postgres 57844 57843 0 10:58 pts/1 00:00:00 -bash
  4. postgres 1 0 11:01 pts/1 00:00:00 /usr/local/psql-9.5.3/bin/postgres -D db2
  5. postgres 57981 57844 0 11:02 pts/1 00:00:00 ps -ef
  6. postgres 57982 57844 0 11:02 pts/1 00:00:00 grep --color=auto postgres 

进入调试模式,需要等30s:

  1. cgdb -p 57977
  2.  
  3. (gdb) b main.c:64
  4. Breakpoint 1 at 0x676229: file main.c, line 64.
  5. (gdb) c
  6. Continuing.
  7.  
  8. Breakpoint 1, main (argc=3, argv=0x7ffceddcbe88) at main.c:64
  9. (gdb)

首先pg进入main函数,最先是获取到progname:

  1. (gdb) p argv[0]
  2. $1 = 0x7ffceddcd6d6 "/usr/local/psql-9.5.3/bin/postgres"

其实这个里面还是根据main函数的第一个参数进行字符串拆分计算出progname

具体可以跟一下函数"get_progname":

  1. (gdb) p progname
  2. $15 = 0x14f9010 "postgres"

初始化内存(MemoryContextInit):

这个是数据库启动的时候初始化的第一块内存,我们一起来看看里面的内容。

在看这个之前,我们先来了解一下PG几个关于内存的结构体

typedef struct MemoryContextData *MemoryContext;

  1. typedef struct MemoryContextData
  2. {
  3. NodeTag type; /* identifies exact kind of context */
  4. /* these two fields are placed here to minimize alignment wastage: */
  5. bool isReset; /* T = no space alloced since last reset */
  6. bool allowInCritSection; /* allow palloc in critical section */
  7. MemoryContextMethods *methods; /* virtual function table */
  8. MemoryContext parent; /* NULL if no parent (toplevel context) */
  9. MemoryContext firstchild; /* head of linked list of children */
  10. MemoryContext nextchild; /* next child of same parent */
  11. char *name; /* context name (just for debugging) */
  12. MemoryContextCallback *reset_cbs; /* list of reset/delete callbacks */
  13. } MemoryContextData;

我们先看看MemoryContextData是如何被初始化的:

  1. 函数 MemoryContextCreate
  2.  
  3. MemSet(node, 0, size);
  4. node->type = tag;
  5. node->methods = methods;
  6. node->parent = NULL; /* for the moment */
  7. node->firstchild = NULL;
  8. node->nextchild = NULL;
  9. node->isReset = true;
  10. node->name = ((char *) node) + size;
  11. strcpy(node->name, name);
  12.  
  13. (gdb) p *node
  14. $31 = {type = T_AllocSetContext, isReset = 1 '\001', allowInCritSection = 0 '\000', methods = 0xd1bbe0 <AllocSetMethods>, parent = 0x0, firstchild = 0x0, nextchild = 0x0, name = 0x
  15. 14f9c70 "TopMemoryContext", reset_cbs = 0x0}

其实这里最重要的是methods这个参数,这是个函数指针,还有里面其实最主要的就是内存上下文的父子关系

我们回头再细研究这个东东。

函数返回的是MemoryContext转成AllocSet.就是下面的结构体,其实MemoryContextData成了它的header。

这就是后面的NODE那个大enum,直接小转大。

  1. typedef struct AllocSetContext
  2. {
  3. MemoryContextData header; /* Standard memory-context fields */
  4. /* Info about storage allocated in this context: */
  5. AllocBlock blocks; /* head of list of blocks in this set */
  6. AllocChunk freelist[ALLOCSET_NUM_FREELISTS]; /* free chunk lists */
  7. /* Allocation parameters for this context: */
  8. Size initBlockSize; /* initial block size */
  9. Size maxBlockSize; /* maximum block size */
  10. Size nextBlockSize; /* next block size to allocate */
  11. Size allocChunkLimit; /* effective chunk size limit */
  12. AllocBlock keeper; /* if not NULL, keep this block over resets */
  13. } AllocSetContext;
  14.  
  15. typedef AllocSetContext *AllocSet;

我们来看看这个set内容,里面包含了数据库初始化的数据块大小,最大块,下一个块以及chunk的limit.

  1. TopMemoryContext = AllocSetContextCreate((MemoryContext) NULL,
  2. "TopMemoryContext",
  3. ,
  4. * ,
  5. * );

在初始化"TopMemoryContext"的时候,默认以及设定了最小块为0,初始化为8*1024,最大为8*1024

  1. (gdb) p *set
  2. $ = {header = {type = T_AllocSetContext, isReset = '\001', allowInCritSection = '\000', methods = 0xd1bbe0 <AllocSetMethods>, parent = 0x0, firstchild = 0x0, nextchild = 0x0,
  3. name = 0x14f9c70 "TopMemoryContext", reset_cbs = 0x0}, blocks = 0x0, freelist = {0x0 <repeats times>}, initBlockSize = , maxBlockSize = , nextBlockSize = , allocChunkLimit = , keeper = 0x0}

可以到初始化的数据块为8K,最大8K,allocChunkLimit为1024.

这样就把这个"TopMemoryContext"初始化完成了,然后把该内存上下文赋值给CurrentMemoryContext。

  1. (gdb) p CurrentMemoryContext
  2. $ = (MemoryContext) 0x14f9bb0
  3. (gdb) p TopMemoryContext
  4. $ = (MemoryContext) 0x14f9bb0
  5. (gdb)

现在初始化"TopMemoryContext"的第一个孩子,"ErrorContext"。

处理方式跟上面的区别,就是parent是"TopMemoryContext",并且内存上下文是通过MemoryContext->methods->AllocSetAlloc这个函数指针来分配内存的

这个后面要单独分析。

  1. (gdb) p *ErrorContext
  2. $ = {type = T_AllocSetContext, isReset = '\001', allowInCritSection = '\000', methods = 0xd1bbe0 <AllocSetMethods>, parent = 0x14f9bb0, firstchild = 0x0, nextchild = 0x0, nam
  3. e = 0x14f9d80 "ErrorContext", reset_cbs = 0x0}
  4. (gdb) p *ErrorContext->parent
  5. $ = {type = T_AllocSetContext, isReset = '\000', allowInCritSection = '\000', methods = 0xd1bbe0 <AllocSetMethods>, parent = 0x0, firstchild = 0x14f9cc0, nextchild = 0x0, nam
  6. e = 0x14f9c70 "TopMemoryContext", reset_cbs = 0x0}
  7. (gdb) p *ErrorContext->parent->firstchild
  8. $ = {type = T_AllocSetContext, isReset = '\001', allowInCritSection = '\000', methods = 0xd1bbe0 <AllocSetMethods>, parent = 0x14f9bb0, firstchild = 0x0, nextchild = 0x0, nam
  9. e = 0x14f9d80 "ErrorContext", reset_cbs = 0x0}
  10. (gdb)

这样就把ErrorContext初始化完成了。PG中所有的内存上下文都挂载"TopMemoryContext"下面。

后面就是大量的环境变量设置了,以及root校验。

最后调用函数"PostmasterMain(argc,argv)"。这就是我们的大管家,后面再写这部分。

PostgreSQL启动main函数都干了什么(一)的更多相关文章

  1. swoft| 源码解读系列二: 启动阶段, swoft 都干了些啥?

    date: 2018-8-01 14:22:17title: swoft| 源码解读系列二: 启动阶段, swoft 都干了些啥?description: 阅读 sowft 框架源码, 了解 sowf ...

  2. loadView在App启动时到底都干了些什么?

    loadView在App启动时到底都干了些什么? 查阅苹果官方文档如下: 1. 当你访问一个ViewController的view属性时,如果此时view的值是nil,那么,ViewControlle ...

  3. iOS应用启动main函数

    #import <UIKit/UIKit.h> #import "AppDelegate.h" int main(int argc, char * argv[]) { ...

  4. C++ main()函数及其参数

    1.首先,想想C/C++在main函数之前和之后会做些什么? 我们看看底层的汇编代码: __start: : init stack; init heap; open stdin; open stdou ...

  5. WPF点滴(1) Main 函数

    应用程序的入口函数是main函数,在Console程序和Winform程序main函数都有清晰的定义,可以很容易找到,但是WPF的工程文件中却找不到main函数的定义,是WPF不需要main函数吗?N ...

  6. C语言中main函数的参数

    转自:http://blog.csdn.net/cnctloveyu/article/details/3905720 我们经常用的main函数都是不带参数的.因此main 后的括号都是空括号.实际上, ...

  7. C++向main函数传递参数的方法(实例已上传至github)

    通常情况下,我们定义的main函数都只有空形参列表: int main(){...} 然而,有时我们确实需要给mian传递实参,一种常见的情况是用户设置一组选项来确定函数所要执行的操作.例如,假定ma ...

  8. Linux C编程--main函数参数解析

    我们经常用的main函数都是不带参数的.因此main 后的括号都是空括号.实际上,main函数可以带参数,这个参数可以认为是 main函数的形式参数.C语言规定main函数的参数只能有两个, 习惯上这 ...

  9. c/c++中main函数参数讲解

    参考地址: http://blog.csdn.net/cnctloveyu/article/details/3905720 我们经常用的main函数都是不带参数的.因此main 后的括号都是空括号.实 ...

随机推荐

  1. jvm - 垃圾回收

    jvm - 垃圾回收 注意 : 本系列文章为学习系列,部分内容会取自相关书籍或者网络资源,在文章中间和末尾处会有标注 垃圾回收的意义 它使得java程序员不再时时刻刻的关注内存管理方面的工作. 垃圾回 ...

  2. JAVA提高十一:LinkedList深入分析

    上一节,我们学习了ArrayList 类,本节我们来学习一下LinkedList,LinkedList相对ArrayList而言其使用频率并不是很高,因为其访问元素的性能相对于ArrayList而言比 ...

  3. Leetcode题解(十四)

    39.Combination Sum 题目 题目要求找出和为target的数字组合,并且每个整数可以多次使用.仔细思考可以发现,这道题目可以采用递归的方法来完成,比如举的例子,target=7,一开始 ...

  4. phpmailer 发送邮件(一)

    phpmailer下载地址: PHPMailer : https://github.com/PHPMailer/PHPMailer 一.基本要求 Web访问正常(apache可以正常访问) php 设 ...

  5. linux-mkdir

    mkdir mkdir : 可以用来创建目录,如果不加创建路径即在本路径下创建一个新的指定的目录,否则即在给出的路径下创建目录. 目录创建:目录名尽量见名知意,根据不同需要分层创建,尽量避免在同一目录 ...

  6. selenium页面元素操作(简易版)

    介绍一下,这是处理页面元素的基本方法,@selenium 发送文字    element.send_keys(keys_to_send) 单击    element.click() 提交表单   el ...

  7. smm框架整合实现登录功能

    一.准备所需的jar包 1.1所需jar包 1.Spring框架jar包 2.Mybatis框架jar包 3.Spring的AOP事务jar包 4.Mybatis整合Spring中间件jar包 5.a ...

  8. 【Kafka源码】KafkaConsumer

    [TOC] KafkaConsumer是从kafka集群消费消息的客户端.这是kafka的高级消费者,而SimpleConsumer是kafka的低级消费者.何为高级?何为低级? 我们所谓的高级,就是 ...

  9. EF错误

    The model backing the 'XXXXDBContext' context has changed since the database was created. Either man ...

  10. [转]Oracle High Water Level高水位分析

    PLSQL_性能优化系列14_Oracle High Water Level高水位分析 http://www.cnblogs.com/eastsea/p/4005814.html 一.摘要 PLSQL ...