LCOV - differential code coverage report
Current view: top level - src/backend/utils/adt - array_expanded.c (source / functions) Coverage Total Hit UBC CBC
Current: Differential Code Coverage HEAD vs 15 Lines: 96.1 % 154 148 6 148
Current Date: 2023-04-08 15:15:32 Functions: 100.0 % 8 8 8
Baseline: 15
Baseline Date: 2023-04-08 15:09:40
Legend: Lines: hit not hit

           TLA  Line data    Source code
       1                 : /*-------------------------------------------------------------------------
       2                 :  *
       3                 :  * array_expanded.c
       4                 :  *    Basic functions for manipulating expanded arrays.
       5                 :  *
       6                 :  * Portions Copyright (c) 1996-2023, PostgreSQL Global Development Group
       7                 :  * Portions Copyright (c) 1994, Regents of the University of California
       8                 :  *
       9                 :  *
      10                 :  * IDENTIFICATION
      11                 :  *    src/backend/utils/adt/array_expanded.c
      12                 :  *
      13                 :  *-------------------------------------------------------------------------
      14                 :  */
      15                 : #include "postgres.h"
      16                 : 
      17                 : #include "access/tupmacs.h"
      18                 : #include "utils/array.h"
      19                 : #include "utils/lsyscache.h"
      20                 : #include "utils/memutils.h"
      21                 : 
      22                 : 
      23                 : /* "Methods" required for an expanded object */
      24                 : static Size EA_get_flat_size(ExpandedObjectHeader *eohptr);
      25                 : static void EA_flatten_into(ExpandedObjectHeader *eohptr,
      26                 :                             void *result, Size allocated_size);
      27                 : 
      28                 : static const ExpandedObjectMethods EA_methods =
      29                 : {
      30                 :     EA_get_flat_size,
      31                 :     EA_flatten_into
      32                 : };
      33                 : 
      34                 : /* Other local functions */
      35                 : static void copy_byval_expanded_array(ExpandedArrayHeader *eah,
      36                 :                                       ExpandedArrayHeader *oldeah);
      37                 : 
      38                 : 
      39                 : /*
      40                 :  * expand_array: convert an array Datum into an expanded array
      41                 :  *
      42                 :  * The expanded object will be a child of parentcontext.
      43                 :  *
      44                 :  * Some callers can provide cache space to avoid repeated lookups of element
      45                 :  * type data across calls; if so, pass a metacache pointer, making sure that
      46                 :  * metacache->element_type is initialized to InvalidOid before first call.
      47                 :  * If no cross-call caching is required, pass NULL for metacache.
      48                 :  */
      49                 : Datum
      50 CBC        8202 : expand_array(Datum arraydatum, MemoryContext parentcontext,
      51                 :              ArrayMetaState *metacache)
      52                 : {
      53                 :     ArrayType  *array;
      54                 :     ExpandedArrayHeader *eah;
      55                 :     MemoryContext objcxt;
      56                 :     MemoryContext oldcxt;
      57                 :     ArrayMetaState fakecache;
      58                 : 
      59                 :     /*
      60                 :      * Allocate private context for expanded object.  We start by assuming
      61                 :      * that the array won't be very large; but if it does grow a lot, don't
      62                 :      * constrain aset.c's large-context behavior.
      63                 :      */
      64            8202 :     objcxt = AllocSetContextCreate(parentcontext,
      65                 :                                    "expanded array",
      66                 :                                    ALLOCSET_START_SMALL_SIZES);
      67                 : 
      68                 :     /* Set up expanded array header */
      69                 :     eah = (ExpandedArrayHeader *)
      70            8202 :         MemoryContextAlloc(objcxt, sizeof(ExpandedArrayHeader));
      71                 : 
      72            8202 :     EOH_init_header(&eah->hdr, &EA_methods, objcxt);
      73            8202 :     eah->ea_magic = EA_MAGIC;
      74                 : 
      75                 :     /* If the source is an expanded array, we may be able to optimize */
      76            8202 :     if (VARATT_IS_EXTERNAL_EXPANDED(DatumGetPointer(arraydatum)))
      77                 :     {
      78              50 :         ExpandedArrayHeader *oldeah = (ExpandedArrayHeader *) DatumGetEOHP(arraydatum);
      79                 : 
      80              50 :         Assert(oldeah->ea_magic == EA_MAGIC);
      81                 : 
      82                 :         /*
      83                 :          * Update caller's cache if provided; we don't need it this time, but
      84                 :          * next call might be for a non-expanded source array.  Furthermore,
      85                 :          * if the caller didn't provide a cache area, use some local storage
      86                 :          * to cache anyway, thereby avoiding a catalog lookup in the case
      87                 :          * where we fall through to the flat-copy code path.
      88                 :          */
      89              50 :         if (metacache == NULL)
      90              17 :             metacache = &fakecache;
      91              50 :         metacache->element_type = oldeah->element_type;
      92              50 :         metacache->typlen = oldeah->typlen;
      93              50 :         metacache->typbyval = oldeah->typbyval;
      94              50 :         metacache->typalign = oldeah->typalign;
      95                 : 
      96                 :         /*
      97                 :          * If element type is pass-by-value and we have a Datum-array
      98                 :          * representation, just copy the source's metadata and Datum/isnull
      99                 :          * arrays.  The original flat array, if present at all, adds no
     100                 :          * additional information so we need not copy it.
     101                 :          */
     102              50 :         if (oldeah->typbyval && oldeah->dvalues != NULL)
     103                 :         {
     104              12 :             copy_byval_expanded_array(eah, oldeah);
     105                 :             /* return a R/W pointer to the expanded array */
     106              12 :             return EOHPGetRWDatum(&eah->hdr);
     107                 :         }
     108                 : 
     109                 :         /*
     110                 :          * Otherwise, either we have only a flat representation or the
     111                 :          * elements are pass-by-reference.  In either case, the best thing
     112                 :          * seems to be to copy the source as a flat representation and then
     113                 :          * deconstruct that later if necessary.  For the pass-by-ref case, we
     114                 :          * could perhaps save some cycles with custom code that generates the
     115                 :          * deconstructed representation in parallel with copying the values,
     116                 :          * but it would be a lot of extra code for fairly marginal gain.  So,
     117                 :          * fall through into the flat-source code path.
     118                 :          */
     119                 :     }
     120                 : 
     121                 :     /*
     122                 :      * Detoast and copy source array into private context, as a flat array.
     123                 :      *
     124                 :      * Note that this coding risks leaking some memory in the private context
     125                 :      * if we have to fetch data from a TOAST table; however, experimentation
     126                 :      * says that the leak is minimal.  Doing it this way saves a copy step,
     127                 :      * which seems worthwhile, especially if the array is large enough to need
     128                 :      * external storage.
     129                 :      */
     130            8190 :     oldcxt = MemoryContextSwitchTo(objcxt);
     131            8190 :     array = DatumGetArrayTypePCopy(arraydatum);
     132            8190 :     MemoryContextSwitchTo(oldcxt);
     133                 : 
     134            8190 :     eah->ndims = ARR_NDIM(array);
     135                 :     /* note these pointers point into the fvalue header! */
     136            8190 :     eah->dims = ARR_DIMS(array);
     137            8190 :     eah->lbound = ARR_LBOUND(array);
     138                 : 
     139                 :     /* Save array's element-type data for possible use later */
     140            8190 :     eah->element_type = ARR_ELEMTYPE(array);
     141            8190 :     if (metacache && metacache->element_type == eah->element_type)
     142                 :     {
     143                 :         /* We have a valid cache of representational data */
     144             329 :         eah->typlen = metacache->typlen;
     145             329 :         eah->typbyval = metacache->typbyval;
     146             329 :         eah->typalign = metacache->typalign;
     147                 :     }
     148                 :     else
     149                 :     {
     150                 :         /* No, so look it up */
     151            7861 :         get_typlenbyvalalign(eah->element_type,
     152                 :                              &eah->typlen,
     153                 :                              &eah->typbyval,
     154                 :                              &eah->typalign);
     155                 :         /* Update cache if provided */
     156            7861 :         if (metacache)
     157                 :         {
     158             519 :             metacache->element_type = eah->element_type;
     159             519 :             metacache->typlen = eah->typlen;
     160             519 :             metacache->typbyval = eah->typbyval;
     161             519 :             metacache->typalign = eah->typalign;
     162                 :         }
     163                 :     }
     164                 : 
     165                 :     /* we don't make a deconstructed representation now */
     166            8190 :     eah->dvalues = NULL;
     167            8190 :     eah->dnulls = NULL;
     168            8190 :     eah->dvalueslen = 0;
     169            8190 :     eah->nelems = 0;
     170            8190 :     eah->flat_size = 0;
     171                 : 
     172                 :     /* remember we have a flat representation */
     173            8190 :     eah->fvalue = array;
     174            8190 :     eah->fstartptr = ARR_DATA_PTR(array);
     175            8190 :     eah->fendptr = ((char *) array) + ARR_SIZE(array);
     176                 : 
     177                 :     /* return a R/W pointer to the expanded array */
     178            8190 :     return EOHPGetRWDatum(&eah->hdr);
     179                 : }
     180                 : 
     181                 : /*
     182                 :  * helper for expand_array(): copy pass-by-value Datum-array representation
     183                 :  */
     184                 : static void
     185              12 : copy_byval_expanded_array(ExpandedArrayHeader *eah,
     186                 :                           ExpandedArrayHeader *oldeah)
     187                 : {
     188              12 :     MemoryContext objcxt = eah->hdr.eoh_context;
     189              12 :     int         ndims = oldeah->ndims;
     190              12 :     int         dvalueslen = oldeah->dvalueslen;
     191                 : 
     192                 :     /* Copy array dimensionality information */
     193              12 :     eah->ndims = ndims;
     194                 :     /* We can alloc both dimensionality arrays with one palloc */
     195              12 :     eah->dims = (int *) MemoryContextAlloc(objcxt, ndims * 2 * sizeof(int));
     196              12 :     eah->lbound = eah->dims + ndims;
     197                 :     /* .. but don't assume the source's arrays are contiguous */
     198              12 :     memcpy(eah->dims, oldeah->dims, ndims * sizeof(int));
     199              12 :     memcpy(eah->lbound, oldeah->lbound, ndims * sizeof(int));
     200                 : 
     201                 :     /* Copy element-type data */
     202              12 :     eah->element_type = oldeah->element_type;
     203              12 :     eah->typlen = oldeah->typlen;
     204              12 :     eah->typbyval = oldeah->typbyval;
     205              12 :     eah->typalign = oldeah->typalign;
     206                 : 
     207                 :     /* Copy the deconstructed representation */
     208              12 :     eah->dvalues = (Datum *) MemoryContextAlloc(objcxt,
     209                 :                                                 dvalueslen * sizeof(Datum));
     210              12 :     memcpy(eah->dvalues, oldeah->dvalues, dvalueslen * sizeof(Datum));
     211              12 :     if (oldeah->dnulls)
     212                 :     {
     213 UBC           0 :         eah->dnulls = (bool *) MemoryContextAlloc(objcxt,
     214                 :                                                   dvalueslen * sizeof(bool));
     215               0 :         memcpy(eah->dnulls, oldeah->dnulls, dvalueslen * sizeof(bool));
     216                 :     }
     217                 :     else
     218 CBC          12 :         eah->dnulls = NULL;
     219              12 :     eah->dvalueslen = dvalueslen;
     220              12 :     eah->nelems = oldeah->nelems;
     221              12 :     eah->flat_size = oldeah->flat_size;
     222                 : 
     223                 :     /* we don't make a flat representation */
     224              12 :     eah->fvalue = NULL;
     225              12 :     eah->fstartptr = NULL;
     226              12 :     eah->fendptr = NULL;
     227              12 : }
     228                 : 
     229                 : /*
     230                 :  * get_flat_size method for expanded arrays
     231                 :  */
     232                 : static Size
     233            5997 : EA_get_flat_size(ExpandedObjectHeader *eohptr)
     234                 : {
     235            5997 :     ExpandedArrayHeader *eah = (ExpandedArrayHeader *) eohptr;
     236                 :     int         nelems;
     237                 :     int         ndims;
     238                 :     Datum      *dvalues;
     239                 :     bool       *dnulls;
     240                 :     Size        nbytes;
     241                 :     int         i;
     242                 : 
     243            5997 :     Assert(eah->ea_magic == EA_MAGIC);
     244                 : 
     245                 :     /* Easy if we have a valid flattened value */
     246            5997 :     if (eah->fvalue)
     247            2988 :         return ARR_SIZE(eah->fvalue);
     248                 : 
     249                 :     /* If we have a cached size value, believe that */
     250            3009 :     if (eah->flat_size)
     251            2128 :         return eah->flat_size;
     252                 : 
     253                 :     /*
     254                 :      * Compute space needed by examining dvalues/dnulls.  Note that the result
     255                 :      * array will have a nulls bitmap if dnulls isn't NULL, even if the array
     256                 :      * doesn't actually contain any nulls now.
     257                 :      */
     258             881 :     nelems = eah->nelems;
     259             881 :     ndims = eah->ndims;
     260             881 :     Assert(nelems == ArrayGetNItems(ndims, eah->dims));
     261             881 :     dvalues = eah->dvalues;
     262             881 :     dnulls = eah->dnulls;
     263             881 :     nbytes = 0;
     264            3212 :     for (i = 0; i < nelems; i++)
     265                 :     {
     266            2331 :         if (dnulls && dnulls[i])
     267 UBC           0 :             continue;
     268 CBC        2331 :         nbytes = att_addlength_datum(nbytes, eah->typlen, dvalues[i]);
     269            2331 :         nbytes = att_align_nominal(nbytes, eah->typalign);
     270                 :         /* check for overflow of total request */
     271            2331 :         if (!AllocSizeIsValid(nbytes))
     272 UBC           0 :             ereport(ERROR,
     273                 :                     (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
     274                 :                      errmsg("array size exceeds the maximum allowed (%d)",
     275                 :                             (int) MaxAllocSize)));
     276                 :     }
     277                 : 
     278 CBC         881 :     if (dnulls)
     279 UBC           0 :         nbytes += ARR_OVERHEAD_WITHNULLS(ndims, nelems);
     280                 :     else
     281 CBC         881 :         nbytes += ARR_OVERHEAD_NONULLS(ndims);
     282                 : 
     283                 :     /* cache for next time */
     284             881 :     eah->flat_size = nbytes;
     285                 : 
     286             881 :     return nbytes;
     287                 : }
     288                 : 
     289                 : /*
     290                 :  * flatten_into method for expanded arrays
     291                 :  */
     292                 : static void
     293            4514 : EA_flatten_into(ExpandedObjectHeader *eohptr,
     294                 :                 void *result, Size allocated_size)
     295                 : {
     296            4514 :     ExpandedArrayHeader *eah = (ExpandedArrayHeader *) eohptr;
     297            4514 :     ArrayType  *aresult = (ArrayType *) result;
     298                 :     int         nelems;
     299                 :     int         ndims;
     300                 :     int32       dataoffset;
     301                 : 
     302            4514 :     Assert(eah->ea_magic == EA_MAGIC);
     303                 : 
     304                 :     /* Easy if we have a valid flattened value */
     305            4514 :     if (eah->fvalue)
     306                 :     {
     307            2961 :         Assert(allocated_size == ARR_SIZE(eah->fvalue));
     308            2961 :         memcpy(result, eah->fvalue, allocated_size);
     309            2961 :         return;
     310                 :     }
     311                 : 
     312                 :     /* Else allocation should match previous get_flat_size result */
     313            1553 :     Assert(allocated_size == eah->flat_size);
     314                 : 
     315                 :     /* Fill result array from dvalues/dnulls */
     316            1553 :     nelems = eah->nelems;
     317            1553 :     ndims = eah->ndims;
     318                 : 
     319            1553 :     if (eah->dnulls)
     320 UBC           0 :         dataoffset = ARR_OVERHEAD_WITHNULLS(ndims, nelems);
     321                 :     else
     322 CBC        1553 :         dataoffset = 0;         /* marker for no null bitmap */
     323                 : 
     324                 :     /* We must ensure that any pad space is zero-filled */
     325            1553 :     memset(aresult, 0, allocated_size);
     326                 : 
     327            1553 :     SET_VARSIZE(aresult, allocated_size);
     328            1553 :     aresult->ndim = ndims;
     329            1553 :     aresult->dataoffset = dataoffset;
     330            1553 :     aresult->elemtype = eah->element_type;
     331            1553 :     memcpy(ARR_DIMS(aresult), eah->dims, ndims * sizeof(int));
     332            1553 :     memcpy(ARR_LBOUND(aresult), eah->lbound, ndims * sizeof(int));
     333                 : 
     334            1553 :     CopyArrayEls(aresult,
     335                 :                  eah->dvalues, eah->dnulls, nelems,
     336            1553 :                  eah->typlen, eah->typbyval, eah->typalign,
     337                 :                  false);
     338                 : }
     339                 : 
     340                 : /*
     341                 :  * Argument fetching support code
     342                 :  */
     343                 : 
     344                 : /*
     345                 :  * DatumGetExpandedArray: get a writable expanded array from an input argument
     346                 :  *
     347                 :  * Caution: if the input is a read/write pointer, this returns the input
     348                 :  * argument; so callers must be sure that their changes are "safe", that is
     349                 :  * they cannot leave the array in a corrupt state.
     350                 :  */
     351                 : ExpandedArrayHeader *
     352            1818 : DatumGetExpandedArray(Datum d)
     353                 : {
     354                 :     /* If it's a writable expanded array already, just return it */
     355            1818 :     if (VARATT_IS_EXTERNAL_EXPANDED_RW(DatumGetPointer(d)))
     356                 :     {
     357             979 :         ExpandedArrayHeader *eah = (ExpandedArrayHeader *) DatumGetEOHP(d);
     358                 : 
     359             979 :         Assert(eah->ea_magic == EA_MAGIC);
     360             979 :         return eah;
     361                 :     }
     362                 : 
     363                 :     /* Else expand the hard way */
     364             839 :     d = expand_array(d, CurrentMemoryContext, NULL);
     365             839 :     return (ExpandedArrayHeader *) DatumGetEOHP(d);
     366                 : }
     367                 : 
     368                 : /*
     369                 :  * As above, when caller has the ability to cache element type info
     370                 :  */
     371                 : ExpandedArrayHeader *
     372             958 : DatumGetExpandedArrayX(Datum d, ArrayMetaState *metacache)
     373                 : {
     374                 :     /* If it's a writable expanded array already, just return it */
     375             958 :     if (VARATT_IS_EXTERNAL_EXPANDED_RW(DatumGetPointer(d)))
     376                 :     {
     377             127 :         ExpandedArrayHeader *eah = (ExpandedArrayHeader *) DatumGetEOHP(d);
     378                 : 
     379             127 :         Assert(eah->ea_magic == EA_MAGIC);
     380                 :         /* Update cache if provided */
     381             127 :         if (metacache)
     382                 :         {
     383             127 :             metacache->element_type = eah->element_type;
     384             127 :             metacache->typlen = eah->typlen;
     385             127 :             metacache->typbyval = eah->typbyval;
     386             127 :             metacache->typalign = eah->typalign;
     387                 :         }
     388             127 :         return eah;
     389                 :     }
     390                 : 
     391                 :     /* Else expand using caller's cache if any */
     392             831 :     d = expand_array(d, CurrentMemoryContext, metacache);
     393             831 :     return (ExpandedArrayHeader *) DatumGetEOHP(d);
     394                 : }
     395                 : 
     396                 : /*
     397                 :  * DatumGetAnyArrayP: return either an expanded array or a detoasted varlena
     398                 :  * array.  The result must not be modified in-place.
     399                 :  */
     400                 : AnyArrayType *
     401        12185951 : DatumGetAnyArrayP(Datum d)
     402                 : {
     403                 :     ExpandedArrayHeader *eah;
     404                 : 
     405                 :     /*
     406                 :      * If it's an expanded array (RW or RO), return the header pointer.
     407                 :      */
     408        12185951 :     if (VARATT_IS_EXTERNAL_EXPANDED(DatumGetPointer(d)))
     409                 :     {
     410            9113 :         eah = (ExpandedArrayHeader *) DatumGetEOHP(d);
     411            9113 :         Assert(eah->ea_magic == EA_MAGIC);
     412            9113 :         return (AnyArrayType *) eah;
     413                 :     }
     414                 : 
     415                 :     /* Else do regular detoasting as needed */
     416        12176838 :     return (AnyArrayType *) PG_DETOAST_DATUM(d);
     417                 : }
     418                 : 
     419                 : /*
     420                 :  * Create the Datum/isnull representation of an expanded array object
     421                 :  * if we didn't do so previously
     422                 :  */
     423                 : void
     424            6861 : deconstruct_expanded_array(ExpandedArrayHeader *eah)
     425                 : {
     426            6861 :     if (eah->dvalues == NULL)
     427                 :     {
     428            5488 :         MemoryContext oldcxt = MemoryContextSwitchTo(eah->hdr.eoh_context);
     429                 :         Datum      *dvalues;
     430                 :         bool       *dnulls;
     431                 :         int         nelems;
     432                 : 
     433            5488 :         dnulls = NULL;
     434            5488 :         deconstruct_array(eah->fvalue,
     435                 :                           eah->element_type,
     436            5488 :                           eah->typlen, eah->typbyval, eah->typalign,
     437                 :                           &dvalues,
     438            5488 :                           ARR_HASNULL(eah->fvalue) ? &dnulls : NULL,
     439                 :                           &nelems);
     440                 : 
     441                 :         /*
     442                 :          * Update header only after successful completion of this step.  If
     443                 :          * deconstruct_array fails partway through, worst consequence is some
     444                 :          * leaked memory in the object's context.  If the caller fails at a
     445                 :          * later point, that's fine, since the deconstructed representation is
     446                 :          * valid anyhow.
     447                 :          */
     448            5488 :         eah->dvalues = dvalues;
     449            5488 :         eah->dnulls = dnulls;
     450            5488 :         eah->dvalueslen = eah->nelems = nelems;
     451            5488 :         MemoryContextSwitchTo(oldcxt);
     452                 :     }
     453            6861 : }
        

Generated by: LCOV version v1.16-55-g56c0a2a