LCOV - differential code coverage report
Current view: top level - src/backend/executor - nodeSamplescan.c (source / functions) Coverage Total Hit UIC UBC GIC GNC CBC DCB
Current: bed3ffbf9d952be6c7d739d068cdce44c046dfb7 vs 574581b50ac9c63dd9e4abebb731a3b67e5b50f6 Lines: 96.2 % 106 102 4 3 99 2
Current Date: 2026-05-05 10:23:31 +0900 Functions: 87.5 % 8 7 1 1 6
Baseline: lcov-20260505-025707-baseline Branches: 78.6 % 42 33 1 8 1 32
Baseline Date: 2026-05-05 10:27:06 +0900 Line coverage date bins:
Legend: Lines:     hit not hit
Branches: + taken - not taken # not executed
(30,360] days: 100.0 % 3 3 3
(360..) days: 96.1 % 103 99 4 99
Function coverage date bins:
(360..) days: 87.5 % 8 7 1 1 6
Branch coverage date bins:
(360..) days: 78.6 % 42 33 1 8 1 32

 Age         Owner                    Branch data    TLA  Line data    Source code
                                  1                 :                : /*-------------------------------------------------------------------------
                                  2                 :                :  *
                                  3                 :                :  * nodeSamplescan.c
                                  4                 :                :  *    Support routines for sample scans of relations (table sampling).
                                  5                 :                :  *
                                  6                 :                :  * Portions Copyright (c) 1996-2026, PostgreSQL Global Development Group
                                  7                 :                :  * Portions Copyright (c) 1994, Regents of the University of California
                                  8                 :                :  *
                                  9                 :                :  *
                                 10                 :                :  * IDENTIFICATION
                                 11                 :                :  *    src/backend/executor/nodeSamplescan.c
                                 12                 :                :  *
                                 13                 :                :  *-------------------------------------------------------------------------
                                 14                 :                :  */
                                 15                 :                : #include "postgres.h"
                                 16                 :                : 
                                 17                 :                : #include "access/relscan.h"
                                 18                 :                : #include "access/tableam.h"
                                 19                 :                : #include "access/tsmapi.h"
                                 20                 :                : #include "common/pg_prng.h"
                                 21                 :                : #include "executor/executor.h"
                                 22                 :                : #include "executor/nodeSamplescan.h"
                                 23                 :                : #include "utils/fmgrprotos.h"
                                 24                 :                : #include "utils/rel.h"
                                 25                 :                : 
                                 26                 :                : static TupleTableSlot *SampleNext(SampleScanState *node);
                                 27                 :                : static void tablesample_init(SampleScanState *scanstate);
                                 28                 :                : static TupleTableSlot *tablesample_getnext(SampleScanState *scanstate);
                                 29                 :                : 
                                 30                 :                : /* ----------------------------------------------------------------
                                 31                 :                :  *                      Scan Support
                                 32                 :                :  * ----------------------------------------------------------------
                                 33                 :                :  */
                                 34                 :                : 
                                 35                 :                : /* ----------------------------------------------------------------
                                 36                 :                :  *      SampleNext
                                 37                 :                :  *
                                 38                 :                :  *      This is a workhorse for ExecSampleScan
                                 39                 :                :  * ----------------------------------------------------------------
                                 40                 :                :  */
                                 41                 :                : static TupleTableSlot *
 4008 simon@2ndQuadrant.co       42                 :CBC      160843 : SampleNext(SampleScanState *node)
                                 43                 :                : {
                                 44                 :                :     /*
                                 45                 :                :      * if this is first call within a scan, initialize
                                 46                 :                :      */
 3937 tgl@sss.pgh.pa.us          47         [ +  + ]:         160843 :     if (!node->begun)
                                 48                 :            139 :         tablesample_init(node);
                                 49                 :                : 
                                 50                 :                :     /*
                                 51                 :                :      * get the next tuple, and store it in our result slot
                                 52                 :                :      */
 2593 andres@anarazel.de         53                 :         160817 :     return tablesample_getnext(node);
                                 54                 :                : }
                                 55                 :                : 
                                 56                 :                : /*
                                 57                 :                :  * SampleRecheck -- access method routine to recheck a tuple in EvalPlanQual
                                 58                 :                :  */
                                 59                 :                : static bool
 4008 simon@2ndQuadrant.co       60                 :UBC           0 : SampleRecheck(SampleScanState *node, TupleTableSlot *slot)
                                 61                 :                : {
                                 62                 :                :     /*
                                 63                 :                :      * No need to recheck for SampleScan, since like SeqScan we don't pass any
                                 64                 :                :      * checkable keys to heap_beginscan.
                                 65                 :                :      */
                                 66                 :              0 :     return true;
                                 67                 :                : }
                                 68                 :                : 
                                 69                 :                : /* ----------------------------------------------------------------
                                 70                 :                :  *      ExecSampleScan(node)
                                 71                 :                :  *
                                 72                 :                :  *      Scans the relation using the sampling method and returns
                                 73                 :                :  *      the next qualifying tuple.
                                 74                 :                :  *      We call the ExecScan() routine and pass it the appropriate
                                 75                 :                :  *      access method functions.
                                 76                 :                :  * ----------------------------------------------------------------
                                 77                 :                :  */
                                 78                 :                : static TupleTableSlot *
 3214 andres@anarazel.de         79                 :CBC      160839 : ExecSampleScan(PlanState *pstate)
                                 80                 :                : {
                                 81                 :         160839 :     SampleScanState *node = castNode(SampleScanState, pstate);
                                 82                 :                : 
                                 83                 :         160839 :     return ExecScan(&node->ss,
                                 84                 :                :                     (ExecScanAccessMtd) SampleNext,
                                 85                 :                :                     (ExecScanRecheckMtd) SampleRecheck);
                                 86                 :                : }
                                 87                 :                : 
                                 88                 :                : /* ----------------------------------------------------------------
                                 89                 :                :  *      ExecInitSampleScan
                                 90                 :                :  * ----------------------------------------------------------------
                                 91                 :                :  */
                                 92                 :                : SampleScanState *
 4008 simon@2ndQuadrant.co       93                 :            200 : ExecInitSampleScan(SampleScan *node, EState *estate, int eflags)
                                 94                 :                : {
                                 95                 :                :     SampleScanState *scanstate;
 3937 tgl@sss.pgh.pa.us          96                 :            200 :     TableSampleClause *tsc = node->tablesample;
                                 97                 :                :     TsmRoutine *tsm;
                                 98                 :                : 
 4008 simon@2ndQuadrant.co       99         [ -  + ]:            200 :     Assert(outerPlan(node) == NULL);
                                100         [ -  + ]:            200 :     Assert(innerPlan(node) == NULL);
                                101                 :                : 
                                102                 :                :     /*
                                103                 :                :      * create state structure
                                104                 :                :      */
                                105                 :            200 :     scanstate = makeNode(SampleScanState);
                                106                 :            200 :     scanstate->ss.ps.plan = (Plan *) node;
                                107                 :            200 :     scanstate->ss.ps.state = estate;
 3214 andres@anarazel.de        108                 :            200 :     scanstate->ss.ps.ExecProcNode = ExecSampleScan;
                                109                 :                : 
                                110                 :                :     /*
                                111                 :                :      * Miscellaneous initialization
                                112                 :                :      *
                                113                 :                :      * create expression context for node
                                114                 :                :      */
 4008 simon@2ndQuadrant.co      115                 :            200 :     ExecAssignExprContext(estate, &scanstate->ss.ps);
                                116                 :                : 
                                117                 :                :     /*
                                118                 :                :      * open the scan relation
                                119                 :                :      */
 3000 andres@anarazel.de        120                 :            200 :     scanstate->ss.ss_currentRelation =
                                121                 :            200 :         ExecOpenScanRelation(estate,
                                122                 :                :                              node->scan.scanrelid,
                                123                 :                :                              eflags);
                                124                 :                : 
                                125                 :                :     /* we won't set up the HeapScanDesc till later */
                                126                 :            200 :     scanstate->ss.ss_currentScanDesc = NULL;
                                127                 :                : 
                                128                 :                :     /* and create slot with appropriate rowtype */
                                129                 :            200 :     ExecInitScanTupleSlot(estate, &scanstate->ss,
 2728                           130                 :            200 :                           RelationGetDescr(scanstate->ss.ss_currentRelation),
                                131                 :                :                           table_slot_callbacks(scanstate->ss.ss_currentRelation),
                                132                 :                :                           TTS_FLAG_OBEYS_NOT_NULL_CONSTRAINTS);
                                133                 :                : 
                                134                 :                :     /*
                                135                 :                :      * Initialize result type and projection.
                                136                 :                :      */
 2734                           137                 :            200 :     ExecInitResultTypeTL(&scanstate->ss.ps);
 3000                           138                 :            200 :     ExecAssignScanProjectionInfo(&scanstate->ss);
                                139                 :                : 
                                140                 :                :     /*
                                141                 :                :      * initialize child expressions
                                142                 :                :      */
                                143                 :            200 :     scanstate->ss.ps.qual =
                                144                 :            200 :         ExecInitQual(node->scan.plan.qual, (PlanState *) scanstate);
                                145                 :                : 
                                146                 :            200 :     scanstate->args = ExecInitExprList(tsc->args, (PlanState *) scanstate);
                                147                 :            200 :     scanstate->repeatable =
                                148                 :            200 :         ExecInitExpr(tsc->repeatable, (PlanState *) scanstate);
                                149                 :                : 
                                150                 :                :     /*
                                151                 :                :      * If we don't have a REPEATABLE clause, select a random seed.  We want to
                                152                 :                :      * do this just once, since the seed shouldn't change over rescans.
                                153                 :                :      */
 3937 tgl@sss.pgh.pa.us         154         [ +  + ]:            200 :     if (tsc->repeatable == NULL)
 1619                           155                 :            112 :         scanstate->seed = pg_prng_uint32(&pg_global_prng_state);
                                156                 :                : 
                                157                 :                :     /*
                                158                 :                :      * Finally, initialize the TABLESAMPLE method handler.
                                159                 :                :      */
 3937                           160                 :            200 :     tsm = GetTsmRoutine(tsc->tsmhandler);
                                161                 :            200 :     scanstate->tsmroutine = tsm;
                                162                 :            200 :     scanstate->tsm_state = NULL;
                                163                 :                : 
                                164         [ +  - ]:            200 :     if (tsm->InitSampleScan)
                                165                 :            200 :         tsm->InitSampleScan(scanstate, eflags);
                                166                 :                : 
                                167                 :                :     /* We'll do BeginSampleScan later; we can't evaluate params yet */
                                168                 :            200 :     scanstate->begun = false;
                                169                 :                : 
 4008 simon@2ndQuadrant.co      170                 :            200 :     return scanstate;
                                171                 :                : }
                                172                 :                : 
                                173                 :                : /* ----------------------------------------------------------------
                                174                 :                :  *      ExecEndSampleScan
                                175                 :                :  *
                                176                 :                :  *      frees any storage allocated through C routines.
                                177                 :                :  * ----------------------------------------------------------------
                                178                 :                :  */
                                179                 :                : void
                                180                 :            174 : ExecEndSampleScan(SampleScanState *node)
                                181                 :                : {
                                182                 :                :     /*
                                183                 :                :      * Tell sampling function that we finished the scan.
                                184                 :                :      */
 3937 tgl@sss.pgh.pa.us         185         [ -  + ]:            174 :     if (node->tsmroutine->EndSampleScan)
 3937 tgl@sss.pgh.pa.us         186                 :UBC           0 :         node->tsmroutine->EndSampleScan(node);
                                187                 :                : 
                                188                 :                :     /*
                                189                 :                :      * close heap scan
                                190                 :                :      */
 3937 tgl@sss.pgh.pa.us         191         [ +  + ]:CBC         174 :     if (node->ss.ss_currentScanDesc)
 2612 andres@anarazel.de        192                 :             94 :         table_endscan(node->ss.ss_currentScanDesc);
 4008 simon@2ndQuadrant.co      193                 :            174 : }
                                194                 :                : 
                                195                 :                : /* ----------------------------------------------------------------
                                196                 :                :  *      ExecReScanSampleScan
                                197                 :                :  *
                                198                 :                :  *      Rescans the relation.
                                199                 :                :  *
                                200                 :                :  * ----------------------------------------------------------------
                                201                 :                :  */
                                202                 :                : void
                                203                 :             37 : ExecReScanSampleScan(SampleScanState *node)
                                204                 :                : {
                                205                 :                :     /* Remember we need to do BeginSampleScan again (if we did it at all) */
 3937 tgl@sss.pgh.pa.us         206                 :             37 :     node->begun = false;
 2593 andres@anarazel.de        207                 :             37 :     node->done = false;
                                208                 :             37 :     node->haveblock = false;
                                209                 :             37 :     node->donetuples = 0;
                                210                 :                : 
 3937 tgl@sss.pgh.pa.us         211                 :             37 :     ExecScanReScan(&node->ss);
                                212                 :             37 : }
                                213                 :                : 
                                214                 :                : 
                                215                 :                : /*
                                216                 :                :  * Initialize the TABLESAMPLE method: evaluate params and call BeginSampleScan.
                                217                 :                :  */
                                218                 :                : static void
                                219                 :            139 : tablesample_init(SampleScanState *scanstate)
                                220                 :                : {
                                221                 :            139 :     TsmRoutine *tsm = scanstate->tsmroutine;
                                222                 :            139 :     ExprContext *econtext = scanstate->ss.ps.ps_ExprContext;
                                223                 :                :     Datum      *params;
                                224                 :                :     Datum       datum;
                                225                 :                :     bool        isnull;
                                226                 :                :     uint32      seed;
                                227                 :                :     bool        allow_sync;
                                228                 :                :     int         i;
                                229                 :                :     ListCell   *arg;
                                230                 :                : 
 2593 andres@anarazel.de        231                 :            139 :     scanstate->donetuples = 0;
  146 michael@paquier.xyz       232                 :GNC         139 :     params = palloc_array(Datum, list_length(scanstate->args));
                                233                 :                : 
 3937 tgl@sss.pgh.pa.us         234                 :CBC         139 :     i = 0;
                                235   [ +  -  +  +  :            274 :     foreach(arg, scanstate->args)
                                              +  + ]
                                236                 :                :     {
                                237                 :            139 :         ExprState  *argstate = (ExprState *) lfirst(arg);
                                238                 :                : 
                                239                 :            139 :         params[i] = ExecEvalExprSwitchContext(argstate,
                                240                 :                :                                               econtext,
                                241                 :                :                                               &isnull);
                                242         [ +  + ]:            139 :         if (isnull)
                                243         [ +  - ]:              4 :             ereport(ERROR,
                                244                 :                :                     (errcode(ERRCODE_INVALID_TABLESAMPLE_ARGUMENT),
                                245                 :                :                      errmsg("TABLESAMPLE parameter cannot be null")));
                                246                 :            135 :         i++;
                                247                 :                :     }
                                248                 :                : 
                                249         [ +  + ]:            135 :     if (scanstate->repeatable)
                                250                 :                :     {
                                251                 :             48 :         datum = ExecEvalExprSwitchContext(scanstate->repeatable,
                                252                 :                :                                           econtext,
                                253                 :                :                                           &isnull);
                                254         [ +  + ]:             48 :         if (isnull)
                                255         [ +  - ]:              4 :             ereport(ERROR,
                                256                 :                :                     (errcode(ERRCODE_INVALID_TABLESAMPLE_REPEAT),
                                257                 :                :                      errmsg("TABLESAMPLE REPEATABLE parameter cannot be null")));
                                258                 :                : 
                                259                 :                :         /*
                                260                 :                :          * The REPEATABLE parameter has been coerced to float8 by the parser.
                                261                 :                :          * The reason for using float8 at the SQL level is that it will
                                262                 :                :          * produce unsurprising results both for users used to databases that
                                263                 :                :          * accept only integers in the REPEATABLE clause and for those who
                                264                 :                :          * might expect that REPEATABLE works like setseed() (a float in the
                                265                 :                :          * range from -1 to 1).
                                266                 :                :          *
                                267                 :                :          * We use hashfloat8() to convert the supplied value into a suitable
                                268                 :                :          * seed.  For regression-testing purposes, that has the convenient
                                269                 :                :          * property that REPEATABLE(0) gives a machine-independent result.
                                270                 :                :          */
                                271                 :             44 :         seed = DatumGetUInt32(DirectFunctionCall1(hashfloat8, datum));
                                272                 :                :     }
                                273                 :                :     else
                                274                 :                :     {
                                275                 :                :         /* Use the seed selected by ExecInitSampleScan */
                                276                 :             87 :         seed = scanstate->seed;
                                277                 :                :     }
                                278                 :                : 
                                279                 :                :     /* Set default values for params that BeginSampleScan can adjust */
                                280                 :            131 :     scanstate->use_bulkread = true;
                                281                 :            131 :     scanstate->use_pagemode = true;
                                282                 :                : 
                                283                 :                :     /* Let tablesample method do its thing */
                                284                 :            131 :     tsm->BeginSampleScan(scanstate,
                                285                 :                :                          params,
                                286                 :            131 :                          list_length(scanstate->args),
                                287                 :                :                          seed);
                                288                 :                : 
                                289                 :                :     /* We'll use syncscan if there's no NextSampleBlock function */
                                290                 :            113 :     allow_sync = (tsm->NextSampleBlock == NULL);
                                291                 :                : 
                                292                 :                :     /* Now we can create or reset the HeapScanDesc */
                                293         [ +  + ]:            113 :     if (scanstate->ss.ss_currentScanDesc == NULL)
                                294                 :                :     {
                                295                 :             94 :         scanstate->ss.ss_currentScanDesc =
 2612 andres@anarazel.de        296         [ +  - ]:             94 :             table_beginscan_sampling(scanstate->ss.ss_currentRelation,
                                297                 :             94 :                                      scanstate->ss.ps.state->es_snapshot,
                                298                 :                :                                      0, NULL,
                                299                 :             94 :                                      scanstate->use_bulkread,
                                300                 :                :                                      allow_sync,
   36 melanieplageman@gmai      301                 :GNC          94 :                                      scanstate->use_pagemode,
                                302                 :             94 :                                      ScanRelIsReadOnly(&scanstate->ss) ?
                                303                 :                :                                      SO_HINT_REL_READ_ONLY : SO_NONE);
                                304                 :                :     }
                                305                 :                :     else
                                306                 :                :     {
 2612 andres@anarazel.de        307                 :CBC          19 :         table_rescan_set_params(scanstate->ss.ss_currentScanDesc, NULL,
                                308                 :             19 :                                 scanstate->use_bulkread,
                                309                 :                :                                 allow_sync,
                                310                 :             19 :                                 scanstate->use_pagemode);
                                311                 :                :     }
                                312                 :                : 
 3937 tgl@sss.pgh.pa.us         313                 :            113 :     pfree(params);
                                314                 :                : 
                                315                 :                :     /* And we're initialized. */
                                316                 :            113 :     scanstate->begun = true;
                                317                 :            113 : }
                                318                 :                : 
                                319                 :                : /*
                                320                 :                :  * Get next tuple from TABLESAMPLE method.
                                321                 :                :  */
                                322                 :                : static TupleTableSlot *
                                323                 :         160817 : tablesample_getnext(SampleScanState *scanstate)
                                324                 :                : {
 2612 andres@anarazel.de        325                 :         160817 :     TableScanDesc scan = scanstate->ss.ss_currentScanDesc;
 2593                           326                 :         160817 :     TupleTableSlot *slot = scanstate->ss.ss_ScanTupleSlot;
                                327                 :                : 
                                328                 :         160817 :     ExecClearTuple(slot);
                                329                 :                : 
                                330         [ +  - ]:         160817 :     if (scanstate->done)
 2593 andres@anarazel.de        331                 :UBC           0 :         return NULL;
                                332                 :                : 
                                333                 :                :     for (;;)
                                334                 :                :     {
 2593 andres@anarazel.de        335         [ +  + ]:CBC      169289 :         if (!scanstate->haveblock)
                                336                 :                :         {
                                337         [ +  + ]:           8585 :             if (!table_scan_sample_next_block(scan, scanstate))
                                338                 :                :             {
                                339                 :            109 :                 scanstate->haveblock = false;
                                340                 :            109 :                 scanstate->done = true;
                                341                 :                : 
                                342                 :                :                 /* exhausted relation */
                                343                 :            109 :                 return NULL;
                                344                 :                :             }
                                345                 :                : 
                                346                 :           8476 :             scanstate->haveblock = true;
                                347                 :                :         }
                                348                 :                : 
                                349         [ +  + ]:         169180 :         if (!table_scan_sample_next_tuple(scan, scanstate, slot))
                                350                 :                :         {
                                351                 :                :             /*
                                352                 :                :              * If we get here, it means we've exhausted the items on this page
                                353                 :                :              * and it's time to move to the next.
                                354                 :                :              */
                                355                 :           8472 :             scanstate->haveblock = false;
                                356                 :           8472 :             continue;
                                357                 :                :         }
                                358                 :                : 
                                359                 :                :         /* Found visible tuple, return it. */
                                360                 :         160708 :         break;
                                361                 :                :     }
                                362                 :                : 
                                363                 :         160708 :     scanstate->donetuples++;
                                364                 :                : 
                                365                 :         160708 :     return slot;
                                366                 :                : }
        

Generated by: LCOV version 2.5.0-beta