`

PostgreSQL启动过程中的那些事七:初始化共享内存和信号八:shmem中初始化常规锁管理器

阅读更多

 

       这一节 pg 初始化锁管理器,通过 InitLocks 例程实现,主要是创建了三个哈希表 第一个哈希表"LOCK hash" 用于管理锁,第二个哈希表"PROCLOCK hash" 用于管理进程锁,第三个"LOCALLOCK hash" 用于管理本地锁信息。其中第一个和第二个哈希表都是共享哈希表,第三个是非关系哈希表。初始化第三个哈希表"LOCALLOCK hash" 时在共享内存哈希表索引" ShmemIndex " 里没有创建索引,因为这个哈希表不在共享内存里,而是在MemoryContext "LOCALLOCK hash" 里分配的内存。

pg 中的锁有三种类型:自旋 锁(spinlock )、轻量锁(LWLock )、常规锁(Lock ), 作为一个主题另行讨论。

 

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

 


初始化 Lockmgr 方法调用流程图

 

2 初始化 xlog 相关结构

话说 main()-> ->PostmasterMain()-> ->reset_shared() -> CreateSharedMemoryAndSemaphores()> ->InitLocks() ,在 shmem 里分配了三个哈希表 第一个哈希表"LOCK hash" 用于管理锁,第二个哈希表"PROCLOCK hash" 用于管理进程锁,第三个"LOCALLOCK hash" 用于管理本地锁信息。其中第一个和第二个哈希表都是共享哈希表,第三个是非关系哈希表。初始化第三个哈希表"LOCALLOCK hash" 时在共享内存哈希表索引" ShmemIndex " 里没有创建索引,因为这个哈希表不在共享内存里,而是在MemoryContext "LOCALLOCK hash" 里分配的内存。

InitLocks()->ShmemInitHash ()->ShmemInitStruct() 在其中 调用 hash_search() 在哈希表索引 "ShmemIndex" 中查找 "LOCK hash" ,如果没有,就在 shmemIndex 中给 "LOCK hash" 分一个 HashElement ShmemIndexEnt entry ,在其中的 Entry 中写上 "LOCK hash" 。返回 ShmemInitStruct() ,再调用 ShmemAlloc() 在共享内存上给 "LOCK hash" 相关结构(见下面 "LOCK hash" 相关结构图 )分配空间,设置 entry (在这儿即ShmemIndexEnt 类型变量)的成员 location 指向该空间, size 成员记录该空间大小,然后返回 ShmemInitHash() ,调用 hash_create() ,创建哈希表 "LOCK hash" ,最后返回 ShmemInitHash() ,让 HTAB * 类型静态 全局变量 LockMethodLockHash 指向 哈希表 "LOCK hash"

接着 InitLocks()->ShmemInitHash ()->ShmemInitStruct() 在其中 调用 hash_search() 在哈希表索引 "ShmemIndex" 中查找 "PROCLOCK hash" ,如果没有,就在 shmemIndex 中给 "PROCLOCK hash" 分一个 HashElement ShmemIndexEnt entry ,在其中的 Entry 中写上 "PROCLOCK hash" 。返回 ShmemInitStruct() ,再调用 ShmemAlloc() 在共享内存上给 "PROCLOCK hash" 相关结构(见下面 "PROCLOCK hash" 相关结构图 )分配空间,设置 entry (在这儿即ShmemIndexEnt 类型变量)的成员 location 指向该空间, size 成员记录该空间大小,然后返回 ShmemInitHash() ,调用 hash_create() ,创建哈希表 "PROCLOCK hash" ,最后返回 ShmemInitHash() ,让 HTAB * 类型静态 全局变量 LockMethodProcLockHash 指向 哈希表 "PROCLOCK hash"

接着 InitLocks()->hash_create() 在其中 调用 AllocSetContextCreate() ,创建 MemoryContext "LOCALLOCK hash" (做个回顾,当前 MemoryContext 见下面的 当前 MemoryContext 结构图 ),调用 DynaHashAlloc() ,在 MemoryContext "LOCALLOCK hash" 上分配空间,创建 哈希表索引 "LOCALLOCK hash" (见下面 "LOCALLOCK hash" 相关结构图 ,最后返回 InitLocks() ,让 HTAB * 类型静态 全局变量 LockMethodLocalHash 指向 哈希表 "LOCALLOCK hash"

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

LOCKTAG 结构被定义用于填充 16 字节。 请注意,如果 pg 要扩大 OID BlockNumber ,或 TransactionId 超过 32 位,这将需要调整。

       LOCKTAG 包含 lockmethodid 是为了共享内存里一个哈希表能够存储不同 lockmemthods 的锁。

typedef struct LOCKTAG

{

    uint32      locktag_field1; /* a 32-bit ID field */

    uint32      locktag_field2; /* a 32-bit ID field */

    uint32      locktag_field3; /* a 32-bit ID field */

    uint16      locktag_field4; /* a 16-bit ID field */

    uint8       locktag_type;   /* see enum LockTagType */

    uint8       locktag_lockmethodid;   /* lockmethod indicator */

} LOCKTAG;

 

每个被锁对象的锁信息:

tag :可锁对象的唯一标识符

grantMask :目前授予该对象的所有类型锁的位掩码

/*

  * Per-locked-object lock information:

  *

  * tag -- uniquely identifies the object being locked

  * grantMask -- bitmask for all lock types currently granted on this object.

  * waitMask -- bitmask for all lock types currently awaited on this object.

  * procLocks -- list of PROCLOCK objects for this lock.

  * waitProcs -- queue of processes waiting for this lock.

  * requested -- count of each lock type currently requested on the lock

  *      (includes requests already granted!!).

  * nRequested -- total requested locks of all types.

  * granted -- count of each lock type currently granted on the lock.

  * nGranted -- total granted locks of all types.

  *

  * Note: these counts count 1 for each backend.  Internally to a backend,

  * there may be multiple grabs on a particular lock, but this is not reflected

  * into shared memory.

  */

typedef struct LOCK

{

    /* hash key */

    LOCKTAG     tag;            /* unique identifier of lockable object */

 

    /* data */

    LOCKMASK    grantMask;      /* bitmask for lock types already granted */

    LOCKMASK    waitMask;       /* bitmask for lock types awaited */

    SHM_QUEUE   procLocks;      /* list of PROCLOCK objects assoc. with lock */

    PROC_QUEUE  waitProcs;      /* list of PGPROC objects waiting on lock */

    int          requested[MAX_LOCKMODES];       /* counts of requested locks */

    int          nRequested;     /* total of requested[] array */

    int          granted[MAX_LOCKMODES]; /* counts of granted locks */

    int          nGranted;       /* total of granted[] array */

} LOCK;

 

pg 可以有多个不同的 backend 进程在同一个开锁对象上持有或等待锁。 pg 需要为每个持有者 / 等待着存储一些信息。这些保存在结构 PROCLOCK 里。

PROCLOCKTAG 是在 proclock 哈希表里查找一个 PROCLOCK 项的关键信息。一个 PROCLOCKTAG 值唯一的标识一个可锁对象和一个持有者 / 等待着的组合。(这儿 pg 能使用指针,因为 PROCLOCKTAG 仅需要在 PROCLOCK 的生命周期里唯一,且不会在 lock proc 生存期以为)

为了不同的目的, backend 进程可以持有同一个锁:独立于会话锁, backend 进程跟踪事务锁。但是,这个在共享内存状态中没有反映出来: pg 仅跟踪持有锁的 backend 进程。这是可以的,因为 backend 进程不能阻塞自己。

holdMask 字段显示已经授予的由 proclock 代表的锁。注意,可能有一个具有 0 holdMask proclock 对象,对于任何锁,进程当前正在等待它。负责, holdMask 0 proclock 对象在方便的时候被尽快回收。

 

* releaseMask is workspace for LockReleaseAll(): it shows the locks due

  * to be released during the current call.  This must only be examined or

  * set by the backend owning the PROCLOCK.

每一个 PROCLOCK 对象被链接到链表,为了相关 LOCK 对象和所属 PGPROC 对象。注意, PROCLOCK 对象一被创建就就加入到这些链表,甚至还没有锁 lock 被授予的时候。等待 lock 锁被授予的 PGPROC 进程也会被链接到锁的等待进程( waitProcs )队列。

 

typedef struct PROCLOCKTAG

{

    /* NB: we assume this struct contains no padding! */

    LOCK       *myLock;         /* link to per-lockable-object information */

    PGPROC     *myProc;         /* link to PGPROC of owning backend */

} PROCLOCKTAG;

 

typedef struct PROCLOCK

{

    /* tag */

    PROCLOCKTAG tag;            /* unique identifier of proclock object */

 

    /* data */

    LOCKMASK    holdMask;       /* bitmask for lock types currently held */

    LOCKMASK    releaseMask;    /* bitmask for lock types to be released */

    SHM_QUEUE   lockLink;       /* list link in LOCK's list of proclocks */

    SHM_QUEUE   procLink;       /* list link in PGPROC's list of proclocks */

} PROCLOCK;

 

每一个backend 进程还维持一个本地哈希表,其记录着目前感兴趣的每一个锁lock 的信息。特别的,本地表记录着获得的这些锁的时间。这允许不额外访问共享内存的情况下对同一个锁多请求被执行。为了pg 能释放属于某个特别资源属主(ResourceOwner) 的锁。pg 还跟踪每个资源属主(ResourceOwner) 获得的锁数

typedef struct LOCALLOCKTAG

{

    LOCKTAG     lock;           /* identifies the lockable object */

    LOCKMODE    mode;           /* lock mode for this table entry */

} LOCALLOCKTAG;

 

typedef struct LOCALLOCKOWNER

{

    /*

      * Note: if owner is NULL then the lock is held on behalf of the session;

      * otherwise it is held on behalf of my current transaction.

      *

      * Must use a forward struct reference to avoid circularity.

      */

    struct ResourceOwnerData *owner;

    int64 nLocks;           /* # of times held by this owner */

} LOCALLOCKOWNER;

 

typedef struct LOCALLOCK

{

    /* tag */

    LOCALLOCKTAG tag;           /* unique identifier of locallock entry */

 

    /* data */

    LOCK       *lock;           /* associated LOCK object in shared mem */

    PROCLOCK   *proclock;       /* associated PROCLOCK object in shmem */

    uint32      hashcode;       /* copy of LOCKTAG's hash value */

    int64            nLocks;         /* total number of times lock is held */

    int          numLockOwners;  /* # of relevant ResourceOwners */

    int          maxLockOwners;  /* allocated size of array */

    LOCALLOCKOWNER *lockOwners; /* dynamically resizable array */

} LOCALLOCK;

 

#define LOCALLOCK_LOCKMETHOD(llock) ((llock).tag.lock.locktag_lockmethodid)

 


 

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

       为了精简上图,把创建 shmem 的哈希表索引 "ShmemIndex" 时创建的 HCTL 结构删掉了,这个结构的作用是记录创建可扩展哈希表的相关信息。增加了左边灰色底的部分,描述 共享内存 /shmem 里各变量物理布局概览,由下往上,由低地址到高地址。 图中黄色的索引项就是本节新增加的索引项。 其中的 " LOCK hash " 相关结构内存图、 " PROCLOCK hash " 相关结构内存 图下面分别给出,要不上面的图太大太复杂了。在MemoryContext " PROCLOCK hash " 中分配的哈希表 " PROCLOCK hash " 相关结构内存图和当前pg中的 MemoryContext 相关实例图也在下边一并给出。


"LOCK hash" 相关结构图

 


"PROCLOCK hash" 相关结构图

 


"LOCALLOCK hash" 相关结构图


当前 pg 中的 MemoryContext 结构图

 

  • 大小: 168.6 KB
  • 大小: 376.9 KB
  • 大小: 147 KB
  • 大小: 141 KB
  • 大小: 5.8 KB
  • 大小: 127.9 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语句主要用于...

    PostgreSQL中文手册9.2

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

    关于PostGreSQL中的存储过程

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

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

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

    MySQL和PostgreSQL的比较

    一个实例同样可以管理多个数据库,但这些数据库被组织成一个集群,存储在一个初始化时设定的磁盘区域中,该区域由一个目录构成,存储着所有数据。首次数据库创建则通过`initdb`命令完成。值得注意的是,PostgreSQL也...

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

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

    postgresql8.2.3 中文文档

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

    linux配置postgresql

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

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

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

    Postgresql存储过程

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

    PostgreSQL12.2中文手册.chm.7z

    PostgreSQL是一种开源的对象关系型数据库管理系统(ORM DBMS),它以其强大的功能、高度的可扩展性和稳定性在全球范围内被广泛使用。版本12.2是PostgreSQL的一个重要版本,提供了许多新特性和性能优化,旨在提升...

    Postgresql-10安装包

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

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

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

    PostgreSQL 12.2 安装手册

    初始化数据库是为了使 PostgreSQL 数据库能够正常运行。包括设置数据库密码、创建数据库目录、初始化数据库结构等步骤。 登录数据库 登录数据库是为了对 PostgreSQL 数据库进行管理和维护。使用 psql 命令可以连接...

    Linux下PostgreSQL安装与开机启动

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

    PostgreSql+PostGis创建空间数据库

    PostgreSql 是一个功能强大且开源的关系数据库管理系统,它提供了一个强大的平台来存储和管理数据。PostGis 是一个基于PostgreSql 的空间数据库扩展,它提供了对空间数据的支持,允许用户存储、查询和分析空间数据。...

    postgresql

    5. **启动与停止服务**:使用`pg_ctl`或系统服务管理工具(如Windows的Services或Linux的Systemd)启动和停止PostgreSQL服务。 6. **连接与管理**:可以通过命令行客户端`psql`连接到数据库,进行数据操作、查询和...

    PostgreSQL 8.4

    PostgreSQL 是一款功能强大的开源关系型数据库管理系统,因其稳定性、安全性和灵活性而备受业界推崇。在 PostgreSQL 8.4 版本中,用户可以享受到许多新特性,如改进的性能、增强的并发性以及更丰富的数据类型支持。...

Global site tag (gtag.js) - Google Analytics