LCOV - code coverage report
Current view: top level - src/backend/executor - nodeUnique.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 34 40 85.0 %
Date: 2025-01-28 23:15:34 Functions: 3 4 75.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * nodeUnique.c
       4             :  *    Routines to handle unique'ing of queries where appropriate
       5             :  *
       6             :  * Unique is a very simple node type that just filters out duplicate
       7             :  * tuples from a stream of sorted tuples from its subplan.  It's essentially
       8             :  * a dumbed-down form of Group: the duplicate-removal functionality is
       9             :  * identical.  However, Unique doesn't do projection nor qual checking,
      10             :  * so it's marginally more efficient for cases where neither is needed.
      11             :  * (It's debatable whether the savings justifies carrying two plan node
      12             :  * types, though.)
      13             :  *
      14             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
      15             :  * Portions Copyright (c) 1994, Regents of the University of California
      16             :  *
      17             :  *
      18             :  * IDENTIFICATION
      19             :  *    src/backend/executor/nodeUnique.c
      20             :  *
      21             :  *-------------------------------------------------------------------------
      22             :  */
      23             : /*
      24             :  * INTERFACE ROUTINES
      25             :  *      ExecUnique      - generate a unique'd temporary relation
      26             :  *      ExecInitUnique  - initialize node and subnodes
      27             :  *      ExecEndUnique   - shutdown node and subnodes
      28             :  *
      29             :  * NOTES
      30             :  *      Assumes tuples returned from subplan arrive in
      31             :  *      sorted order.
      32             :  */
      33             : 
      34             : #include "postgres.h"
      35             : 
      36             : #include "executor/executor.h"
      37             : #include "executor/nodeUnique.h"
      38             : #include "miscadmin.h"
      39             : 
      40             : 
      41             : /* ----------------------------------------------------------------
      42             :  *      ExecUnique
      43             :  * ----------------------------------------------------------------
      44             :  */
      45             : static TupleTableSlot *         /* return: a tuple or NULL */
      46       93006 : ExecUnique(PlanState *pstate)
      47             : {
      48       93006 :     UniqueState *node = castNode(UniqueState, pstate);
      49       93006 :     ExprContext *econtext = node->ps.ps_ExprContext;
      50             :     TupleTableSlot *resultTupleSlot;
      51             :     TupleTableSlot *slot;
      52             :     PlanState  *outerPlan;
      53             : 
      54       93006 :     CHECK_FOR_INTERRUPTS();
      55             : 
      56             :     /*
      57             :      * get information from the node
      58             :      */
      59       93006 :     outerPlan = outerPlanState(node);
      60       93006 :     resultTupleSlot = node->ps.ps_ResultTupleSlot;
      61             : 
      62             :     /*
      63             :      * now loop, returning only non-duplicate tuples. We assume that the
      64             :      * tuples arrive in sorted order so we can detect duplicates easily. The
      65             :      * first tuple of each group is returned.
      66             :      */
      67             :     for (;;)
      68             :     {
      69             :         /*
      70             :          * fetch a tuple from the outer subplan
      71             :          */
      72      463336 :         slot = ExecProcNode(outerPlan);
      73      463336 :         if (TupIsNull(slot))
      74             :         {
      75             :             /* end of subplan, so we're done */
      76        4720 :             ExecClearTuple(resultTupleSlot);
      77        4720 :             return NULL;
      78             :         }
      79             : 
      80             :         /*
      81             :          * Always return the first tuple from the subplan.
      82             :          */
      83      458616 :         if (TupIsNull(resultTupleSlot))
      84             :             break;
      85             : 
      86             :         /*
      87             :          * Else test if the new tuple and the previously returned tuple match.
      88             :          * If so then we loop back and fetch another new tuple from the
      89             :          * subplan.
      90             :          */
      91      456716 :         econtext->ecxt_innertuple = slot;
      92      456716 :         econtext->ecxt_outertuple = resultTupleSlot;
      93      456716 :         if (!ExecQualAndReset(node->eqfunction, econtext))
      94       86386 :             break;
      95             :     }
      96             : 
      97             :     /*
      98             :      * We have a new tuple different from the previous saved tuple (if any).
      99             :      * Save it and return it.  We must copy it because the source subplan
     100             :      * won't guarantee that this source tuple is still accessible after
     101             :      * fetching the next source tuple.
     102             :      */
     103       88286 :     return ExecCopySlot(resultTupleSlot, slot);
     104             : }
     105             : 
     106             : /* ----------------------------------------------------------------
     107             :  *      ExecInitUnique
     108             :  *
     109             :  *      This initializes the unique node state structures and
     110             :  *      the node's subplan.
     111             :  * ----------------------------------------------------------------
     112             :  */
     113             : UniqueState *
     114        5062 : ExecInitUnique(Unique *node, EState *estate, int eflags)
     115             : {
     116             :     UniqueState *uniquestate;
     117             : 
     118             :     /* check for unsupported flags */
     119             :     Assert(!(eflags & (EXEC_FLAG_BACKWARD | EXEC_FLAG_MARK)));
     120             : 
     121             :     /*
     122             :      * create state structure
     123             :      */
     124        5062 :     uniquestate = makeNode(UniqueState);
     125        5062 :     uniquestate->ps.plan = (Plan *) node;
     126        5062 :     uniquestate->ps.state = estate;
     127        5062 :     uniquestate->ps.ExecProcNode = ExecUnique;
     128             : 
     129             :     /*
     130             :      * create expression context
     131             :      */
     132        5062 :     ExecAssignExprContext(estate, &uniquestate->ps);
     133             : 
     134             :     /*
     135             :      * then initialize outer plan
     136             :      */
     137        5062 :     outerPlanState(uniquestate) = ExecInitNode(outerPlan(node), estate, eflags);
     138             : 
     139             :     /*
     140             :      * Initialize result slot and type. Unique nodes do no projections, so
     141             :      * initialize projection info for this node appropriately.
     142             :      */
     143        5062 :     ExecInitResultTupleSlotTL(&uniquestate->ps, &TTSOpsMinimalTuple);
     144        5062 :     uniquestate->ps.ps_ProjInfo = NULL;
     145             : 
     146             :     /*
     147             :      * Precompute fmgr lookup data for inner loop
     148             :      */
     149        5062 :     uniquestate->eqfunction =
     150        5062 :         execTuplesMatchPrepare(ExecGetResultType(outerPlanState(uniquestate)),
     151             :                                node->numCols,
     152        5062 :                                node->uniqColIdx,
     153        5062 :                                node->uniqOperators,
     154        5062 :                                node->uniqCollations,
     155             :                                &uniquestate->ps);
     156             : 
     157        5062 :     return uniquestate;
     158             : }
     159             : 
     160             : /* ----------------------------------------------------------------
     161             :  *      ExecEndUnique
     162             :  *
     163             :  *      This shuts down the subplan and frees resources allocated
     164             :  *      to this node.
     165             :  * ----------------------------------------------------------------
     166             :  */
     167             : void
     168        5062 : ExecEndUnique(UniqueState *node)
     169             : {
     170        5062 :     ExecEndNode(outerPlanState(node));
     171        5062 : }
     172             : 
     173             : 
     174             : void
     175           0 : ExecReScanUnique(UniqueState *node)
     176             : {
     177           0 :     PlanState  *outerPlan = outerPlanState(node);
     178             : 
     179             :     /* must clear result tuple so first input tuple is returned */
     180           0 :     ExecClearTuple(node->ps.ps_ResultTupleSlot);
     181             : 
     182             :     /*
     183             :      * if chgParam of subnode is not null then plan will be re-scanned by
     184             :      * first ExecProcNode.
     185             :      */
     186           0 :     if (outerPlan->chgParam == NULL)
     187           0 :         ExecReScan(outerPlan);
     188           0 : }

Generated by: LCOV version 1.14