热门标签 | HotTags
当前位置:  开发笔记 > 数据库 > 正文

PostgreSQL启动过程之加载GUC参数

1先上个图,看一下函数调用过程梗概,中间细节有略GUC参数初始化分两步,第一步先读取buildin/compiled-in的GUC参数默认值,这里包括全部的GUC参数,建立GUC参数相关结构变量,第二步读取postgresql.conf配置文件中的参数设置之。从上图中能看出来,这个读

1先上个图,看一下函数调用过程梗概,中间细节有略

 

      GUC参数初始化分两步,第一步先读取buildin/compiled-in的GUC参数默认值,这里包括全部的GUC参数,建立GUC参数相关结构变量,第二步读取postgresql.conf配置文件中的参数设置之。从上图中能看出来,这个读取并设置postgresql.conf中参数的过程还是挺复杂的。

2初始化GUC相关数据结构并取hardcode/buildin的参数值。

       pg里的GUC参数按设置的值分五种类型,分别是bool、int、real、string、enum,根据这五种类型,定义了五种结构类型,再根据这五种结构,每个类型建一个对应的静态数组,用于存储这些相应类型的GUC参数。这五种类型是config_bool、config_int、config_real、config_string、config_enum,对应的静态数组是ConfigureNamesBool、ConfigureNamesInt、ConfigureNamesReal、ConfigureNamesString、ConfigureNamesEnum。具体结构和数组定义见下面。

 

五个结构定义:

struct config_bool

{

    struct config_generic gen;

    /* these fields must be set correctly in initial value: */

    /* (all but reset_val are constants) */

    bool       *variable;

    bool        boot_val;

    GucIntCheckHookcheck_hook;

    GucBoolAssignHookassign_hook;

    GucShowHookshow_hook;

    /* variable fields, initialized at runtime: */

    bool        reset_val;

    void        * reset_extra

};

 

struct config_int

{

    struct config_generic gen;

/*constant fields, must be set correctly in initial value: */

    int        *variable;

    int         boot_val;

    int         min;

    int         max;

    GucIntCheckHookcheck_hook;

    GucIntAssignHookassign_hook;

    GucShowHookshow_hook;

    /* variable fields, initialized at runtime: */

    int         reset_val;

    void        * reset_extra

};

 

struct config_real

{

    struct config_generic gen;

/* constantfields, must be set correctly in initial value: */

    double     *variable;

    double      boot_val;

    double      min;

    double      max;

    GucIntCheckHookcheck_hook;

    GucRealAssignHookassign_hook;

    GucShowHookshow_hook;

    /* variable fields, initialized at runtime: */

    double      reset_val;

    void        * reset_extra

};

 

struct config_string

{

    struct config_generic gen;

/* constant fields, must beset correctly in initial value: */

    char      **variable;

    const char *boot_val;

    GucIntCheckHookcheck_hook;

    GucStringAssignHookassign_hook;

    GucShowHookshow_hook;

    /* variable fields, initialized at runtime: */

    char       *reset_val;

void        * reset_extra

};

struct config_enum

{

    struct config_generic gen;

/* constant fields, must beset correctly in initial value: */

    int   *variable;

    int     boot_val;

    GucIntCheckHookcheck_hook;

    GucStringAssignHookassign_hook;

    GucShowHookshow_hook;

    /* variable fields, initialized at runtime: */

    int    reset_val;

void        * reset_extra

};

 

和结构类型对应的五个静态数组:

static struct config_boolConfigureNamesBool[] =

{

       {

{"enable_seqscan",PGC_USERSET, QUERY_TUNING_METHOD,

gettext_noop("Enablesthe planner's use of sequential-scan plans."),

NULL

},

&enable_seqscan,

true,

NULL,NULL, NULL

       },

……

       /*End-of-list marker */

       {

{NULL,0, 0, NULL, NULL}, NULL, false, NULL, NULL, NULL

       }

};

 

static struct config_int ConfigureNamesInt[]=

{

       {

{"archive_timeout",PGC_SIGHUP, WAL_ARCHIVING,

gettext_noop("Forcesa switch to the next xlog file if a "

 "new file has not been started within Nseconds."),

NULL,

GUC_UNIT_S

},

&XLogArchiveTimeout,

0,0, INT_MAX,

NULL,NULL, NULL

       },

……

       /*End-of-list marker */

       {

{NULL,0, 0, NULL, NULL}, NULL, 0, 0, 0, NULL, NULL, NULL

       }

};

 

 

static struct config_realConfigureNamesReal[] =

{

       {

{"seq_page_cost",PGC_USERSET, QUERY_TUNING_COST,

gettext_noop("Setsthe planner's estimate of the cost of a "

 "sequentially fetched disk page."),

NULL

},

&seq_page_cost,

DEFAULT_SEQ_PAGE_COST,0, DBL_MAX,

NULL,NULL, NULL

       },

