LCOV - code coverage report
Current view: top level - src/backend/executor - nodeLockRows.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 105 125 84.0 %
Date: 2025-01-18 04:15:08 Functions: 4 4 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * nodeLockRows.c
       4             :  *    Routines to handle FOR UPDATE/FOR SHARE row locking
       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/nodeLockRows.c
      12             :  *
      13             :  *-------------------------------------------------------------------------
      14             :  */
      15             : /*
      16             :  * INTERFACE ROUTINES
      17             :  *      ExecLockRows        - fetch locked rows
      18             :  *      ExecInitLockRows    - initialize node and subnodes..
      19             :  *      ExecEndLockRows     - shutdown node and subnodes
      20             :  */
      21             : 
      22             : #include "postgres.h"
      23             : 
      24             : #include "access/tableam.h"
      25             : #include "access/xact.h"
      26             : #include "executor/executor.h"
      27             : #include "executor/nodeLockRows.h"
      28             : #include "foreign/fdwapi.h"
      29             : #include "miscadmin.h"
      30             : #include "utils/rel.h"
      31             : 
      32             : 
      33             : /* ----------------------------------------------------------------
      34             :  *      ExecLockRows
      35             :  * ----------------------------------------------------------------
      36             :  */
      37             : static TupleTableSlot *         /* return: a tuple or NULL */
      38       19746 : ExecLockRows(PlanState *pstate)
      39             : {
      40       19746 :     LockRowsState *node = castNode(LockRowsState, pstate);
      41             :     TupleTableSlot *slot;
      42             :     EState     *estate;
      43             :     PlanState  *outerPlan;
      44             :     bool        epq_needed;
      45             :     ListCell   *lc;
      46             : 
      47       19746 :     CHECK_FOR_INTERRUPTS();
      48             : 
      49             :     /*
      50             :      * get information from the node
      51             :      */
      52       19746 :     estate = node->ps.state;
      53       19746 :     outerPlan = outerPlanState(node);
      54             : 
      55             :     /*
      56             :      * Get next tuple from subplan, if any.
      57             :      */
      58       19850 : lnext:
      59       19850 :     slot = ExecProcNode(outerPlan);
      60             : 
      61       19850 :     if (TupIsNull(slot))
      62             :     {
      63             :         /* Release any resources held by EPQ mechanism before exiting */
      64        3360 :         EvalPlanQualEnd(&node->lr_epqstate);
      65        3360 :         return NULL;
      66             :     }
      67             : 
      68             :     /* We don't need EvalPlanQual unless we get updated tuple version(s) */
      69       16490 :     epq_needed = false;
      70             : 
      71             :     /*
      72             :      * Attempt to lock the source tuple(s).  (Note we only have locking
      73             :      * rowmarks in lr_arowMarks.)
      74             :      */
      75       33728 :     foreach(lc, node->lr_arowMarks)
      76             :     {
      77       17384 :         ExecAuxRowMark *aerm = (ExecAuxRowMark *) lfirst(lc);
      78       17384 :         ExecRowMark *erm = aerm->rowmark;
      79             :         Datum       datum;
      80             :         bool        isNull;
      81             :         ItemPointerData tid;
      82             :         TM_FailureData tmfd;
      83             :         LockTupleMode lockmode;
      84       17384 :         int         lockflags = 0;
      85             :         TM_Result   test;
      86             :         TupleTableSlot *markSlot;
      87             : 
      88             :         /* clear any leftover test tuple for this rel */
      89       17384 :         markSlot = EvalPlanQualSlot(&node->lr_epqstate, erm->relation, erm->rti);
      90       17384 :         ExecClearTuple(markSlot);
      91             : 
      92             :         /* if child rel, must check whether it produced this row */
      93       17384 :         if (erm->rti != erm->prti)
      94             :         {
      95             :             Oid         tableoid;
      96             : 
      97        1592 :             datum = ExecGetJunkAttribute(slot,
      98        1592 :                                          aerm->toidAttNo,
      99             :                                          &isNull);
     100             :             /* shouldn't ever get a null result... */
     101        1592 :             if (isNull)
     102           0 :                 elog(ERROR, "tableoid is NULL");
     103        1592 :             tableoid = DatumGetObjectId(datum);
     104             : 
     105             :             Assert(OidIsValid(erm->relid));
     106        1592 :             if (tableoid != erm->relid)
     107             :             {
     108             :                 /* this child is inactive right now */
     109         400 :                 erm->ermActive = false;
     110         400 :                 ItemPointerSetInvalid(&(erm->curCtid));
     111         400 :                 continue;
     112             :             }
     113             :         }
     114       16984 :         erm->ermActive = true;
     115             : 
     116             :         /* fetch the tuple's ctid */
     117       16984 :         datum = ExecGetJunkAttribute(slot,
     118       16984 :                                      aerm->ctidAttNo,
     119             :                                      &isNull);
     120             :         /* shouldn't ever get a null result... */
     121       16984 :         if (isNull)
     122           0 :             elog(ERROR, "ctid is NULL");
     123             : 
     124             :         /* requests for foreign tables must be passed to their FDW */
     125       16984 :         if (erm->relation->rd_rel->relkind == RELKIND_FOREIGN_TABLE)
     126             :         {
     127             :             FdwRoutine *fdwroutine;
     128           0 :             bool        updated = false;
     129             : 
     130           0 :             fdwroutine = GetFdwRoutineForRelation(erm->relation, false);
     131             :             /* this should have been checked already, but let's be safe */
     132           0 :             if (fdwroutine->RefetchForeignRow == NULL)
     133           0 :                 ereport(ERROR,
     134             :                         (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
     135             :                          errmsg("cannot lock rows in foreign table \"%s\"",
     136             :                                 RelationGetRelationName(erm->relation))));
     137             : 
     138           0 :             fdwroutine->RefetchForeignRow(estate,
     139             :                                           erm,
     140             :                                           datum,
     141             :                                           markSlot,
     142             :                                           &updated);
     143           0 :             if (TupIsNull(markSlot))
     144             :             {
     145             :                 /* couldn't get the lock, so skip this row */
     146           0 :                 goto lnext;
     147             :             }
     148             : 
     149             :             /*
     150             :              * if FDW says tuple was updated before getting locked, we need to
     151             :              * perform EPQ testing to see if quals are still satisfied
     152             :              */
     153           0 :             if (updated)
     154           0 :                 epq_needed = true;
     155             : 
     156           0 :             continue;
     157             :         }
     158             : 
     159             :         /* okay, try to lock (and fetch) the tuple */
     160       16984 :         tid = *((ItemPointer) DatumGetPointer(datum));
     161       16984 :         switch (erm->markType)
     162             :         {
     163        7938 :             case ROW_MARK_EXCLUSIVE:
     164        7938 :                 lockmode = LockTupleExclusive;
     165        7938 :                 break;
     166          72 :             case ROW_MARK_NOKEYEXCLUSIVE:
     167          72 :                 lockmode = LockTupleNoKeyExclusive;
     168          72 :                 break;
     169        2512 :             case ROW_MARK_SHARE:
     170        2512 :                 lockmode = LockTupleShare;
     171        2512 :                 break;
     172        6462 :             case ROW_MARK_KEYSHARE:
     173        6462 :                 lockmode = LockTupleKeyShare;
     174        6462 :                 break;
     175           0 :             default:
     176           0 :                 elog(ERROR, "unsupported rowmark type");
     177             :                 lockmode = LockTupleNoKeyExclusive; /* keep compiler quiet */
     178             :                 break;
     179             :         }
     180             : 
     181       16984 :         lockflags = TUPLE_LOCK_FLAG_LOCK_UPDATE_IN_PROGRESS;
     182       16984 :         if (!IsolationUsesXactSnapshot())
     183        9242 :             lockflags |= TUPLE_LOCK_FLAG_FIND_LAST_VERSION;
     184             : 
     185       16984 :         test = table_tuple_lock(erm->relation, &tid, estate->es_snapshot,
     186             :                                 markSlot, estate->es_output_cid,
     187             :                                 lockmode, erm->waitPolicy,
     188             :                                 lockflags,
     189             :                                 &tmfd);
     190             : 
     191       16958 :         switch (test)
     192             :         {
     193          74 :             case TM_WouldBlock:
     194             :                 /* couldn't lock tuple in SKIP LOCKED mode */
     195          74 :                 goto lnext;
     196             : 
     197           6 :             case TM_SelfModified:
     198             : 
     199             :                 /*
     200             :                  * The target tuple was already updated or deleted by the
     201             :                  * current command, or by a later command in the current
     202             :                  * transaction.  We *must* ignore the tuple in the former
     203             :                  * case, so as to avoid the "Halloween problem" of repeated
     204             :                  * update attempts.  In the latter case it might be sensible
     205             :                  * to fetch the updated tuple instead, but doing so would
     206             :                  * require changing heap_update and heap_delete to not
     207             :                  * complain about updating "invisible" tuples, which seems
     208             :                  * pretty scary (table_tuple_lock will not complain, but few
     209             :                  * callers expect TM_Invisible, and we're not one of them). So
     210             :                  * for now, treat the tuple as deleted and do not process.
     211             :                  */
     212           6 :                 goto lnext;
     213             : 
     214       16838 :             case TM_Ok:
     215             : 
     216             :                 /*
     217             :                  * Got the lock successfully, the locked tuple saved in
     218             :                  * markSlot for, if needed, EvalPlanQual testing below.
     219             :                  */
     220       16838 :                 if (tmfd.traversed)
     221          58 :                     epq_needed = true;
     222       16838 :                 break;
     223             : 
     224          24 :             case TM_Updated:
     225          24 :                 if (IsolationUsesXactSnapshot())
     226          24 :                     ereport(ERROR,
     227             :                             (errcode(ERRCODE_T_R_SERIALIZATION_FAILURE),
     228             :                              errmsg("could not serialize access due to concurrent update")));
     229           0 :                 elog(ERROR, "unexpected table_tuple_lock status: %u",
     230             :                      test);
     231             :                 break;
     232             : 
     233          16 :             case TM_Deleted:
     234          16 :                 if (IsolationUsesXactSnapshot())
     235           4 :                     ereport(ERROR,
     236             :                             (errcode(ERRCODE_T_R_SERIALIZATION_FAILURE),
     237             :                              errmsg("could not serialize access due to concurrent update")));
     238             :                 /* tuple was deleted so don't return it */
     239          12 :                 goto lnext;
     240             : 
     241           0 :             case TM_Invisible:
     242           0 :                 elog(ERROR, "attempted to lock invisible tuple");
     243             :                 break;
     244             : 
     245           0 :             default:
     246           0 :                 elog(ERROR, "unrecognized table_tuple_lock status: %u",
     247             :                      test);
     248             :         }
     249             : 
     250             :         /* Remember locked tuple's TID for EPQ testing and WHERE CURRENT OF */
     251       16838 :         erm->curCtid = tid;
     252             :     }
     253             : 
     254             :     /*
     255             :      * If we need to do EvalPlanQual testing, do so.
     256             :      */
     257       16344 :     if (epq_needed)
     258             :     {
     259             :         /* Initialize EPQ machinery */
     260          56 :         EvalPlanQualBegin(&node->lr_epqstate);
     261             : 
     262             :         /*
     263             :          * To fetch non-locked source rows the EPQ logic needs to access junk
     264             :          * columns from the tuple being tested.
     265             :          */
     266          56 :         EvalPlanQualSetSlot(&node->lr_epqstate, slot);
     267             : 
     268             :         /*
     269             :          * And finally we can re-evaluate the tuple.
     270             :          */
     271          56 :         slot = EvalPlanQualNext(&node->lr_epqstate);
     272          56 :         if (TupIsNull(slot))
     273             :         {
     274             :             /* Updated tuple fails qual, so ignore it and go on */
     275          12 :             goto lnext;
     276             :         }
     277             :     }
     278             : 
     279             :     /* Got all locks, so return the current tuple */
     280       16332 :     return slot;
     281             : }
     282             : 
     283             : /* ----------------------------------------------------------------
     284             :  *      ExecInitLockRows
     285             :  *
     286             :  *      This initializes the LockRows node state structures and
     287             :  *      the node's subplan.
     288             :  * ----------------------------------------------------------------
     289             :  */
     290             : LockRowsState *
     291        8214 : ExecInitLockRows(LockRows *node, EState *estate, int eflags)
     292             : {
     293             :     LockRowsState *lrstate;
     294        8214 :     Plan       *outerPlan = outerPlan(node);
     295             :     List       *epq_arowmarks;
     296             :     ListCell   *lc;
     297             : 
     298             :     /* check for unsupported flags */
     299             :     Assert(!(eflags & EXEC_FLAG_MARK));
     300             : 
     301             :     /*
     302             :      * create state structure
     303             :      */
     304        8214 :     lrstate = makeNode(LockRowsState);
     305        8214 :     lrstate->ps.plan = (Plan *) node;
     306        8214 :     lrstate->ps.state = estate;
     307        8214 :     lrstate->ps.ExecProcNode = ExecLockRows;
     308             : 
     309             :     /*
     310             :      * Miscellaneous initialization
     311             :      *
     312             :      * LockRows nodes never call ExecQual or ExecProject, therefore no
     313             :      * ExprContext is needed.
     314             :      */
     315             : 
     316             :     /*
     317             :      * Initialize result type.
     318             :      */
     319        8214 :     ExecInitResultTypeTL(&lrstate->ps);
     320             : 
     321             :     /*
     322             :      * then initialize outer plan
     323             :      */
     324        8214 :     outerPlanState(lrstate) = ExecInitNode(outerPlan, estate, eflags);
     325             : 
     326             :     /* node returns unmodified slots from the outer plan */
     327        8214 :     lrstate->ps.resultopsset = true;
     328        8214 :     lrstate->ps.resultops = ExecGetResultSlotOps(outerPlanState(lrstate),
     329             :                                                  &lrstate->ps.resultopsfixed);
     330             : 
     331             :     /*
     332             :      * LockRows nodes do no projections, so initialize projection info for
     333             :      * this node appropriately
     334             :      */
     335        8214 :     lrstate->ps.ps_ProjInfo = NULL;
     336             : 
     337             :     /*
     338             :      * Locate the ExecRowMark(s) that this node is responsible for, and
     339             :      * construct ExecAuxRowMarks for them.  (InitPlan should already have
     340             :      * built the global list of ExecRowMarks.)
     341             :      */
     342        8214 :     lrstate->lr_arowMarks = NIL;
     343        8214 :     epq_arowmarks = NIL;
     344       18854 :     foreach(lc, node->rowMarks)
     345             :     {
     346       10640 :         PlanRowMark *rc = lfirst_node(PlanRowMark, lc);
     347             :         ExecRowMark *erm;
     348             :         ExecAuxRowMark *aerm;
     349             : 
     350             :         /* ignore "parent" rowmarks; they are irrelevant at runtime */
     351       10640 :         if (rc->isParent)
     352        1780 :             continue;
     353             : 
     354             :         /* find ExecRowMark and build ExecAuxRowMark */
     355        8860 :         erm = ExecFindRowMark(estate, rc->rti, false);
     356        8860 :         aerm = ExecBuildAuxRowMark(erm, outerPlan->targetlist);
     357             : 
     358             :         /*
     359             :          * Only locking rowmarks go into our own list.  Non-locking marks are
     360             :          * passed off to the EvalPlanQual machinery.  This is because we don't
     361             :          * want to bother fetching non-locked rows unless we actually have to
     362             :          * do an EPQ recheck.
     363             :          */
     364        8860 :         if (RowMarkRequiresRowShareLock(erm->markType))
     365        8564 :             lrstate->lr_arowMarks = lappend(lrstate->lr_arowMarks, aerm);
     366             :         else
     367         296 :             epq_arowmarks = lappend(epq_arowmarks, aerm);
     368             :     }
     369             : 
     370             :     /* Now we have the info needed to set up EPQ state */
     371        8214 :     EvalPlanQualInit(&lrstate->lr_epqstate, estate,
     372             :                      outerPlan, epq_arowmarks, node->epqParam, NIL);
     373             : 
     374        8214 :     return lrstate;
     375             : }
     376             : 
     377             : /* ----------------------------------------------------------------
     378             :  *      ExecEndLockRows
     379             :  *
     380             :  *      This shuts down the subplan and frees resources allocated
     381             :  *      to this node.
     382             :  * ----------------------------------------------------------------
     383             :  */
     384             : void
     385        8128 : ExecEndLockRows(LockRowsState *node)
     386             : {
     387             :     /* We may have shut down EPQ already, but no harm in another call */
     388        8128 :     EvalPlanQualEnd(&node->lr_epqstate);
     389        8128 :     ExecEndNode(outerPlanState(node));
     390        8128 : }
     391             : 
     392             : 
     393             : void
     394          16 : ExecReScanLockRows(LockRowsState *node)
     395             : {
     396          16 :     PlanState  *outerPlan = outerPlanState(node);
     397             : 
     398             :     /*
     399             :      * if chgParam of subnode is not null then plan will be re-scanned by
     400             :      * first ExecProcNode.
     401             :      */
     402          16 :     if (outerPlan->chgParam == NULL)
     403           0 :         ExecReScan(outerPlan);
     404          16 : }

Generated by: LCOV version 1.14