LCOV - code coverage report
Current view: top level - src/backend/access/common - tupconvert.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 90 90 100.0 %
Date: 2025-01-18 04:15:08 Functions: 7 7 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * tupconvert.c
       4             :  *    Tuple conversion support.
       5             :  *
       6             :  * These functions provide conversion between rowtypes that are logically
       7             :  * equivalent but might have columns in a different order or different sets of
       8             :  * dropped columns.
       9             :  *
      10             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
      11             :  * Portions Copyright (c) 1994, Regents of the University of California
      12             :  *
      13             :  *
      14             :  * IDENTIFICATION
      15             :  *    src/backend/access/common/tupconvert.c
      16             :  *
      17             :  *-------------------------------------------------------------------------
      18             :  */
      19             : #include "postgres.h"
      20             : 
      21             : #include "access/tupconvert.h"
      22             : #include "executor/tuptable.h"
      23             : 
      24             : 
      25             : /*
      26             :  * The conversion setup routines have the following common API:
      27             :  *
      28             :  * The setup routine checks using attmap.c whether the given source and
      29             :  * destination tuple descriptors are logically compatible.  If not, it throws
      30             :  * an error.  If so, it returns NULL if they are physically compatible (ie, no
      31             :  * conversion is needed), else a TupleConversionMap that can be used by
      32             :  * execute_attr_map_tuple or execute_attr_map_slot to perform the conversion.
      33             :  *
      34             :  * The TupleConversionMap, if needed, is palloc'd in the caller's memory
      35             :  * context.  Also, the given tuple descriptors are referenced by the map,
      36             :  * so they must survive as long as the map is needed.
      37             :  *
      38             :  * The caller must supply a suitable primary error message to be used if
      39             :  * a compatibility error is thrown.  Recommended coding practice is to use
      40             :  * gettext_noop() on this string, so that it is translatable but won't
      41             :  * actually be translated unless the error gets thrown.
      42             :  *
      43             :  *
      44             :  * Implementation notes:
      45             :  *
      46             :  * The key component of a TupleConversionMap is an attrMap[] array with
      47             :  * one entry per output column.  This entry contains the 1-based index of
      48             :  * the corresponding input column, or zero to force a NULL value (for
      49             :  * a dropped output column).  The TupleConversionMap also contains workspace
      50             :  * arrays.
      51             :  */
      52             : 
      53             : 
      54             : /*
      55             :  * Set up for tuple conversion, matching input and output columns by
      56             :  * position.  (Dropped columns are ignored in both input and output.)
      57             :  */
      58             : TupleConversionMap *
      59        9524 : convert_tuples_by_position(TupleDesc indesc,
      60             :                            TupleDesc outdesc,
      61             :                            const char *msg)
      62             : {
      63             :     TupleConversionMap *map;
      64             :     int         n;
      65             :     AttrMap    *attrMap;
      66             : 
      67             :     /* Verify compatibility and prepare attribute-number map */
      68        9524 :     attrMap = build_attrmap_by_position(indesc, outdesc, msg);
      69             : 
      70        9492 :     if (attrMap == NULL)
      71             :     {
      72             :         /* runtime conversion is not needed */
      73        9398 :         return NULL;
      74             :     }
      75             : 
      76             :     /* Prepare the map structure */
      77          94 :     map = (TupleConversionMap *) palloc(sizeof(TupleConversionMap));
      78          94 :     map->indesc = indesc;
      79          94 :     map->outdesc = outdesc;
      80          94 :     map->attrMap = attrMap;
      81             :     /* preallocate workspace for Datum arrays */
      82          94 :     n = outdesc->natts + 1;      /* +1 for NULL */
      83          94 :     map->outvalues = (Datum *) palloc(n * sizeof(Datum));
      84          94 :     map->outisnull = (bool *) palloc(n * sizeof(bool));
      85          94 :     n = indesc->natts + 1;       /* +1 for NULL */
      86          94 :     map->invalues = (Datum *) palloc(n * sizeof(Datum));
      87          94 :     map->inisnull = (bool *) palloc(n * sizeof(bool));
      88          94 :     map->invalues[0] = (Datum) 0;    /* set up the NULL entry */
      89          94 :     map->inisnull[0] = true;
      90             : 
      91          94 :     return map;
      92             : }
      93             : 
      94             : /*
      95             :  * Set up for tuple conversion, matching input and output columns by name.
      96             :  * (Dropped columns are ignored in both input and output.)  This is intended
      97             :  * for use when the rowtypes are related by inheritance, so we expect an exact
      98             :  * match of both type and typmod.  The error messages will be a bit unhelpful
      99             :  * unless both rowtypes are named composite types.
     100             :  */
     101             : TupleConversionMap *
     102        3580 : convert_tuples_by_name(TupleDesc indesc,
     103             :                        TupleDesc outdesc)
     104             : {
     105             :     AttrMap    *attrMap;
     106             : 
     107             :     /* Verify compatibility and prepare attribute-number map */
     108        3580 :     attrMap = build_attrmap_by_name_if_req(indesc, outdesc, false);
     109             : 
     110        3580 :     if (attrMap == NULL)
     111             :     {
     112             :         /* runtime conversion is not needed */
     113        2760 :         return NULL;
     114             :     }
     115             : 
     116         820 :     return convert_tuples_by_name_attrmap(indesc, outdesc, attrMap);
     117             : }
     118             : 
     119             : /*
     120             :  * Set up tuple conversion for input and output TupleDescs using the given
     121             :  * AttrMap.
     122             :  */
     123             : TupleConversionMap *
     124        2272 : convert_tuples_by_name_attrmap(TupleDesc indesc,
     125             :                                TupleDesc outdesc,
     126             :                                AttrMap *attrMap)
     127             : {
     128        2272 :     int         n = outdesc->natts;
     129             :     TupleConversionMap *map;
     130             : 
     131             :     Assert(attrMap != NULL);
     132             : 
     133             :     /* Prepare the map structure */
     134        2272 :     map = (TupleConversionMap *) palloc(sizeof(TupleConversionMap));
     135        2272 :     map->indesc = indesc;
     136        2272 :     map->outdesc = outdesc;
     137        2272 :     map->attrMap = attrMap;
     138             :     /* preallocate workspace for Datum arrays */
     139        2272 :     map->outvalues = (Datum *) palloc(n * sizeof(Datum));
     140        2272 :     map->outisnull = (bool *) palloc(n * sizeof(bool));
     141        2272 :     n = indesc->natts + 1;       /* +1 for NULL */
     142        2272 :     map->invalues = (Datum *) palloc(n * sizeof(Datum));
     143        2272 :     map->inisnull = (bool *) palloc(n * sizeof(bool));
     144        2272 :     map->invalues[0] = (Datum) 0;    /* set up the NULL entry */
     145        2272 :     map->inisnull[0] = true;
     146             : 
     147        2272 :     return map;
     148             : }
     149             : 
     150             : /*
     151             :  * Perform conversion of a tuple according to the map.
     152             :  */
     153             : HeapTuple
     154      107106 : execute_attr_map_tuple(HeapTuple tuple, TupleConversionMap *map)
     155             : {
     156      107106 :     AttrMap    *attrMap = map->attrMap;
     157      107106 :     Datum      *invalues = map->invalues;
     158      107106 :     bool       *inisnull = map->inisnull;
     159      107106 :     Datum      *outvalues = map->outvalues;
     160      107106 :     bool       *outisnull = map->outisnull;
     161             :     int         i;
     162             : 
     163             :     /*
     164             :      * Extract all the values of the old tuple, offsetting the arrays so that
     165             :      * invalues[0] is left NULL and invalues[1] is the first source attribute;
     166             :      * this exactly matches the numbering convention in attrMap.
     167             :      */
     168      107106 :     heap_deform_tuple(tuple, map->indesc, invalues + 1, inisnull + 1);
     169             : 
     170             :     /*
     171             :      * Transpose into proper fields of the new tuple.
     172             :      */
     173             :     Assert(attrMap->maplen == map->outdesc->natts);
     174      426482 :     for (i = 0; i < attrMap->maplen; i++)
     175             :     {
     176      319376 :         int         j = attrMap->attnums[i];
     177             : 
     178      319376 :         outvalues[i] = invalues[j];
     179      319376 :         outisnull[i] = inisnull[j];
     180             :     }
     181             : 
     182             :     /*
     183             :      * Now form the new tuple.
     184             :      */
     185      107106 :     return heap_form_tuple(map->outdesc, outvalues, outisnull);
     186             : }
     187             : 
     188             : /*
     189             :  * Perform conversion of a tuple slot according to the map.
     190             :  */
     191             : TupleTableSlot *
     192      143700 : execute_attr_map_slot(AttrMap *attrMap,
     193             :                       TupleTableSlot *in_slot,
     194             :                       TupleTableSlot *out_slot)
     195             : {
     196             :     Datum      *invalues;
     197             :     bool       *inisnull;
     198             :     Datum      *outvalues;
     199             :     bool       *outisnull;
     200             :     int         outnatts;
     201             :     int         i;
     202             : 
     203             :     /* Sanity checks */
     204             :     Assert(in_slot->tts_tupleDescriptor != NULL &&
     205             :            out_slot->tts_tupleDescriptor != NULL);
     206             :     Assert(in_slot->tts_values != NULL && out_slot->tts_values != NULL);
     207             : 
     208      143700 :     outnatts = out_slot->tts_tupleDescriptor->natts;
     209             : 
     210             :     /* Extract all the values of the in slot. */
     211      143700 :     slot_getallattrs(in_slot);
     212             : 
     213             :     /* Before doing the mapping, clear any old contents from the out slot */
     214      143700 :     ExecClearTuple(out_slot);
     215             : 
     216      143700 :     invalues = in_slot->tts_values;
     217      143700 :     inisnull = in_slot->tts_isnull;
     218      143700 :     outvalues = out_slot->tts_values;
     219      143700 :     outisnull = out_slot->tts_isnull;
     220             : 
     221             :     /* Transpose into proper fields of the out slot. */
     222      579510 :     for (i = 0; i < outnatts; i++)
     223             :     {
     224      435810 :         int         j = attrMap->attnums[i] - 1;
     225             : 
     226             :         /* attrMap->attnums[i] == 0 means it's a NULL datum. */
     227      435810 :         if (j == -1)
     228             :         {
     229        2600 :             outvalues[i] = (Datum) 0;
     230        2600 :             outisnull[i] = true;
     231             :         }
     232             :         else
     233             :         {
     234      433210 :             outvalues[i] = invalues[j];
     235      433210 :             outisnull[i] = inisnull[j];
     236             :         }
     237             :     }
     238             : 
     239      143700 :     ExecStoreVirtualTuple(out_slot);
     240             : 
     241      143700 :     return out_slot;
     242             : }
     243             : 
     244             : /*
     245             :  * Perform conversion of bitmap of columns according to the map.
     246             :  *
     247             :  * The input and output bitmaps are offset by
     248             :  * FirstLowInvalidHeapAttributeNumber to accommodate system cols, like the
     249             :  * column-bitmaps in RangeTblEntry.
     250             :  */
     251             : Bitmapset *
     252         494 : execute_attr_map_cols(AttrMap *attrMap, Bitmapset *in_cols)
     253             : {
     254             :     Bitmapset  *out_cols;
     255             :     int         out_attnum;
     256             : 
     257             :     /* fast path for the common trivial case */
     258         494 :     if (in_cols == NULL)
     259           4 :         return NULL;
     260             : 
     261             :     /*
     262             :      * For each output column, check which input column it corresponds to.
     263             :      */
     264         490 :     out_cols = NULL;
     265             : 
     266         490 :     for (out_attnum = FirstLowInvalidHeapAttributeNumber;
     267        6248 :          out_attnum <= attrMap->maplen;
     268        5758 :          out_attnum++)
     269             :     {
     270             :         int         in_attnum;
     271             : 
     272        5758 :         if (out_attnum < 0)
     273             :         {
     274             :             /* System column. No mapping. */
     275        3430 :             in_attnum = out_attnum;
     276             :         }
     277        2328 :         else if (out_attnum == 0)
     278         490 :             continue;
     279             :         else
     280             :         {
     281             :             /* normal user column */
     282        1838 :             in_attnum = attrMap->attnums[out_attnum - 1];
     283             : 
     284        1838 :             if (in_attnum == 0)
     285         142 :                 continue;
     286             :         }
     287             : 
     288        5126 :         if (bms_is_member(in_attnum - FirstLowInvalidHeapAttributeNumber, in_cols))
     289         532 :             out_cols = bms_add_member(out_cols, out_attnum - FirstLowInvalidHeapAttributeNumber);
     290             :     }
     291             : 
     292         490 :     return out_cols;
     293             : }
     294             : 
     295             : /*
     296             :  * Free a TupleConversionMap structure.
     297             :  */
     298             : void
     299         168 : free_conversion_map(TupleConversionMap *map)
     300             : {
     301             :     /* indesc and outdesc are not ours to free */
     302         168 :     free_attrmap(map->attrMap);
     303         168 :     pfree(map->invalues);
     304         168 :     pfree(map->inisnull);
     305         168 :     pfree(map->outvalues);
     306         168 :     pfree(map->outisnull);
     307         168 :     pfree(map);
     308         168 : }

Generated by: LCOV version 1.14