       /*End-of-list marker */

       {

{NULL,0, 0, NULL, NULL}, NULL, 0.0, 0.0, 0.0, NULL, NULL, NULL

       }

};

 

 

static struct config_stringConfigureNamesString[] =

{

       {

{"archive_command",PGC_SIGHUP, WAL_ARCHIVING,

gettext_noop("Setsthe shell command that will be called to archive a WAL file."),

NULL

},

&XLogArchiveCommand,

"",

NULL,NULL, show_archive_command

       },

……

       /*End-of-list marker */

       {

{NULL,0, 0, NULL, NULL}, NULL, NULL, NULL, NULL, NULL

       }

};

 

 

static struct config_enumConfigureNamesEnum[] =

{

       {

{"backslash_quote",PGC_USERSET, COMPAT_OPTIONS_PREVIOUS,

gettext_noop("Setswhether \"\\'\" is allowed in string literals."),

NULL

},

&backslash_quote,

BACKSLASH_QUOTE_SAFE_ENCODING,backslash_quote_options,

NULL,NULL, NULL

       },

……

       /*End-of-list marker */

       {

{NULL,0, 0, NULL, NULL}, NULL, 0, NULL, NULL, NULL, NULL

       }

};

 

上面五个结构定义中,每个结构的第一个成员变量都是一个config_generic结构的gen成员,下面是config_generic的结构定义:

struct config_generic

{

       /* constantfields, must be set correctly in initial value: */

       const char *name;                 /* name of variable - MUST BE FIRST */

       GucContext       context;              /* context required to set the variable */

       enumconfig_group group;   /* to help organize variables by function */

       const char *short_desc;         /* short desc. of this variable's purpose */

       const char *long_desc;          /* long desc. of this variable's purpose */

       int                 flags;                   /* flag bits, see below*/

       /* variablefields, initialized at runtime: */

       enumconfig_type vartype;  /* type of variable (set only at startup) */

       int                 status;                 /* status bits, see below*/

       GucSource  reset_source;      /* source of thereset_value */

       GucSource  source;                /*source of the current actual value */

       GucStack  *stack;               /* stacked outside-of-transaction states */

       void         *extra;                   /*"extra" pointer for current actual value */

       char         *sourcefile;           /* filecurrent setting is from (NULL if not

 * file) */

       int                 sourceline;          /* line in source file */

};

 

然后,定义一个config_generic **类型的静态变量数组guc_variables,再计算参数总数,所有参数以config_generic*类型计算所需内存空间,冗余25%内存后malloc分配内存空间。把guc_variables每一个元素指向ConfigureNamesBool、ConfigureNamesInt、ConfigureNamesReal、ConfigureNamesString、ConfigureNamesEnum这五个数组的config_generic类型成员gen的地址,然后按照参数名称把所有元素做了快速排序。这个过程中还设置了一些GUC参数的默认值。

static struct config_generic **guc_variables;

    后面查询GUC参数都是在guc_variables这个已排序的数组里找。这样GUC参数的数据结构就搭建完成了,下面看看GUC参数相关的数据结构图吧。

    先把涉及到的结构的图分别列出,再画个这些结构的组织关系示意图。







3加载postgresql.conf参数配置文件里的参数设置

       从main->PostmasterMain->SelectConfigFiles->ProcessConfigFile开始处理参数配置文件postgresql.conf,读取postgresql.conf配置文件的调用过程是ProcessConfigFile -> ParseConfigFile -> AllocateFile->fopen,最后用fopen打开文件,其中ProcessConfigFile、ParseConfigFile在文件src\backend\utils\misc\guc-file.l中,AllocateFile在文件src\backend\storage\file\fd.c中。

    pg使用 flex 去处理 conf 文件。在ParseConfigFile中把配置文件中的配置项组织成一个链表,调用set_config_option检查这些值是否有效,若可以设置就调用set_config_option设置这些值。

 

这里以"max_connections"做例子,从配置文件读取"max_connections",然后从guc_variables数组中找元素" max_connections ",比较参数结构的GucContext (枚举类参数能被设置的时机。定义见下面)枚举类型成员context和当前时间,看是否可以此刻修改。接着比较参数结构的GucSource(枚举了当前GUC参数设置的来源。除非参数新值的来源等级不小于原参数的来源等级时,新设置才能生效。例如,修改配置文件不能覆盖postmaster command line的设置。定义见下面)枚举类型成员source和新参数值的来源,看是否可以修改。如果可以,把config_generic结构类型的元素" max_connections "类型转换为config_int类型,修改variable成员为新值,修改该参数的来源source为当前来源PGC_S_FILE,如果元素" max_connections "的reset_source <= source,修改reset_val成员为新值,修改该参数的reset_source为当前来源PGC_S_FILE。

 

typedef enum

