`

PostgreSQL启动过程中的那些事七:初始化共享内存和信号七:shmem中初始化bufferpool

阅读更多

       pg 初始化完 shmem ,给其加上索引 "ShmemIndex" 后,接着就在 shmem 里初始化管理各种事务和事务本身相关结构的实例。然后就是初始化缓冲池( buffer pool )。

       缓冲区 (buffers) 存在于一个空闲内存块列表和一个哈希表查询数据结构。下面简述一下和缓冲池相关的概念。

       查找缓冲区( buffer )时必须注意,在 I/O 开始之前缓冲区必须可用。负责尝试读缓冲区的第二个进程会分配自己的复制,这样缓冲池就不一致了。

       缓冲区同步, IO_IN_PROGRESS ——这是缓冲区描述符里的一个标签。当一个 IO 被启动和在 IO 结束被清除时必须设置该标签。这样是为了保证在另一个进程使用该缓冲区的时候其他进程不能开始使用该缓冲区。

    缓冲区缓冲块的引用计数——计进程在缓冲区上持有pin 的数目。缓冲区在IO 期间,BufferAlloc() 之后立即被pin 住。事务结束之前pin 被释放。

    私有引用计数——每一个缓冲区有一个私有的引用计数,保持当前进程里pin 住该缓冲区的次数跟踪。用这个有两个目的:第一,如果pin 住一个缓冲区多于一次,pg 仅需要改变共享的引用计数一次,这样只锁共享状态一次;第二,当事务退出时,它应该仅unpin 缓冲区正好自己pin 住该缓冲区的次数,这样该事务就可以不破坏了另一个后台进程/backend 的缓冲区。

 

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

 


初始化缓冲池 /buffer pool 方法调用流程图

 

2 初始化 xlog 相关结构

话说 main()->…->PostmasterMain()->…->reset_shared() -> CreateSharedMemoryAndSemaphores()->…-> InitBufferPool () ,初始化 缓冲池及 相关数据结构 BufferDesc 等,然后又初始化了一个可扩展哈希表 "shared buffer lookup table" ,用作内存里管理缓冲池。

InitBufferPool () ->ShmemInitStruct() 在其中 调用 hash_search() 在哈希表索引 "ShmemIndex" 中查找 " Buffer Descriptors " ,如果没有,就在 shmemIndex 中给 " Buffer Descriptors " 分一个 HashElement ShmemIndexEnt entry ,在其中的 Entry 中写上 " Buffer Descriptors " 。返回 ShmemInitStruct() ,再调用 ShmemAlloc() 在共享内存上给 " Buffer Descriptors " 相关结构(是该结构的数组,数组数目根据shared_buffer 计算,有数万甚至数十万以上,具体见下面“ BufferPool 相关结构和策略控制结构图 )分配空间,设置 entry (在这儿及ShmemIndexEnt 类型变量)的成员 location 指向该空间, size 成员记录该空间大小 最后返回 InitBufferPool () ,让 BufferDesc * 类型 全局变量 BufferDescriptors 指向BufferDesc 类型实例的起始地址就是在shmem 里给 " Buffer Descriptors " 相关结构分配的内存起始地址,设置其中BufferDesc 结构类型的成员值。

接着 InitBufferPool () ->ShmemInitStruct() 在其中 调用 hash_search() 在哈希表索引 "ShmemIndex" 中查找 " Buffer Blocks " ,如果没有,就在 shmemIndex 中给 " Buffer Blocks " 分一个 HashElement ShmemIndexEnt entry ,在其中的 Entry 中写上 " Buffer Blocks " 。返回 ShmemInitStruct() ,再调用 ShmemAlloc() 在共享内存上给 " Buffer Blocks " 分配空间,设置 entry (在这儿及ShmemIndexEnt 类型变量)的成员 location 指向该空间, size 成员记录该空间大小 。每个缓冲区块大小默认为8k (可以根据设置变),共shared_buffer/BLCKSZ 个,一般会有数万个甚至数十万个以上,一个缓冲区块一个BufferDescriptors 最后返回 InitBufferPool () ,让 char * 类型 全局变量 BufferBlocks 指向 该内存段 的起始地址。

然后 StrategyInitialize() ->InitBufferTable () -> ShmemInitHash() 在其中创建一个用于管理和查找缓冲区块的可扩展哈希表 "shared buffer lookup table" (图在下面)。返回 InitBufferPool () ,让 HTAB * 类型 全局静态变量 SharedBufHash 指向 "shared buffer lookup table"

最后 StrategyInitialize() -> ShmemInitStruct() 在其中 调用 hash_search() 在哈希表索引 "ShmemIndex" 中查找 "Buffer Strategy Status" ,如果没有,就在 shmemIndex 中给 "Buffer Strategy Status" 分一个 HashElement ShmemIndexEnt entry ,在其中的 Entry 中写上 "Buffer Strategy Status" 。返回 ShmemInitStruct() ,再调用 ShmemAlloc() 在共享内存上给 "Buffer Strategy Status" 相关结构(见下面“ BufferPool 相关结构和策略控制结构图 )分配空间,设置 entry (在这儿及ShmemIndexEnt 类型变量)的成员 location 指向该空间, size 成员记录该空间大小 最后返回 StrategyInitialize() ,让 BufferStrategyControl * 类型 全局静态变量 StrategyControl 指向BufferStrategyControl 类型实例的起始地址就是在shmem 里给 "Buffer Strategy Status" 相关结构分配的内存起始地址,设置其中BufferStrategyControl 结构类型的成员值。

相关变量、结构定义和 初始化完成后数据结构图在下面。

 

typedef struct sbufdesc

{

       BufferTag      tag;                     /* ID of page contained in buffer */

       BufFlags       flags;                   /* see bit definitions above */

       uint16           usage_count;     /* usage counter for clock sweep code */

       unsigned     refcount;             /* # of backends holding pins on buffer */

       int                 wait_backend_pid;        /* backend PID of pin-count waiter */

 

       slock_t          buf_hdr_lock;     /* protects the above fields */

 

       int                 buf_id;                /* buffer's index number (from 0) */

       int                 freeNext;             /* link in freelist chain */

 

       LWLockId     io_in_progress_lock;  /* to wait for I/O to complete */

       LWLockId     content_lock;     /* to lock access to buffer contents */

} BufferDesc;

 

/*

  * The shared freelist control information.

  */

typedef struct

{

    /* Clock sweep hand: index of next buffer to consider grabbing */

    int          nextVictimBuffer;

 

    int          firstFreeBuffer;    /* Head of list of unused buffers */

    int          lastFreeBuffer; /* Tail of list of unused buffers */

 

    /*

      * NOTE: lastFreeBuffer is undefined when firstFreeBuffer is -1 (that is,

      * when the list is empty)

      */

 

    /*

      * Statistics.  These counters should be wide enough that they can't

      * overflow during a single bgwriter cycle.

      */

    uint32      completePasses;     /* Complete cycles of the clock sweep */

    uint32      numBufferAllocs;    /* Buffers allocated since last reset */    

} BufferStrategyControl;

 

/* Pointers to shared state */

static BufferStrategyControl *StrategyControl = NULL;

 

static HTAB *SharedBufHash;

 

typedef struct buftag

{

    RelFileNode rnode;          /* physical relation identifier */

    ForkNumber forkNum;

    BlockNumber blockNum;       /* blknum relative to begin of reln */

} BufferTag;

 

typedef struct

{

    BufferTag   key;            /* Tag of a disk page */

    int          id;             /* Associated buffer ID */

} BufferLookupEnt;

 


初始化完 BufferPool 相关结构 的内存结构图

 

 

 

BufferPool 哈希表索引“ Shared Buffer Lookup Table

 

 

 

BufferPool 相关结构和策略控制结构图

       上图中左面是 BufferPool 和策略控制结构图,根据 shared_buffer 的设置,有 25600 BufferDesc 数组和 25600 buffer block 数组,还有一个 BufferStrategyControl 结构。右面是 25600 BufferDesc 数组中每个 BufferDesc 的成员的初始化值。已经体现了一个 buffer block 两个轻量锁以及和磁盘上的数据文件的对应情况。

  • 大小: 80.7 KB
  • 大小: 88.9 KB
  • 大小: 154.2 KB
  • 大小: 303.4 KB
0
0
分享到:
评论

相关推荐

    nacos-2.0.1 postgresql初始化脚本

    nacos-2.0.1 postgresql初始化脚本

    quartz-2.2.3版本的quartz初始化sql语句

    在Quartz 2.2.3版本中,初始化数据库是使用Quartz的关键步骤,因为Quartz依赖于一个持久化存储来保存作业和触发器的信息。这个过程通常涉及执行一系列SQL语句来创建必要的表结构。 Quartz的初始化SQL语句主要用于...

    8基础 5:初始化 MySQL 数据库并建立连接(3).md

    在本节内容中,我们将深入了解如何使用Go语言和GORM库初始化MySQL数据库,并建立与该数据库的连接。GORM是一个流行的Go语言ORM(对象关系映射)库,它允许开发者通过编程方式与数据库交互,而无需编写大量的SQL代码...

    Postgresql存储过程

    Postgresql存储过程详解 Postgresql存储过程是指在Postgresql数据库中定义的一组SQL语句的...Postgresql存储过程是Postgresql数据库中的一种强大工具,可以实现复杂的操作,提高数据库服务器的性能和应用程序的性能。

    关于PostGreSQL中的存储过程

    其中,存储过程是一个非常重要的概念,本文将对 PostgreSQL 中的存储过程进行详细的介绍和解释。 什么是存储过程 存储过程是一组为了完成特定任务而编写的 SQL 语句集合。它可以将复杂的操作封装起来,以便于重复...

    PostgreSQL中文手册9.2

    一、服务器进程的启动和关闭: 一、服务器进程的启动和关闭: 一、服务器进程的启动和关闭: 一、服务器进程的启动和关闭: 一、服务器进程的启动和关闭: 一、服务器进程的启动和关闭: . 50 PostgreSQL PostgreSQL...

    MySQL和PostgreSQL的比较

    MySQL的数据缓冲区配置主要通过`innodb_buffer_pool_size`参数进行,它决定了InnoDB引擎用于缓存表数据和索引的内存缓冲区大小,在专用服务器上,这一参数可设置至物理内存的80%。而在PostgreSQL中,数据缓冲区管理...

    linux配置postgresql

    在 Linux 系统中,可以通过以下命令启动 PostgreSQL: ``` # su – postgres pg$ /usr/local/pgsql/bin/postmaster ``` 这将启动 PostgreSQL 数据库服务器,并允许用户访问数据库。 配置 PostgreSQL 数据库需要多个...

    postgresql-42.3.1-API文档-中文版.zip

    赠送jar包:postgresql-42.3.1.jar; 赠送原API文档:postgresql-42.3.1-javadoc.jar; 赠送源代码:postgresql-42.3.1-sources.jar;...人性化翻译,文档中的代码和结构保持不变,注释和说明精准翻译,请放心使用。

    在windows下手动初始化PostgreSQL数据库教程

    在初始化过程中,initdb会生成一系列的配置文件和数据库模板,并设置默认的最大连接数、共享缓冲区等参数。完成初始化之后,会得到一系列的成功消息,表明数据库系统已经准备就绪。 最后,使用pg_ctl工具来启动...

    windows环境下新版12.2postgreSQL的安装+初始化配置+启动

    总之,安装和配置Windows上的PostgreSQL 12.2涉及到下载安装文件、初始化数据库、设置服务启动和配置连接参数等步骤。遵循这些步骤,你就可以顺利地在Windows环境中搭建起一个功能齐全的PostgreSQL数据库系统了。 ...

    Linux下PostgreSQL安装与开机启动

    初始化数据库是设置PostgreSQL的重要步骤之一,它创建数据库集群。 ##### 操作命令: ```bash su postgres /usr/local/pgsql/bin/initdb -D /usr/local/pgsql/data ``` **解释:** - `su postgres`:切换到`...

    postgresql-42.2.5-API文档-中英对照版.zip

    赠送jar包:postgresql-42.2.5.jar; 赠送原API文档:postgresql-42.2.5-javadoc.jar;...人性化翻译,文档中的代码和结构保持不变,注释和说明精准翻译,请放心使用。 双语对照,边学技术、边学英语。

    PostgreSQL修炼之道 从小工到专家.pptx

    * PostgreSQL配置:包括设置数据库参数、内存管理和日志记录等方面。 数据类型与表达式 * 数据类型:包括整数、字符串、日期时间、布尔值等基本数据类型。 * 表达式:包括算术运算符、比较运算符、逻辑运算符和...

    postgresql8.2.3 中文文档

    1. **安装与配置**:这部分内容会指导用户如何在不同操作系统上安装 PostgreSQL,包括设置数据目录、初始化数据库集群、配置服务器参数以及启动和停止服务。 2. **SQL语言**:文档详细介绍了SQL的使用,包括数据...

    Ubuntu 下源码安装Postgresql

    七、初始化数据库目录 使用 initdb 命令初始化数据库目录: ./initdb -D PGDATA/postgres 八、启动数据库 使用以下命令启动数据库: ./postgres -D PGDATA/postgres 九、配置开机启动 为了使数据库在开机时...

    Postgresql-10安装包

    10. **阅读官方文档**:PostgreSQL 提供详细的官方文档,包括安装、配置和使用指南,强烈建议在安装和使用过程中参考。 **四、安装脚本的使用** 如果你从博主那里获取了安装脚本,通常这是一个包含上述步骤的自动...

    PostGreSQL安装部署系列:Centos 7.9 安装指定PostGreSQL-15版本数据库

    初始化数据库是PostgreSQL安装过程中必不可少的一步,可以通过执行以下命令完成: ```bash sudo /usr/pgsql-15/bin/postgresql-15-setup initdb ``` 初始化完成后,将会创建数据库目录并设置默认权限。 ##### 3.4...

    框架使用到的初始化脚本

    在IT行业中,数据库初始化脚本是项目启动前的关键步骤,特别是在使用像Spring MVC这样的MVC框架构建应用时。本文将详细解析标题“框架使用到的初始化脚本”以及描述中涉及的知识点,主要关注与MySQL数据库相关的初始...

    新建SQL数据库并初始化

    这些文件对于理解数据库的创建和初始化过程具有重要价值。 总之,新建和初始化SQL数据库是一项关键的任务,涉及数据库管理系统的选择、安装、设计表结构、设置权限等多个环节。掌握这些技能对于任何IT专业人员来说...

Global site tag (gtag.js) - Google Analytics