LCOV - code coverage report
Current view: top level - src/backend/executor - nodeWorktablescan.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 37 39 94.9 %
Date: 2025-01-18 04:15:08 Functions: 4 5 80.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * nodeWorktablescan.c
       4             :  *    routines to handle WorkTableScan nodes.
       5             :  *
       6             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       7             :  * Portions Copyright (c) 1994, Regents of the University of California
       8             :  *
       9             :  *
      10             :  * IDENTIFICATION
      11             :  *    src/backend/executor/nodeWorktablescan.c
      12             :  *
      13             :  *-------------------------------------------------------------------------
      14             :  */
      15             : 
      16             : #include "postgres.h"
      17             : 
      18             : #include "executor/executor.h"
      19             : #include "executor/nodeWorktablescan.h"
      20             : 
      21             : static TupleTableSlot *WorkTableScanNext(WorkTableScanState *node);
      22             : 
      23             : /* ----------------------------------------------------------------
      24             :  *      WorkTableScanNext
      25             :  *
      26             :  *      This is a workhorse for ExecWorkTableScan
      27             :  * ----------------------------------------------------------------
      28             :  */
      29             : static TupleTableSlot *
      30       50892 : WorkTableScanNext(WorkTableScanState *node)
      31             : {
      32             :     TupleTableSlot *slot;
      33             :     Tuplestorestate *tuplestorestate;
      34             : 
      35             :     /*
      36             :      * get information from the estate and scan state
      37             :      *
      38             :      * Note: we intentionally do not support backward scan.  Although it would
      39             :      * take only a couple more lines here, it would force nodeRecursiveunion.c
      40             :      * to create the tuplestore with backward scan enabled, which has a
      41             :      * performance cost.  In practice backward scan is never useful for a
      42             :      * worktable plan node, since it cannot appear high enough in the plan
      43             :      * tree of a scrollable cursor to be exposed to a backward-scan
      44             :      * requirement.  So it's not worth expending effort to support it.
      45             :      *
      46             :      * Note: we are also assuming that this node is the only reader of the
      47             :      * worktable.  Therefore, we don't need a private read pointer for the
      48             :      * tuplestore, nor do we need to tell tuplestore_gettupleslot to copy.
      49             :      */
      50             :     Assert(ScanDirectionIsForward(node->ss.ps.state->es_direction));
      51             : 
      52       50892 :     tuplestorestate = node->rustate->working_table;
      53             : 
      54             :     /*
      55             :      * Get the next tuple from tuplestore. Return NULL if no more tuples.
      56             :      */
      57       50892 :     slot = node->ss.ss_ScanTupleSlot;
      58       50892 :     (void) tuplestore_gettupleslot(tuplestorestate, true, false, slot);
      59       50892 :     return slot;
      60             : }
      61             : 
      62             : /*
      63             :  * WorkTableScanRecheck -- access method routine to recheck a tuple in EvalPlanQual
      64             :  */
      65             : static bool
      66           0 : WorkTableScanRecheck(WorkTableScanState *node, TupleTableSlot *slot)
      67             : {
      68             :     /* nothing to check */
      69           0 :     return true;
      70             : }
      71             : 
      72             : /* ----------------------------------------------------------------
      73             :  *      ExecWorkTableScan(node)
      74             :  *
      75             :  *      Scans the worktable sequentially and returns the next qualifying tuple.
      76             :  *      We call the ExecScan() routine and pass it the appropriate
      77             :  *      access method functions.
      78             :  * ----------------------------------------------------------------
      79             :  */
      80             : static TupleTableSlot *
      81       36812 : ExecWorkTableScan(PlanState *pstate)
      82             : {
      83       36812 :     WorkTableScanState *node = castNode(WorkTableScanState, pstate);
      84             : 
      85             :     /*
      86             :      * On the first call, find the ancestor RecursiveUnion's state via the
      87             :      * Param slot reserved for it.  (We can't do this during node init because
      88             :      * there are corner cases where we'll get the init call before the
      89             :      * RecursiveUnion does.)
      90             :      */
      91       36812 :     if (node->rustate == NULL)
      92             :     {
      93         762 :         WorkTableScan *plan = (WorkTableScan *) node->ss.ps.plan;
      94         762 :         EState     *estate = node->ss.ps.state;
      95             :         ParamExecData *param;
      96             : 
      97         762 :         param = &(estate->es_param_exec_vals[plan->wtParam]);
      98             :         Assert(param->execPlan == NULL);
      99             :         Assert(!param->isnull);
     100         762 :         node->rustate = castNode(RecursiveUnionState, DatumGetPointer(param->value));
     101             :         Assert(node->rustate);
     102             : 
     103             :         /*
     104             :          * The scan tuple type (ie, the rowtype we expect to find in the work
     105             :          * table) is the same as the result rowtype of the ancestor
     106             :          * RecursiveUnion node.  Note this depends on the assumption that
     107             :          * RecursiveUnion doesn't allow projection.
     108             :          */
     109         762 :         ExecAssignScanType(&node->ss,
     110         762 :                            ExecGetResultType(&node->rustate->ps));
     111             : 
     112             :         /*
     113             :          * Now we can initialize the projection info.  This must be completed
     114             :          * before we can call ExecScan().
     115             :          */
     116         762 :         ExecAssignScanProjectionInfo(&node->ss);
     117             :     }
     118             : 
     119       36812 :     return ExecScan(&node->ss,
     120             :                     (ExecScanAccessMtd) WorkTableScanNext,
     121             :                     (ExecScanRecheckMtd) WorkTableScanRecheck);
     122             : }
     123             : 
     124             : 
     125             : /* ----------------------------------------------------------------
     126             :  *      ExecInitWorkTableScan
     127             :  * ----------------------------------------------------------------
     128             :  */
     129             : WorkTableScanState *
     130         816 : ExecInitWorkTableScan(WorkTableScan *node, EState *estate, int eflags)
     131             : {
     132             :     WorkTableScanState *scanstate;
     133             : 
     134             :     /* check for unsupported flags */
     135             :     Assert(!(eflags & (EXEC_FLAG_BACKWARD | EXEC_FLAG_MARK)));
     136             : 
     137             :     /*
     138             :      * WorkTableScan should not have any children.
     139             :      */
     140             :     Assert(outerPlan(node) == NULL);
     141             :     Assert(innerPlan(node) == NULL);
     142             : 
     143             :     /*
     144             :      * create new WorkTableScanState for node
     145             :      */
     146         816 :     scanstate = makeNode(WorkTableScanState);
     147         816 :     scanstate->ss.ps.plan = (Plan *) node;
     148         816 :     scanstate->ss.ps.state = estate;
     149         816 :     scanstate->ss.ps.ExecProcNode = ExecWorkTableScan;
     150         816 :     scanstate->rustate = NULL;   /* we'll set this later */
     151             : 
     152             :     /*
     153             :      * Miscellaneous initialization
     154             :      *
     155             :      * create expression context for node
     156             :      */
     157         816 :     ExecAssignExprContext(estate, &scanstate->ss.ps);
     158             : 
     159             :     /*
     160             :      * tuple table initialization
     161             :      */
     162         816 :     ExecInitResultTypeTL(&scanstate->ss.ps);
     163             : 
     164             :     /* signal that return type is not yet known */
     165         816 :     scanstate->ss.ps.resultopsset = true;
     166         816 :     scanstate->ss.ps.resultopsfixed = false;
     167             : 
     168         816 :     ExecInitScanTupleSlot(estate, &scanstate->ss, NULL, &TTSOpsMinimalTuple);
     169             : 
     170             :     /*
     171             :      * initialize child expressions
     172             :      */
     173         816 :     scanstate->ss.ps.qual =
     174         816 :         ExecInitQual(node->scan.plan.qual, (PlanState *) scanstate);
     175             : 
     176             :     /*
     177             :      * Do not yet initialize projection info, see ExecWorkTableScan() for
     178             :      * details.
     179             :      */
     180             : 
     181         816 :     return scanstate;
     182             : }
     183             : 
     184             : /* ----------------------------------------------------------------
     185             :  *      ExecReScanWorkTableScan
     186             :  *
     187             :  *      Rescans the relation.
     188             :  * ----------------------------------------------------------------
     189             :  */
     190             : void
     191        6320 : ExecReScanWorkTableScan(WorkTableScanState *node)
     192             : {
     193        6320 :     if (node->ss.ps.ps_ResultTupleSlot)
     194        5894 :         ExecClearTuple(node->ss.ps.ps_ResultTupleSlot);
     195             : 
     196        6320 :     ExecScanReScan(&node->ss);
     197             : 
     198             :     /* No need (or way) to rescan if ExecWorkTableScan not called yet */
     199        6320 :     if (node->rustate)
     200        6314 :         tuplestore_rescan(node->rustate->working_table);
     201        6320 : }

Generated by: LCOV version 1.14