LCOV - differential code coverage report
Current view: top level - src/backend/access/brin - brin_xlog.c (source / functions) Coverage Total Hit UBC GBC GNC CBC DCB
Current: c70b6db34ffeab48beef1fb4ce61bcad3772b8dd vs 06473f5a344df8c9594ead90a609b86f6724cff8 Lines: 96.8 % 155 150 5 6 144 6
Current Date: 2025-09-06 07:49:51 +0900 Functions: 100.0 % 9 9 5 4
Baseline: lcov-20250906-005545-baseline Branches: 61.5 % 65 40 25 1 39
Baseline Date: 2025-09-05 08:21:35 +0100 Line coverage date bins:
Legend: Lines:     hit not hit
Branches: + taken - not taken # not executed
(7,30] days: 100.0 % 6 6 6
(360..) days: 96.6 % 149 144 5 144
Function coverage date bins:
(360..) days: 100.0 % 9 9 5 4
Branch coverage date bins:
(360..) days: 61.5 % 65 40 25 1 39

 Age         Owner                    Branch data    TLA  Line data    Source code
                                  1                 :                : /*
                                  2                 :                :  * brin_xlog.c
                                  3                 :                :  *      XLog replay routines for BRIN indexes
                                  4                 :                :  *
                                  5                 :                :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
                                  6                 :                :  * Portions Copyright (c) 1994, Regents of the University of California
                                  7                 :                :  *
                                  8                 :                :  * IDENTIFICATION
                                  9                 :                :  *    src/backend/access/brin/brin_xlog.c
                                 10                 :                :  */
                                 11                 :                : #include "postgres.h"
                                 12                 :                : 
                                 13                 :                : #include "access/brin_page.h"
                                 14                 :                : #include "access/brin_pageops.h"
                                 15                 :                : #include "access/brin_xlog.h"
                                 16                 :                : #include "access/bufmask.h"
                                 17                 :                : #include "access/xlogutils.h"
                                 18                 :                : 
                                 19                 :                : 
                                 20                 :                : /*
                                 21                 :                :  * xlog replay routines
                                 22                 :                :  */
                                 23                 :                : static void
 3943 heikki.linnakangas@i       24                 :CBC          38 : brin_xlog_createidx(XLogReaderState *record)
                                 25                 :                : {
                                 26                 :             38 :     XLogRecPtr  lsn = record->EndRecPtr;
 3956 alvherre@alvh.no-ip.       27                 :             38 :     xl_brin_createidx *xlrec = (xl_brin_createidx *) XLogRecGetData(record);
                                 28                 :                :     Buffer      buf;
                                 29                 :                :     Page        page;
                                 30                 :                : 
                                 31                 :                :     /* create the index' metapage */
 3943 heikki.linnakangas@i       32                 :             38 :     buf = XLogInitBufferForRedo(record, 0);
 3956 alvherre@alvh.no-ip.       33         [ -  + ]:             38 :     Assert(BufferIsValid(buf));
    8 peter@eisentraut.org       34                 :GNC          38 :     page = BufferGetPage(buf);
 3956 alvherre@alvh.no-ip.       35                 :CBC          38 :     brin_metapage_init(page, xlrec->pagesPerRange, xlrec->version);
                                 36                 :             38 :     PageSetLSN(page, lsn);
                                 37                 :             38 :     MarkBufferDirty(buf);
                                 38                 :             38 :     UnlockReleaseBuffer(buf);
                                 39                 :             38 : }
                                 40                 :                : 
                                 41                 :                : /*
                                 42                 :                :  * Common part of an insert or update. Inserts the new tuple and updates the
                                 43                 :                :  * revmap.
                                 44                 :                :  */
                                 45                 :                : static void
 3943 heikki.linnakangas@i       46                 :            431 : brin_xlog_insert_update(XLogReaderState *record,
                                 47                 :                :                         xl_brin_insert *xlrec)
                                 48                 :                : {
                                 49                 :            431 :     XLogRecPtr  lsn = record->EndRecPtr;
                                 50                 :                :     Buffer      buffer;
                                 51                 :                :     BlockNumber regpgno;
                                 52                 :                :     Page        page;
                                 53                 :                :     XLogRedoAction action;
                                 54                 :                : 
                                 55                 :                :     /*
                                 56                 :                :      * If we inserted the first and only tuple on the page, re-initialize the
                                 57                 :                :      * page from scratch.
                                 58                 :                :      */
                                 59         [ +  + ]:            431 :     if (XLogRecGetInfo(record) & XLOG_BRIN_INIT_PAGE)
                                 60                 :                :     {
                                 61                 :             59 :         buffer = XLogInitBufferForRedo(record, 0);
 3426 kgrittn@postgresql.o       62                 :             59 :         page = BufferGetPage(buffer);
 3956 alvherre@alvh.no-ip.       63                 :             59 :         brin_page_init(page, BRIN_PAGETYPE_REGULAR);
                                 64                 :             59 :         action = BLK_NEEDS_REDO;
                                 65                 :                :     }
                                 66                 :                :     else
                                 67                 :                :     {
 3943 heikki.linnakangas@i       68                 :            372 :         action = XLogReadBufferForRedo(record, 0, &buffer);
                                 69                 :                :     }
                                 70                 :                : 
                                 71                 :                :     /* need this page's blkno to store in revmap */
 3725 alvherre@alvh.no-ip.       72                 :            431 :     regpgno = BufferGetBlockNumber(buffer);
                                 73                 :                : 
                                 74                 :                :     /* insert the index item into the page */
 3956                            75         [ +  + ]:            431 :     if (action == BLK_NEEDS_REDO)
                                 76                 :                :     {
                                 77                 :                :         OffsetNumber offnum;
                                 78                 :                :         BrinTuple  *tuple;
                                 79                 :                :         Size        tuplen;
                                 80                 :                : 
 3943 heikki.linnakangas@i       81                 :            427 :         tuple = (BrinTuple *) XLogRecGetBlockData(record, 0, &tuplen);
                                 82                 :                : 
 3956 alvherre@alvh.no-ip.       83         [ -  + ]:            427 :         Assert(tuple->bt_blkno == xlrec->heapBlk);
                                 84                 :                : 
    8 peter@eisentraut.org       85                 :GNC         427 :         page = BufferGetPage(buffer);
 3943 heikki.linnakangas@i       86                 :CBC         427 :         offnum = xlrec->offnum;
 3956 alvherre@alvh.no-ip.       87         [ -  + ]:            427 :         if (PageGetMaxOffsetNumber(page) + 1 < offnum)
 3956 alvherre@alvh.no-ip.       88         [ #  # ]:UBC           0 :             elog(PANIC, "brin_xlog_insert_update: invalid max offset number");
                                 89                 :                : 
 3943 heikki.linnakangas@i       90                 :CBC         427 :         offnum = PageAddItem(page, (Item) tuple, tuplen, offnum, true, false);
 3956 alvherre@alvh.no-ip.       91         [ -  + ]:            427 :         if (offnum == InvalidOffsetNumber)
 3956 alvherre@alvh.no-ip.       92         [ #  # ]:UBC           0 :             elog(PANIC, "brin_xlog_insert_update: failed to add tuple");
                                 93                 :                : 
 3956 alvherre@alvh.no-ip.       94                 :CBC         427 :         PageSetLSN(page, lsn);
                                 95                 :            427 :         MarkBufferDirty(buffer);
                                 96                 :                :     }
                                 97         [ +  - ]:            431 :     if (BufferIsValid(buffer))
                                 98                 :            431 :         UnlockReleaseBuffer(buffer);
                                 99                 :                : 
                                100                 :                :     /* update the revmap */
 3943 heikki.linnakangas@i      101                 :            431 :     action = XLogReadBufferForRedo(record, 1, &buffer);
 3956 alvherre@alvh.no-ip.      102         [ +  + ]:            431 :     if (action == BLK_NEEDS_REDO)
                                103                 :                :     {
                                104                 :                :         ItemPointerData tid;
                                105                 :                : 
 3725                           106                 :            428 :         ItemPointerSet(&tid, regpgno, xlrec->offnum);
    8 peter@eisentraut.org      107                 :GNC         428 :         page = BufferGetPage(buffer);
                                108                 :                : 
 3956 alvherre@alvh.no-ip.      109                 :CBC         428 :         brinSetHeapBlockItemptr(buffer, xlrec->pagesPerRange, xlrec->heapBlk,
                                110                 :                :                                 tid);
                                111                 :            428 :         PageSetLSN(page, lsn);
                                112                 :            428 :         MarkBufferDirty(buffer);
                                113                 :                :     }
                                114         [ +  - ]:            431 :     if (BufferIsValid(buffer))
                                115                 :            431 :         UnlockReleaseBuffer(buffer);
                                116                 :                : 
                                117                 :                :     /* XXX no FSM updates here ... */
                                118                 :            431 : }
                                119                 :                : 
                                120                 :                : /*
                                121                 :                :  * replay a BRIN index insertion
                                122                 :                :  */
                                123                 :                : static void
 3943 heikki.linnakangas@i      124                 :            427 : brin_xlog_insert(XLogReaderState *record)
                                125                 :                : {
 3956 alvherre@alvh.no-ip.      126                 :            427 :     xl_brin_insert *xlrec = (xl_brin_insert *) XLogRecGetData(record);
                                127                 :                : 
 3943 heikki.linnakangas@i      128                 :            427 :     brin_xlog_insert_update(record, xlrec);
 3956 alvherre@alvh.no-ip.      129                 :            427 : }
                                130                 :                : 
                                131                 :                : /*
                                132                 :                :  * replay a BRIN index update
                                133                 :                :  */
                                134                 :                : static void
 3943 heikki.linnakangas@i      135                 :              4 : brin_xlog_update(XLogReaderState *record)
                                136                 :                : {
                                137                 :              4 :     XLogRecPtr  lsn = record->EndRecPtr;
 3956 alvherre@alvh.no-ip.      138                 :              4 :     xl_brin_update *xlrec = (xl_brin_update *) XLogRecGetData(record);
                                139                 :                :     Buffer      buffer;
                                140                 :                :     XLogRedoAction action;
                                141                 :                : 
                                142                 :                :     /* First remove the old tuple */
 3943 heikki.linnakangas@i      143                 :              4 :     action = XLogReadBufferForRedo(record, 2, &buffer);
 3956 alvherre@alvh.no-ip.      144         [ +  - ]:              4 :     if (action == BLK_NEEDS_REDO)
                                145                 :                :     {
                                146                 :                :         Page        page;
                                147                 :                :         OffsetNumber offnum;
                                148                 :                : 
    8 peter@eisentraut.org      149                 :GNC           4 :         page = BufferGetPage(buffer);
                                150                 :                : 
 3943 heikki.linnakangas@i      151                 :CBC           4 :         offnum = xlrec->oldOffnum;
                                152                 :                : 
 3284 tgl@sss.pgh.pa.us         153                 :              4 :         PageIndexTupleDeleteNoCompact(page, offnum);
                                154                 :                : 
 3956 alvherre@alvh.no-ip.      155                 :              4 :         PageSetLSN(page, lsn);
                                156                 :              4 :         MarkBufferDirty(buffer);
                                157                 :                :     }
                                158                 :                : 
                                159                 :                :     /* Then insert the new tuple and update revmap, like in an insertion. */
 3943 heikki.linnakangas@i      160                 :              4 :     brin_xlog_insert_update(record, &xlrec->insert);
                                161                 :                : 
 3956 alvherre@alvh.no-ip.      162         [ +  - ]:              4 :     if (BufferIsValid(buffer))
                                163                 :              4 :         UnlockReleaseBuffer(buffer);
                                164                 :              4 : }
                                165                 :                : 
                                166                 :                : /*
                                167                 :                :  * Update a tuple on a single page.
                                168                 :                :  */
                                169                 :                : static void
 3943 heikki.linnakangas@i      170                 :            617 : brin_xlog_samepage_update(XLogReaderState *record)
                                171                 :                : {
                                172                 :            617 :     XLogRecPtr  lsn = record->EndRecPtr;
                                173                 :                :     xl_brin_samepage_update *xlrec;
                                174                 :                :     Buffer      buffer;
                                175                 :                :     XLogRedoAction action;
                                176                 :                : 
 3956 alvherre@alvh.no-ip.      177                 :            617 :     xlrec = (xl_brin_samepage_update *) XLogRecGetData(record);
 3943 heikki.linnakangas@i      178                 :            617 :     action = XLogReadBufferForRedo(record, 0, &buffer);
 3956 alvherre@alvh.no-ip.      179         [ +  + ]:            617 :     if (action == BLK_NEEDS_REDO)
                                180                 :                :     {
                                181                 :                :         Size        tuplen;
                                182                 :                :         BrinTuple  *brintuple;
                                183                 :                :         Page        page;
                                184                 :                :         OffsetNumber offnum;
                                185                 :                : 
 3701                           186                 :            594 :         brintuple = (BrinTuple *) XLogRecGetBlockData(record, 0, &tuplen);
                                187                 :                : 
    8 peter@eisentraut.org      188                 :GNC         594 :         page = BufferGetPage(buffer);
                                189                 :                : 
 3943 heikki.linnakangas@i      190                 :CBC         594 :         offnum = xlrec->offnum;
                                191                 :                : 
 3284 tgl@sss.pgh.pa.us         192         [ -  + ]:            594 :         if (!PageIndexTupleOverwrite(page, offnum, (Item) brintuple, tuplen))
 3284 tgl@sss.pgh.pa.us         193         [ #  # ]:UBC           0 :             elog(PANIC, "brin_xlog_samepage_update: failed to replace tuple");
                                194                 :                : 
 3956 alvherre@alvh.no-ip.      195                 :CBC         594 :         PageSetLSN(page, lsn);
                                196                 :            594 :         MarkBufferDirty(buffer);
                                197                 :                :     }
                                198         [ +  - ]:            617 :     if (BufferIsValid(buffer))
                                199                 :            617 :         UnlockReleaseBuffer(buffer);
                                200                 :                : 
                                201                 :                :     /* XXX no FSM updates here ... */
                                202                 :            617 : }
                                203                 :                : 
                                204                 :                : /*
                                205                 :                :  * Replay a revmap page extension
                                206                 :                :  */
                                207                 :                : static void
 3943 heikki.linnakangas@i      208                 :             38 : brin_xlog_revmap_extend(XLogReaderState *record)
                                209                 :                : {
                                210                 :             38 :     XLogRecPtr  lsn = record->EndRecPtr;
                                211                 :                :     xl_brin_revmap_extend *xlrec;
                                212                 :                :     Buffer      metabuf;
                                213                 :                :     Buffer      buf;
                                214                 :                :     Page        page;
                                215                 :                :     BlockNumber targetBlk;
                                216                 :                :     XLogRedoAction action;
                                217                 :                : 
 3956 alvherre@alvh.no-ip.      218                 :             38 :     xlrec = (xl_brin_revmap_extend *) XLogRecGetData(record);
                                219                 :                : 
 3943 heikki.linnakangas@i      220                 :             38 :     XLogRecGetBlockTag(record, 1, NULL, NULL, &targetBlk);
                                221         [ -  + ]:             38 :     Assert(xlrec->targetBlk == targetBlk);
                                222                 :                : 
                                223                 :                :     /* Update the metapage */
                                224                 :             38 :     action = XLogReadBufferForRedo(record, 0, &metabuf);
 3956 alvherre@alvh.no-ip.      225         [ +  + ]:             38 :     if (action == BLK_NEEDS_REDO)
                                226                 :                :     {
                                227                 :                :         Page        metapg;
                                228                 :                :         BrinMetaPageData *metadata;
                                229                 :                : 
 3426 kgrittn@postgresql.o      230                 :             37 :         metapg = BufferGetPage(metabuf);
 3956 alvherre@alvh.no-ip.      231                 :             37 :         metadata = (BrinMetaPageData *) PageGetContents(metapg);
                                232                 :                : 
                                233         [ -  + ]:             37 :         Assert(metadata->lastRevmapPage == xlrec->targetBlk - 1);
                                234                 :             37 :         metadata->lastRevmapPage = xlrec->targetBlk;
                                235                 :                : 
                                236                 :             37 :         PageSetLSN(metapg, lsn);
                                237                 :                : 
                                238                 :                :         /*
                                239                 :                :          * Set pd_lower just past the end of the metadata.  This is essential,
                                240                 :                :          * because without doing so, metadata will be lost if xlog.c
                                241                 :                :          * compresses the page.  (We must do this here because pre-v11
                                242                 :                :          * versions of PG did not set the metapage's pd_lower correctly, so a
                                243                 :                :          * pg_upgraded index might contain the wrong value.)
                                244                 :                :          */
 2865 tgl@sss.pgh.pa.us         245                 :             37 :         ((PageHeader) metapg)->pd_lower =
                                246                 :             37 :             ((char *) metadata + sizeof(BrinMetaPageData)) - (char *) metapg;
                                247                 :                : 
 3956 alvherre@alvh.no-ip.      248                 :             37 :         MarkBufferDirty(metabuf);
                                249                 :                :     }
                                250                 :                : 
                                251                 :                :     /*
                                252                 :                :      * Re-init the target block as a revmap page.  There's never a full- page
                                253                 :                :      * image here.
                                254                 :                :      */
                                255                 :                : 
 3943 heikki.linnakangas@i      256                 :             38 :     buf = XLogInitBufferForRedo(record, 1);
    8 peter@eisentraut.org      257                 :GNC          38 :     page = BufferGetPage(buf);
 3956 alvherre@alvh.no-ip.      258                 :CBC          38 :     brin_page_init(page, BRIN_PAGETYPE_REVMAP);
                                259                 :                : 
                                260                 :             38 :     PageSetLSN(page, lsn);
                                261                 :             38 :     MarkBufferDirty(buf);
                                262                 :                : 
                                263                 :             38 :     UnlockReleaseBuffer(buf);
                                264         [ +  - ]:             38 :     if (BufferIsValid(metabuf))
                                265                 :             38 :         UnlockReleaseBuffer(metabuf);
                                266                 :             38 : }
                                267                 :                : 
                                268                 :                : static void
 3080                           269                 :              7 : brin_xlog_desummarize_page(XLogReaderState *record)
                                270                 :                : {
                                271                 :              7 :     XLogRecPtr  lsn = record->EndRecPtr;
                                272                 :                :     xl_brin_desummarize *xlrec;
                                273                 :                :     Buffer      buffer;
                                274                 :                :     XLogRedoAction action;
                                275                 :                : 
                                276                 :              7 :     xlrec = (xl_brin_desummarize *) XLogRecGetData(record);
                                277                 :                : 
                                278                 :                :     /* Update the revmap */
                                279                 :              7 :     action = XLogReadBufferForRedo(record, 0, &buffer);
                                280         [ +  + ]:              7 :     if (action == BLK_NEEDS_REDO)
                                281                 :                :     {
                                282                 :                :         ItemPointerData iptr;
                                283                 :                : 
                                284                 :              4 :         ItemPointerSetInvalid(&iptr);
                                285                 :              4 :         brinSetHeapBlockItemptr(buffer, xlrec->pagesPerRange, xlrec->heapBlk, iptr);
                                286                 :                : 
                                287                 :              4 :         PageSetLSN(BufferGetPage(buffer), lsn);
                                288                 :              4 :         MarkBufferDirty(buffer);
                                289                 :                :     }
                                290         [ +  - ]:              7 :     if (BufferIsValid(buffer))
                                291                 :              7 :         UnlockReleaseBuffer(buffer);
                                292                 :                : 
                                293                 :                :     /* remove the leftover entry from the regular page */
                                294                 :              7 :     action = XLogReadBufferForRedo(record, 1, &buffer);
                                295         [ +  + ]:              7 :     if (action == BLK_NEEDS_REDO)
                                296                 :                :     {
 3034 bruce@momjian.us          297                 :              5 :         Page        regPg = BufferGetPage(buffer);
                                298                 :                : 
 3080 alvherre@alvh.no-ip.      299                 :              5 :         PageIndexTupleDeleteNoCompact(regPg, xlrec->regOffset);
                                300                 :                : 
                                301                 :              5 :         PageSetLSN(regPg, lsn);
                                302                 :              5 :         MarkBufferDirty(buffer);
                                303                 :                :     }
                                304         [ +  - ]:              7 :     if (BufferIsValid(buffer))
                                305                 :              7 :         UnlockReleaseBuffer(buffer);
                                306                 :              7 : }
                                307                 :                : 
                                308                 :                : void
 3943 heikki.linnakangas@i      309                 :           1131 : brin_redo(XLogReaderState *record)
                                310                 :                : {
                                311                 :           1131 :     uint8       info = XLogRecGetInfo(record) & ~XLR_INFO_MASK;
                                312                 :                : 
 3956 alvherre@alvh.no-ip.      313   [ +  +  +  +  :           1131 :     switch (info & XLOG_BRIN_OPMASK)
                                           +  +  - ]
                                314                 :                :     {
                                315                 :             38 :         case XLOG_BRIN_CREATE_INDEX:
 3943 heikki.linnakangas@i      316                 :             38 :             brin_xlog_createidx(record);
 3956 alvherre@alvh.no-ip.      317                 :             38 :             break;
                                318                 :            427 :         case XLOG_BRIN_INSERT:
 3943 heikki.linnakangas@i      319                 :            427 :             brin_xlog_insert(record);
 3956 alvherre@alvh.no-ip.      320                 :            427 :             break;
                                321                 :              4 :         case XLOG_BRIN_UPDATE:
 3943 heikki.linnakangas@i      322                 :              4 :             brin_xlog_update(record);
 3956 alvherre@alvh.no-ip.      323                 :              4 :             break;
                                324                 :            617 :         case XLOG_BRIN_SAMEPAGE_UPDATE:
 3943 heikki.linnakangas@i      325                 :            617 :             brin_xlog_samepage_update(record);
 3956 alvherre@alvh.no-ip.      326                 :            617 :             break;
                                327                 :             38 :         case XLOG_BRIN_REVMAP_EXTEND:
 3943 heikki.linnakangas@i      328                 :             38 :             brin_xlog_revmap_extend(record);
 3956 alvherre@alvh.no-ip.      329                 :             38 :             break;
 3080                           330                 :              7 :         case XLOG_BRIN_DESUMMARIZE:
                                331                 :              7 :             brin_xlog_desummarize_page(record);
                                332                 :              7 :             break;
 3956 alvherre@alvh.no-ip.      333                 :UBC           0 :         default:
                                334         [ #  # ]:              0 :             elog(PANIC, "brin_redo: unknown op code %u", info);
                                335                 :                :     }
 3956 alvherre@alvh.no-ip.      336                 :CBC        1131 : }
                                337                 :                : 
                                338                 :                : /*
                                339                 :                :  * Mask a BRIN page before doing consistency checks.
                                340                 :                :  */
                                341                 :                : void
 3132 rhaas@postgresql.org      342                 :           3150 : brin_mask(char *pagedata, BlockNumber blkno)
                                343                 :                : {
                                344                 :           3150 :     Page        page = (Page) pagedata;
 2865 tgl@sss.pgh.pa.us         345                 :           3150 :     PageHeader  pagehdr = (PageHeader) page;
                                346                 :                : 
 2906 rhaas@postgresql.org      347                 :           3150 :     mask_page_lsn_and_checksum(page);
                                348                 :                : 
 3132                           349                 :           3150 :     mask_page_hint_bits(page);
                                350                 :                : 
                                351                 :                :     /*
                                352                 :                :      * Regular brin pages contain unused space which needs to be masked.
                                353                 :                :      * Similarly for meta pages, but mask it only if pd_lower appears to have
                                354                 :                :      * been set correctly.
                                355                 :                :      */
 2865 tgl@sss.pgh.pa.us         356         [ +  + ]:           3150 :     if (BRIN_IS_REGULAR_PAGE(page) ||
                                357   [ +  +  +  - ]:           1090 :         (BRIN_IS_META_PAGE(page) && pagehdr->pd_lower > SizeOfPageHeaderData))
                                358                 :                :     {
 3132 rhaas@postgresql.org      359                 :           2210 :         mask_unused_space(page);
                                360                 :                :     }
                                361                 :                : 
                                362                 :                :     /*
                                363                 :                :      * BRIN_EVACUATE_PAGE is not WAL-logged, since it's of no use in recovery.
                                364                 :                :      * Mask it.  See brin_start_evacuating_page() for details.
                                365                 :                :      */
 1128 alvherre@alvh.no-ip.      366                 :           3150 :     BrinPageFlags(page) &= ~BRIN_EVACUATE_PAGE;
 3132 rhaas@postgresql.org      367                 :           3150 : }
        

Generated by: LCOV version 2.4-beta