LCOV - code coverage report
Current view: top level - src/test/modules/test_ginpostinglist - test_ginpostinglist.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 21 25 84.0 %
Date: 2025-01-18 05:15:39 Functions: 4 4 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*--------------------------------------------------------------------------
       2             :  *
       3             :  * test_ginpostinglist.c
       4             :  *      Test varbyte-encoding in ginpostinglist.c
       5             :  *
       6             :  * Copyright (c) 2019-2025, PostgreSQL Global Development Group
       7             :  *
       8             :  * IDENTIFICATION
       9             :  *      src/test/modules/test_ginpostinglist/test_ginpostinglist.c
      10             :  *
      11             :  * -------------------------------------------------------------------------
      12             :  */
      13             : #include "postgres.h"
      14             : 
      15             : #include "access/gin_private.h"
      16             : #include "access/ginblock.h"
      17             : #include "access/htup_details.h"
      18             : #include "fmgr.h"
      19             : 
      20           2 : PG_MODULE_MAGIC;
      21             : 
      22           4 : PG_FUNCTION_INFO_V1(test_ginpostinglist);
      23             : 
      24             : /*
      25             :  * Encodes a pair of TIDs, and decodes it back. The first TID is always
      26             :  * (0, 1), the second one is formed from the blk/off arguments. The 'maxsize'
      27             :  * argument is passed to ginCompressPostingList(); it can be used to test the
      28             :  * overflow checks.
      29             :  *
      30             :  * The reason that we test a pair, instead of just a single TID, is that
      31             :  * the GinPostingList stores the first TID as is, and the varbyte-encoding
      32             :  * is only used for the deltas between TIDs. So testing a single TID would
      33             :  * not exercise the varbyte encoding at all.
      34             :  *
      35             :  * This function prints NOTICEs to describe what is tested, and how large the
      36             :  * resulting GinPostingList is. Any incorrect results, e.g. if the encode +
      37             :  * decode round trip doesn't return the original input, are reported as
      38             :  * ERRORs.
      39             :  */
      40             : static void
      41           8 : test_itemptr_pair(BlockNumber blk, OffsetNumber off, int maxsize)
      42             : {
      43             :     ItemPointerData orig_itemptrs[2];
      44             :     ItemPointer decoded_itemptrs;
      45             :     GinPostingList *pl;
      46             :     int         nwritten;
      47             :     int         ndecoded;
      48             : 
      49           8 :     elog(NOTICE, "testing with (%u, %d), (%u, %d), max %d bytes",
      50             :          0, 1, blk, off, maxsize);
      51           8 :     ItemPointerSet(&orig_itemptrs[0], 0, 1);
      52           8 :     ItemPointerSet(&orig_itemptrs[1], blk, off);
      53             : 
      54             :     /* Encode, and decode it back */
      55           8 :     pl = ginCompressPostingList(orig_itemptrs, 2, maxsize, &nwritten);
      56           8 :     elog(NOTICE, "encoded %d item pointers to %zu bytes",
      57             :          nwritten, SizeOfGinPostingList(pl));
      58             : 
      59           8 :     if (SizeOfGinPostingList(pl) > maxsize)
      60           0 :         elog(ERROR, "overflow: result was %zu bytes, max %d",
      61             :              SizeOfGinPostingList(pl), maxsize);
      62             : 
      63           8 :     decoded_itemptrs = ginPostingListDecode(pl, &ndecoded);
      64           8 :     if (nwritten != ndecoded)
      65           0 :         elog(NOTICE, "encoded %d itemptrs, %d came back", nwritten, ndecoded);
      66             : 
      67             :     /* Check the result */
      68           8 :     if (!ItemPointerEquals(&orig_itemptrs[0], &decoded_itemptrs[0]))
      69           0 :         elog(ERROR, "mismatch on first itemptr: (%u, %d) vs (%u, %d)",
      70             :              0, 1,
      71             :              ItemPointerGetBlockNumber(&decoded_itemptrs[0]),
      72             :              ItemPointerGetOffsetNumber(&decoded_itemptrs[0]));
      73             : 
      74           8 :     if (ndecoded == 2 &&
      75           6 :         !ItemPointerEquals(&orig_itemptrs[0], &decoded_itemptrs[0]))
      76             :     {
      77           0 :         elog(ERROR, "mismatch on second itemptr: (%u, %d) vs (%u, %d)",
      78             :              0, 1,
      79             :              ItemPointerGetBlockNumber(&decoded_itemptrs[0]),
      80             :              ItemPointerGetOffsetNumber(&decoded_itemptrs[0]));
      81             :     }
      82           8 : }
      83             : 
      84             : /*
      85             :  * SQL-callable entry point to perform all tests.
      86             :  */
      87             : Datum
      88           2 : test_ginpostinglist(PG_FUNCTION_ARGS)
      89             : {
      90           2 :     test_itemptr_pair(0, 2, 14);
      91           2 :     test_itemptr_pair(0, MaxHeapTuplesPerPage, 14);
      92           2 :     test_itemptr_pair(MaxBlockNumber, MaxHeapTuplesPerPage, 14);
      93           2 :     test_itemptr_pair(MaxBlockNumber, MaxHeapTuplesPerPage, 16);
      94             : 
      95           2 :     PG_RETURN_VOID();
      96             : }

Generated by: LCOV version 1.14