{

    PGC_INTERNAL,

    PGC_POSTMASTER,

    PGC_SIGHUP,

    PGC_BACKEND,

    PGC_SUSET,

    PGC_USERSET

} GucContext;

 

typedef enum

{

    PGC_S_DEFAULT,              /*wired-in default */

    PGC_S_ENV_VAR,              /*postmaster environment variable */

    PGC_S_FILE,                 /*postgresql.conf */

    PGC_S_ARGV,                 /*postmaster command line */

    PGC_S_DATABASE,             /*per-database setting */

    PGC_S_USER,                 /*per-user setting */

    PGC_S_CLIENT,               /* fromclient connection request */

    PGC_S_OVERRIDE,             /*special case to forcibly set default */

    PGC_S_INTERACTIVE,          /* dividingline for error reporting */

    PGC_S_TEST,                 /*test per-database or per-user setting */

    PGC_S_SESSION               /* SETcommand */

} GucSource


推荐阅读
  • PHP 5.2.5 安装与配置指南
    本文详细介绍了 PHP 5.2.5 的安装和配置步骤,帮助开发者解决常见的环境配置问题,特别是上传图片时遇到的错误。通过本教程,您可以顺利搭建并优化 PHP 运行环境。 ... [详细]
  • 构建基于BERT的中文NL2SQL模型:一个简明的基准
    本文探讨了将自然语言转换为SQL语句(NL2SQL)的任务,这是人工智能领域中一项非常实用的研究方向。文章介绍了笔者在公司举办的首届中文NL2SQL挑战赛中的实践,该比赛提供了金融和通用领域的表格数据,并标注了对应的自然语言与SQL语句对,旨在训练准确的NL2SQL模型。 ... [详细]
  • Hadoop入门与核心组件详解
    本文详细介绍了Hadoop的基础知识及其核心组件,包括HDFS、MapReduce和YARN。通过本文,读者可以全面了解Hadoop的生态系统及应用场景。 ... [详细]
  • DNN Community 和 Professional 版本的主要差异
    本文详细解析了 DotNetNuke (DNN) 的两种主要版本:Community 和 Professional。通过对比两者的功能和附加组件,帮助用户选择最适合其需求的版本。 ... [详细]
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
  • 本文详细分析了Hive在启动过程中遇到的权限拒绝错误,并提供了多种解决方案,包括调整文件权限、用户组设置以及环境变量配置等。 ... [详细]
  • 本文详细介绍了如何使用ActionScript 3.0 (AS3) 连接并操作MySQL数据库。通过具体的代码示例和步骤说明,帮助开发者理解并实现这一过程。 ... [详细]
  • 在即将迎来26岁生日之际,作者的人生陷入了低谷。经过近三年的硕士学习后,最终决定退学,并且面临没有工作经验的困境。尽管如此,作者依然坚定地选择为自己的人生负责。 ... [详细]
  • 使用Python在SAE上开发新浪微博应用的初步探索
    最近重新审视了新浪云平台(SAE)提供的服务,发现其已支持Python开发。本文将详细介绍如何利用Django框架构建一个简单的新浪微博应用,并分享开发过程中的关键步骤。 ... [详细]
  • 本文详细介绍了美国最具影响力的十大财团,包括洛克菲勒、摩根、花旗银行等。这些财团在历史发展过程中逐渐形成,并对美国的经济、政治和社会产生深远影响。 ... [详细]
  • 基于KVM的SRIOV直通配置及性能测试
    SRIOV介绍、VF直通配置,以及包转发率性能测试小慢哥的原创文章,欢迎转载目录?1.SRIOV介绍?2.环境说明?3.开启SRIOV?4.生成VF?5.VF ... [详细]
  • 深入探讨CPU虚拟化与KVM内存管理
    本文详细介绍了现代服务器架构中的CPU虚拟化技术,包括SMP、NUMA和MPP三种多处理器结构,并深入探讨了KVM的内存虚拟化机制。通过对比不同架构的特点和应用场景,帮助读者理解如何选择最适合的架构以优化性能。 ... [详细]
  • 解决JAX-WS动态客户端工厂弃用问题并迁移到XFire
    在处理Java项目中的JAR包冲突时,我们遇到了JaxWsDynamicClientFactory被弃用的问题,并成功将其迁移到org.codehaus.xfire.client。本文详细介绍了这一过程及解决方案。 ... [详细]
  • 本题通过将每个矩形视为一个节点,根据其相对位置构建拓扑图,并利用深度优先搜索(DFS)或状态压缩动态规划(DP)求解最小涂色次数。本文详细解析了该问题的建模思路与算法实现。 ... [详细]
  • 本题探讨如何通过最大流算法解决农场排水系统的设计问题。题目要求计算从水源点到汇合点的最大水流速率,使用经典的EK(Edmonds-Karp)和Dinic算法进行求解。 ... [详细]
author-avatar
mobiledu2502875993
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有