LCOV - code coverage report
Current view: top level - src/backend/executor - execCurrent.c (source / functions) Coverage Total Hit
Test: Code coverage Lines: 84.3 % 127 107
Test Date: 2026-01-26 10:56:24 Functions: 100.0 % 3 3
Legend: Lines:     hit not hit
Branches: + taken - not taken # not executed
Branches: 54.6 % 141 77

             Branch data     Line data    Source code
       1                 :             : /*-------------------------------------------------------------------------
       2                 :             :  *
       3                 :             :  * execCurrent.c
       4                 :             :  *        executor support for WHERE CURRENT OF cursor
       5                 :             :  *
       6                 :             :  * Portions Copyright (c) 1996-2026, PostgreSQL Global Development Group
       7                 :             :  * Portions Copyright (c) 1994, Regents of the University of California
       8                 :             :  *
       9                 :             :  *      src/backend/executor/execCurrent.c
      10                 :             :  *
      11                 :             :  *-------------------------------------------------------------------------
      12                 :             :  */
      13                 :             : #include "postgres.h"
      14                 :             : 
      15                 :             : #include "access/genam.h"
      16                 :             : #include "access/relscan.h"
      17                 :             : #include "access/sysattr.h"
      18                 :             : #include "catalog/pg_type.h"
      19                 :             : #include "executor/executor.h"
      20                 :             : #include "utils/builtins.h"
      21                 :             : #include "utils/lsyscache.h"
      22                 :             : #include "utils/portal.h"
      23                 :             : #include "utils/rel.h"
      24                 :             : 
      25                 :             : 
      26                 :             : static char *fetch_cursor_param_value(ExprContext *econtext, int paramId);
      27                 :             : static ScanState *search_plan_tree(PlanState *node, Oid table_oid,
      28                 :             :                                                                    bool *pending_rescan);
      29                 :             : 
      30                 :             : 
      31                 :             : /*
      32                 :             :  * execCurrentOf
      33                 :             :  *
      34                 :             :  * Given a CURRENT OF expression and the OID of a table, determine which row
      35                 :             :  * of the table is currently being scanned by the cursor named by CURRENT OF,
      36                 :             :  * and return the row's TID into *current_tid.
      37                 :             :  *
      38                 :             :  * Returns true if a row was identified.  Returns false if the cursor is valid
      39                 :             :  * for the table but is not currently scanning a row of the table (this is a
      40                 :             :  * legal situation in inheritance cases).  Raises error if cursor is not a
      41                 :             :  * valid updatable scan of the specified table.
      42                 :             :  */
      43                 :             : bool
      44                 :          64 : execCurrentOf(CurrentOfExpr *cexpr,
      45                 :             :                           ExprContext *econtext,
      46                 :             :                           Oid table_oid,
      47                 :             :                           ItemPointer current_tid)
      48                 :             : {
      49                 :          64 :         char       *cursor_name;
      50                 :          64 :         char       *table_name;
      51                 :          64 :         Portal          portal;
      52                 :          64 :         QueryDesc  *queryDesc;
      53                 :             : 
      54                 :             :         /* Get the cursor name --- may have to look up a parameter reference */
      55         [ +  + ]:          64 :         if (cexpr->cursor_name)
      56                 :          44 :                 cursor_name = cexpr->cursor_name;
      57                 :             :         else
      58                 :          20 :                 cursor_name = fetch_cursor_param_value(econtext, cexpr->cursor_param);
      59                 :             : 
      60                 :             :         /* Fetch table name for possible use in error messages */
      61                 :          64 :         table_name = get_rel_name(table_oid);
      62         [ +  - ]:          64 :         if (table_name == NULL)
      63   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for relation %u", table_oid);
      64                 :             : 
      65                 :             :         /* Find the cursor's portal */
      66                 :          64 :         portal = GetPortalByName(cursor_name);
      67         [ +  + ]:          64 :         if (!PortalIsValid(portal))
      68   [ +  -  +  - ]:           1 :                 ereport(ERROR,
      69                 :             :                                 (errcode(ERRCODE_UNDEFINED_CURSOR),
      70                 :             :                                  errmsg("cursor \"%s\" does not exist", cursor_name)));
      71                 :             : 
      72                 :             :         /*
      73                 :             :          * We have to watch out for non-SELECT queries as well as held cursors,
      74                 :             :          * both of which may have null queryDesc.
      75                 :             :          */
      76         [ +  - ]:          63 :         if (portal->strategy != PORTAL_ONE_SELECT)
      77   [ #  #  #  # ]:           0 :                 ereport(ERROR,
      78                 :             :                                 (errcode(ERRCODE_INVALID_CURSOR_STATE),
      79                 :             :                                  errmsg("cursor \"%s\" is not a SELECT query",
      80                 :             :                                                 cursor_name)));
      81                 :          63 :         queryDesc = portal->queryDesc;
      82         [ +  + ]:          63 :         if (queryDesc == NULL || queryDesc->estate == NULL)
      83   [ +  -  +  - ]:           1 :                 ereport(ERROR,
      84                 :             :                                 (errcode(ERRCODE_INVALID_CURSOR_STATE),
      85                 :             :                                  errmsg("cursor \"%s\" is held from a previous transaction",
      86                 :             :                                                 cursor_name)));
      87                 :             : 
      88                 :             :         /*
      89                 :             :          * We have two different strategies depending on whether the cursor uses
      90                 :             :          * FOR UPDATE/SHARE or not.  The reason for supporting both is that the
      91                 :             :          * FOR UPDATE code is able to identify a target table in many cases where
      92                 :             :          * the other code can't, while the non-FOR-UPDATE case allows use of WHERE
      93                 :             :          * CURRENT OF with an insensitive cursor.
      94                 :             :          */
      95         [ +  + ]:          62 :         if (queryDesc->estate->es_rowmarks)
      96                 :             :         {
      97                 :          16 :                 ExecRowMark *erm;
      98                 :          16 :                 Index           i;
      99                 :             : 
     100                 :             :                 /*
     101                 :             :                  * Here, the query must have exactly one FOR UPDATE/SHARE reference to
     102                 :             :                  * the target table, and we dig the ctid info out of that.
     103                 :             :                  */
     104                 :          16 :                 erm = NULL;
     105         [ +  + ]:          57 :                 for (i = 0; i < queryDesc->estate->es_range_table_size; i++)
     106                 :             :                 {
     107                 :          42 :                         ExecRowMark *thiserm = queryDesc->estate->es_rowmarks[i];
     108                 :             : 
     109   [ +  +  +  + ]:          42 :                         if (thiserm == NULL ||
     110                 :          30 :                                 !RowMarkRequiresRowShareLock(thiserm->markType))
     111                 :          16 :                                 continue;               /* ignore non-FOR UPDATE/SHARE items */
     112                 :             : 
     113         [ +  + ]:          26 :                         if (thiserm->relid == table_oid)
     114                 :             :                         {
     115         [ +  + ]:          16 :                                 if (erm)
     116   [ +  -  +  - ]:           1 :                                         ereport(ERROR,
     117                 :             :                                                         (errcode(ERRCODE_INVALID_CURSOR_STATE),
     118                 :             :                                                          errmsg("cursor \"%s\" has multiple FOR UPDATE/SHARE references to table \"%s\"",
     119                 :             :                                                                         cursor_name, table_name)));
     120                 :          15 :                                 erm = thiserm;
     121                 :          15 :                         }
     122      [ -  +  + ]:          41 :                 }
     123                 :             : 
     124         [ +  + ]:          15 :                 if (erm == NULL)
     125   [ +  -  +  - ]:           1 :                         ereport(ERROR,
     126                 :             :                                         (errcode(ERRCODE_INVALID_CURSOR_STATE),
     127                 :             :                                          errmsg("cursor \"%s\" does not have a FOR UPDATE/SHARE reference to table \"%s\"",
     128                 :             :                                                         cursor_name, table_name)));
     129                 :             : 
     130                 :             :                 /*
     131                 :             :                  * The cursor must have a current result row: per the SQL spec, it's
     132                 :             :                  * an error if not.
     133                 :             :                  */
     134         [ +  - ]:          14 :                 if (portal->atStart || portal->atEnd)
     135   [ #  #  #  # ]:           0 :                         ereport(ERROR,
     136                 :             :                                         (errcode(ERRCODE_INVALID_CURSOR_STATE),
     137                 :             :                                          errmsg("cursor \"%s\" is not positioned on a row",
     138                 :             :                                                         cursor_name)));
     139                 :             : 
     140                 :             :                 /* Return the currently scanned TID, if there is one */
     141         [ +  + ]:          14 :                 if (ItemPointerIsValid(&(erm->curCtid)))
     142                 :             :                 {
     143                 :          10 :                         *current_tid = erm->curCtid;
     144                 :          10 :                         return true;
     145                 :             :                 }
     146                 :             : 
     147                 :             :                 /*
     148                 :             :                  * This table didn't produce the cursor's current row; some other
     149                 :             :                  * inheritance child of the same parent must have.  Signal caller to
     150                 :             :                  * do nothing on this table.
     151                 :             :                  */
     152                 :           4 :                 return false;
     153                 :          14 :         }
     154                 :             :         else
     155                 :             :         {
     156                 :             :                 /*
     157                 :             :                  * Without FOR UPDATE, we dig through the cursor's plan to find the
     158                 :             :                  * scan node.  Fail if it's not there or buried underneath
     159                 :             :                  * aggregation.
     160                 :             :                  */
     161                 :          46 :                 ScanState  *scanstate;
     162                 :          46 :                 bool            pending_rescan = false;
     163                 :             : 
     164                 :          46 :                 scanstate = search_plan_tree(queryDesc->planstate, table_oid,
     165                 :             :                                                                          &pending_rescan);
     166         [ +  + ]:          46 :                 if (!scanstate)
     167   [ +  -  +  - ]:           4 :                         ereport(ERROR,
     168                 :             :                                         (errcode(ERRCODE_INVALID_CURSOR_STATE),
     169                 :             :                                          errmsg("cursor \"%s\" is not a simply updatable scan of table \"%s\"",
     170                 :             :                                                         cursor_name, table_name)));
     171                 :             : 
     172                 :             :                 /*
     173                 :             :                  * The cursor must have a current result row: per the SQL spec, it's
     174                 :             :                  * an error if not.  We test this at the top level, rather than at the
     175                 :             :                  * scan node level, because in inheritance cases any one table scan
     176                 :             :                  * could easily not be on a row. We want to return false, not raise
     177                 :             :                  * error, if the passed-in table OID is for one of the inactive scans.
     178                 :             :                  */
     179         [ +  + ]:          42 :                 if (portal->atStart || portal->atEnd)
     180   [ +  -  +  - ]:           2 :                         ereport(ERROR,
     181                 :             :                                         (errcode(ERRCODE_INVALID_CURSOR_STATE),
     182                 :             :                                          errmsg("cursor \"%s\" is not positioned on a row",
     183                 :             :                                                         cursor_name)));
     184                 :             : 
     185                 :             :                 /*
     186                 :             :                  * Now OK to return false if we found an inactive scan.  It is
     187                 :             :                  * inactive either if it's not positioned on a row, or there's a
     188                 :             :                  * rescan pending for it.
     189                 :             :                  */
     190   [ +  -  +  +  :          40 :                 if (TupIsNull(scanstate->ss_ScanTupleSlot) || pending_rescan)
                   -  + ]
     191                 :           4 :                         return false;
     192                 :             : 
     193                 :             :                 /*
     194                 :             :                  * Extract TID of the scan's current row.  The mechanism for this is
     195                 :             :                  * in principle scan-type-dependent, but for most scan types, we can
     196                 :             :                  * just dig the TID out of the physical scan tuple.
     197                 :             :                  */
     198         [ +  + ]:          36 :                 if (IsA(scanstate, IndexOnlyScanState))
     199                 :             :                 {
     200                 :             :                         /*
     201                 :             :                          * For IndexOnlyScan, the tuple stored in ss_ScanTupleSlot may be
     202                 :             :                          * a virtual tuple that does not have the ctid column, so we have
     203                 :             :                          * to get the TID from xs_heaptid.
     204                 :             :                          */
     205                 :           1 :                         IndexScanDesc scan = ((IndexOnlyScanState *) scanstate)->ioss_ScanDesc;
     206                 :             : 
     207                 :           1 :                         *current_tid = scan->xs_heaptid;
     208                 :           1 :                 }
     209                 :             :                 else
     210                 :             :                 {
     211                 :             :                         /*
     212                 :             :                          * Default case: try to fetch TID from the scan node's current
     213                 :             :                          * tuple.  As an extra cross-check, verify tableoid in the current
     214                 :             :                          * tuple.  If the scan hasn't provided a physical tuple, we have
     215                 :             :                          * to fail.
     216                 :             :                          */
     217                 :          35 :                         Datum           ldatum;
     218                 :          35 :                         bool            lisnull;
     219                 :          35 :                         ItemPointer tuple_tid;
     220                 :             : 
     221                 :             : #ifdef USE_ASSERT_CHECKING
     222                 :          35 :                         ldatum = slot_getsysattr(scanstate->ss_ScanTupleSlot,
     223                 :             :                                                                          TableOidAttributeNumber,
     224                 :             :                                                                          &lisnull);
     225         [ +  - ]:          35 :                         if (lisnull)
     226   [ #  #  #  # ]:           0 :                                 ereport(ERROR,
     227                 :             :                                                 (errcode(ERRCODE_INVALID_CURSOR_STATE),
     228                 :             :                                                  errmsg("cursor \"%s\" is not a simply updatable scan of table \"%s\"",
     229                 :             :                                                                 cursor_name, table_name)));
     230         [ +  - ]:          35 :                         Assert(DatumGetObjectId(ldatum) == table_oid);
     231                 :             : #endif
     232                 :             : 
     233                 :          35 :                         ldatum = slot_getsysattr(scanstate->ss_ScanTupleSlot,
     234                 :             :                                                                          SelfItemPointerAttributeNumber,
     235                 :             :                                                                          &lisnull);
     236         [ +  - ]:          35 :                         if (lisnull)
     237   [ #  #  #  # ]:           0 :                                 ereport(ERROR,
     238                 :             :                                                 (errcode(ERRCODE_INVALID_CURSOR_STATE),
     239                 :             :                                                  errmsg("cursor \"%s\" is not a simply updatable scan of table \"%s\"",
     240                 :             :                                                                 cursor_name, table_name)));
     241                 :          35 :                         tuple_tid = (ItemPointer) DatumGetPointer(ldatum);
     242                 :             : 
     243                 :          35 :                         *current_tid = *tuple_tid;
     244                 :          35 :                 }
     245                 :             : 
     246         [ +  - ]:          36 :                 Assert(ItemPointerIsValid(current_tid));
     247                 :             : 
     248                 :          36 :                 return true;
     249                 :          40 :         }
     250                 :          54 : }
     251                 :             : 
     252                 :             : /*
     253                 :             :  * fetch_cursor_param_value
     254                 :             :  *
     255                 :             :  * Fetch the string value of a param, verifying it is of type REFCURSOR.
     256                 :             :  */
     257                 :             : static char *
     258                 :          20 : fetch_cursor_param_value(ExprContext *econtext, int paramId)
     259                 :             : {
     260                 :          20 :         ParamListInfo paramInfo = econtext->ecxt_param_list_info;
     261                 :             : 
     262         [ +  - ]:          20 :         if (paramInfo &&
     263                 :          20 :                 paramId > 0 && paramId <= paramInfo->numParams)
     264                 :             :         {
     265                 :          20 :                 ParamExternData *prm;
     266                 :          20 :                 ParamExternData prmdata;
     267                 :             : 
     268                 :             :                 /* give hook a chance in case parameter is dynamic */
     269         [ +  - ]:          20 :                 if (paramInfo->paramFetch != NULL)
     270                 :          20 :                         prm = paramInfo->paramFetch(paramInfo, paramId, false, &prmdata);
     271                 :             :                 else
     272                 :           0 :                         prm = &paramInfo->params[paramId - 1];
     273                 :             : 
     274   [ +  -  -  + ]:          20 :                 if (OidIsValid(prm->ptype) && !prm->isnull)
     275                 :             :                 {
     276                 :             :                         /* safety check in case hook did something unexpected */
     277         [ +  - ]:          20 :                         if (prm->ptype != REFCURSOROID)
     278   [ #  #  #  # ]:           0 :                                 ereport(ERROR,
     279                 :             :                                                 (errcode(ERRCODE_DATATYPE_MISMATCH),
     280                 :             :                                                  errmsg("type of parameter %d (%s) does not match that when preparing the plan (%s)",
     281                 :             :                                                                 paramId,
     282                 :             :                                                                 format_type_be(prm->ptype),
     283                 :             :                                                                 format_type_be(REFCURSOROID))));
     284                 :             : 
     285                 :             :                         /* We know that refcursor uses text's I/O routines */
     286                 :          20 :                         return TextDatumGetCString(prm->value);
     287                 :             :                 }
     288         [ +  - ]:          20 :         }
     289                 :             : 
     290   [ #  #  #  # ]:           0 :         ereport(ERROR,
     291                 :             :                         (errcode(ERRCODE_UNDEFINED_OBJECT),
     292                 :             :                          errmsg("no value found for parameter %d", paramId)));
     293                 :           0 :         return NULL;
     294                 :          20 : }
     295                 :             : 
     296                 :             : /*
     297                 :             :  * search_plan_tree
     298                 :             :  *
     299                 :             :  * Search through a PlanState tree for a scan node on the specified table.
     300                 :             :  * Return NULL if not found or multiple candidates.
     301                 :             :  *
     302                 :             :  * CAUTION: this function is not charged simply with finding some candidate
     303                 :             :  * scan, but with ensuring that that scan returned the plan tree's current
     304                 :             :  * output row.  That's why we must reject multiple-match cases.
     305                 :             :  *
     306                 :             :  * If a candidate is found, set *pending_rescan to true if that candidate
     307                 :             :  * or any node above it has a pending rescan action, i.e. chgParam != NULL.
     308                 :             :  * That indicates that we shouldn't consider the node to be positioned on a
     309                 :             :  * valid tuple, even if its own state would indicate that it is.  (Caller
     310                 :             :  * must initialize *pending_rescan to false, and should not trust its state
     311                 :             :  * if multiple candidates are found.)
     312                 :             :  */
     313                 :             : static ScanState *
     314                 :          66 : search_plan_tree(PlanState *node, Oid table_oid,
     315                 :             :                                  bool *pending_rescan)
     316                 :             : {
     317                 :          66 :         ScanState  *result = NULL;
     318                 :             : 
     319         [ +  - ]:          66 :         if (node == NULL)
     320                 :           0 :                 return NULL;
     321   [ +  +  -  +  :          66 :         switch (nodeTag(node))
                      - ]
     322                 :             :         {
     323                 :             :                         /*
     324                 :             :                          * Relation scan nodes can all be treated alike: check to see if
     325                 :             :                          * they are scanning the specified table.
     326                 :             :                          *
     327                 :             :                          * ForeignScan and CustomScan might not have a currentRelation, in
     328                 :             :                          * which case we just ignore them.  (We dare not descend to any
     329                 :             :                          * child plan nodes they might have, since we do not know the
     330                 :             :                          * relationship of such a node's current output tuple to the
     331                 :             :                          * children's current outputs.)
     332                 :             :                          */
     333                 :             :                 case T_SeqScanState:
     334                 :             :                 case T_SampleScanState:
     335                 :             :                 case T_IndexScanState:
     336                 :             :                 case T_IndexOnlyScanState:
     337                 :             :                 case T_BitmapHeapScanState:
     338                 :             :                 case T_TidScanState:
     339                 :             :                 case T_TidRangeScanState:
     340                 :             :                 case T_ForeignScanState:
     341                 :             :                 case T_CustomScanState:
     342                 :             :                         {
     343                 :          56 :                                 ScanState  *sstate = (ScanState *) node;
     344                 :             : 
     345   [ +  -  +  + ]:          56 :                                 if (sstate->ss_currentRelation &&
     346                 :          56 :                                         RelationGetRelid(sstate->ss_currentRelation) == table_oid)
     347                 :          42 :                                         result = sstate;
     348                 :             :                                 break;
     349                 :          56 :                         }
     350                 :             : 
     351                 :             :                         /*
     352                 :             :                          * For Append, we can check each input node.  It is safe to
     353                 :             :                          * descend to the inputs because only the input that resulted in
     354                 :             :                          * the Append's current output node could be positioned on a tuple
     355                 :             :                          * at all; the other inputs are either at EOF or not yet started.
     356                 :             :                          * Hence, if the desired table is scanned by some
     357                 :             :                          * currently-inactive input node, we will find that node but then
     358                 :             :                          * our caller will realize that it didn't emit the tuple of
     359                 :             :                          * interest.
     360                 :             :                          *
     361                 :             :                          * We do need to watch out for multiple matches (possible if
     362                 :             :                          * Append was from UNION ALL rather than an inheritance tree).
     363                 :             :                          *
     364                 :             :                          * Note: we can NOT descend through MergeAppend similarly, since
     365                 :             :                          * its inputs are likely all active, and we don't know which one
     366                 :             :                          * returned the current output tuple.  (Perhaps that could be
     367                 :             :                          * fixed if we were to let this code know more about MergeAppend's
     368                 :             :                          * internal state, but it does not seem worth the trouble.  Users
     369                 :             :                          * should not expect plans for ORDER BY queries to be considered
     370                 :             :                          * simply-updatable, since they won't be if the sorting is
     371                 :             :                          * implemented by a Sort node.)
     372                 :             :                          */
     373                 :             :                 case T_AppendState:
     374                 :             :                         {
     375                 :           7 :                                 AppendState *astate = (AppendState *) node;
     376                 :           7 :                                 int                     i;
     377                 :             : 
     378         [ +  + ]:          27 :                                 for (i = 0; i < astate->as_nplans; i++)
     379                 :             :                                 {
     380                 :          40 :                                         ScanState  *elem = search_plan_tree(astate->appendplans[i],
     381                 :          20 :                                                                                                                 table_oid,
     382                 :          20 :                                                                                                                 pending_rescan);
     383                 :             : 
     384         [ +  + ]:          20 :                                         if (!elem)
     385                 :          13 :                                                 continue;
     386         [ -  + ]:           7 :                                         if (result)
     387                 :           0 :                                                 return NULL;    /* multiple matches */
     388                 :           7 :                                         result = elem;
     389      [ +  -  + ]:          20 :                                 }
     390                 :           7 :                                 break;
     391         [ -  + ]:           7 :                         }
     392                 :             : 
     393                 :             :                         /*
     394                 :             :                          * Result and Limit can be descended through (these are safe
     395                 :             :                          * because they always return their input's current row)
     396                 :             :                          */
     397                 :             :                 case T_ResultState:
     398                 :             :                 case T_LimitState:
     399                 :           0 :                         result = search_plan_tree(outerPlanState(node),
     400                 :           0 :                                                                           table_oid,
     401                 :           0 :                                                                           pending_rescan);
     402                 :           0 :                         break;
     403                 :             : 
     404                 :             :                         /*
     405                 :             :                          * SubqueryScan too, but it keeps the child in a different place
     406                 :             :                          */
     407                 :             :                 case T_SubqueryScanState:
     408                 :           0 :                         result = search_plan_tree(((SubqueryScanState *) node)->subplan,
     409                 :           0 :                                                                           table_oid,
     410                 :           0 :                                                                           pending_rescan);
     411                 :           0 :                         break;
     412                 :             : 
     413                 :             :                 default:
     414                 :             :                         /* Otherwise, assume we can't descend through it */
     415                 :           3 :                         break;
     416                 :             :         }
     417                 :             : 
     418                 :             :         /*
     419                 :             :          * If we found a candidate at or below this node, then this node's
     420                 :             :          * chgParam indicates a pending rescan that will affect the candidate.
     421                 :             :          */
     422   [ +  +  +  - ]:          66 :         if (result && node->chgParam != NULL)
     423                 :           0 :                 *pending_rescan = true;
     424                 :             : 
     425                 :          66 :         return result;
     426                 :          66 : }
        

Generated by: LCOV version 2.3.2-1