LCOV - differential code coverage report
Current view: top level - src/include/utils - sortsupport.h (source / functions) Coverage Total Hit UBC CBC
Current: Differential Code Coverage HEAD vs 15 Lines: 87.0 % 77 67 10 67
Current Date: 2023-04-08 17:13:01 Functions: 100.0 % 5 5 5
Baseline: 15 Line coverage date bins:
Baseline Date: 2023-04-08 15:09:40 (240..) days: 87.0 % 77 67 10 67
Legend: Lines: hit not hit Function coverage date bins:
(240..) days: 100.0 % 5 5 5

 Age         Owner                  TLA  Line data    Source code
                                  1                 : /*-------------------------------------------------------------------------
                                  2                 :  *
                                  3                 :  * sortsupport.h
                                  4                 :  *    Framework for accelerated sorting.
                                  5                 :  *
                                  6                 :  * Traditionally, PostgreSQL has implemented sorting by repeatedly invoking
                                  7                 :  * an SQL-callable comparison function "cmp(x, y) returns int" on pairs of
                                  8                 :  * values to be compared, where the comparison function is the BTORDER_PROC
                                  9                 :  * pg_amproc support function of the appropriate btree index opclass.
                                 10                 :  *
                                 11                 :  * This file defines alternative APIs that allow sorting to be performed with
                                 12                 :  * reduced overhead.  To support lower-overhead sorting, a btree opclass may
                                 13                 :  * provide a BTSORTSUPPORT_PROC pg_amproc entry, which must take a single
                                 14                 :  * argument of type internal and return void.  The argument is actually a
                                 15                 :  * pointer to a SortSupportData struct, which is defined below.
                                 16                 :  *
                                 17                 :  * If provided, the BTSORTSUPPORT function will be called during sort setup,
                                 18                 :  * and it must initialize the provided struct with pointers to function(s)
                                 19                 :  * that can be called to perform sorting.  This API is defined to allow
                                 20                 :  * multiple acceleration mechanisms to be supported, but no opclass is
                                 21                 :  * required to provide all of them.  The BTSORTSUPPORT function should
                                 22                 :  * simply not set any function pointers for mechanisms it doesn't support.
                                 23                 :  * Opclasses that provide BTSORTSUPPORT and don't provide a comparator
                                 24                 :  * function will have a shim set up by sort support automatically.  However,
                                 25                 :  * opclasses that support the optional additional abbreviated key capability
                                 26                 :  * must always provide an authoritative comparator used to tie-break
                                 27                 :  * inconclusive abbreviated comparisons and also used when aborting
                                 28                 :  * abbreviation.  Furthermore, a converter and abort/costing function must be
                                 29                 :  * provided.
                                 30                 :  *
                                 31                 :  * All sort support functions will be passed the address of the
                                 32                 :  * SortSupportData struct when called, so they can use it to store
                                 33                 :  * additional private data as needed.  In particular, for collation-aware
                                 34                 :  * datatypes, the ssup_collation field is set before calling BTSORTSUPPORT
                                 35                 :  * and is available to all support functions.  Additional opclass-dependent
                                 36                 :  * data can be stored using the ssup_extra field.  Any such data
                                 37                 :  * should be allocated in the ssup_cxt memory context.
                                 38                 :  *
                                 39                 :  * Note: since pg_amproc functions are indexed by (lefttype, righttype)
                                 40                 :  * it is possible to associate a BTSORTSUPPORT function with a cross-type
                                 41                 :  * comparison.  This could sensibly be used to provide a fast comparator
                                 42                 :  * function for such cases, but probably not any other acceleration method.
                                 43                 :  *
                                 44                 :  *
                                 45                 :  * Portions Copyright (c) 1996-2023, PostgreSQL Global Development Group
                                 46                 :  * Portions Copyright (c) 1994, Regents of the University of California
                                 47                 :  *
                                 48                 :  * src/include/utils/sortsupport.h
                                 49                 :  *
                                 50                 :  *-------------------------------------------------------------------------
                                 51                 :  */
                                 52                 : #ifndef SORTSUPPORT_H
                                 53                 : #define SORTSUPPORT_H
                                 54                 : 
                                 55                 : #include "access/attnum.h"
                                 56                 : #include "utils/relcache.h"
                                 57                 : 
                                 58                 : typedef struct SortSupportData *SortSupport;
                                 59                 : 
                                 60                 : typedef struct SortSupportData
                                 61                 : {
                                 62                 :     /*
                                 63                 :      * These fields are initialized before calling the BTSORTSUPPORT function
                                 64                 :      * and should not be changed later.
                                 65                 :      */
                                 66                 :     MemoryContext ssup_cxt;     /* Context containing sort info */
                                 67                 :     Oid         ssup_collation; /* Collation to use, or InvalidOid */
                                 68                 : 
                                 69                 :     /*
                                 70                 :      * Additional sorting parameters; but unlike ssup_collation, these can be
                                 71                 :      * changed after BTSORTSUPPORT is called, so don't use them in selecting
                                 72                 :      * sort support functions.
                                 73                 :      */
                                 74                 :     bool        ssup_reverse;   /* descending-order sort? */
                                 75                 :     bool        ssup_nulls_first;   /* sort nulls first? */
                                 76                 : 
                                 77                 :     /*
                                 78                 :      * These fields are workspace for callers, and should not be touched by
                                 79                 :      * opclass-specific functions.
                                 80                 :      */
                                 81                 :     AttrNumber  ssup_attno;     /* column number to sort */
                                 82                 : 
                                 83                 :     /*
                                 84                 :      * ssup_extra is zeroed before calling the BTSORTSUPPORT function, and is
                                 85                 :      * not touched subsequently by callers.
                                 86                 :      */
                                 87                 :     void       *ssup_extra;     /* Workspace for opclass functions */
                                 88                 : 
                                 89                 :     /*
                                 90                 :      * Function pointers are zeroed before calling the BTSORTSUPPORT function,
                                 91                 :      * and must be set by it for any acceleration methods it wants to supply.
                                 92                 :      * The comparator pointer must be set, others are optional.
                                 93                 :      */
                                 94                 : 
                                 95                 :     /*
                                 96                 :      * Comparator function has the same API as the traditional btree
                                 97                 :      * comparison function, ie, return <0, 0, or >0 according as x is less
                                 98                 :      * than, equal to, or greater than y.  Note that x and y are guaranteed
                                 99                 :      * not null, and there is no way to return null either.
                                100                 :      *
                                101                 :      * This may be either the authoritative comparator, or the abbreviated
                                102                 :      * comparator.  Core code may switch this over the initial preference of
                                103                 :      * an opclass support function despite originally indicating abbreviation
                                104                 :      * was applicable, by assigning the authoritative comparator back.
                                105                 :      */
                                106                 :     int         (*comparator) (Datum x, Datum y, SortSupport ssup);
                                107                 : 
                                108                 :     /*
                                109                 :      * "Abbreviated key" infrastructure follows.
                                110                 :      *
                                111                 :      * All callbacks must be set by sortsupport opclasses that make use of
                                112                 :      * this optional additional infrastructure (unless for whatever reasons
                                113                 :      * the opclass doesn't proceed with abbreviation, in which case
                                114                 :      * abbrev_converter must not be set).
                                115                 :      *
                                116                 :      * This allows opclass authors to supply a conversion routine, used to
                                117                 :      * create an alternative representation of the underlying type (an
                                118                 :      * "abbreviated key").  This representation must be pass-by-value and
                                119                 :      * typically will use some ad-hoc format that only the opclass has
                                120                 :      * knowledge of.  An alternative comparator, used only with this
                                121                 :      * alternative representation must also be provided (which is assigned to
                                122                 :      * "comparator").  This representation is a simple approximation of the
                                123                 :      * original Datum.  It must be possible to compare datums of this
                                124                 :      * representation with each other using the supplied alternative
                                125                 :      * comparator, and have any non-zero return value be a reliable proxy for
                                126                 :      * what a proper comparison would indicate. Returning zero from the
                                127                 :      * alternative comparator does not indicate equality, as with a
                                128                 :      * conventional support routine 1, though -- it indicates that it wasn't
                                129                 :      * possible to determine how the two abbreviated values compared.  A
                                130                 :      * proper comparison, using "abbrev_full_comparator"/
                                131                 :      * ApplySortAbbrevFullComparator() is therefore required.  In many cases
                                132                 :      * this results in most or all comparisons only using the cheap
                                133                 :      * alternative comparison func, which is typically implemented as code
                                134                 :      * that compiles to just a few CPU instructions.  CPU cache miss penalties
                                135                 :      * are expensive; to get good overall performance, sort infrastructure
                                136                 :      * must heavily weigh cache performance.
                                137                 :      *
                                138                 :      * Opclass authors must consider the final cardinality of abbreviated keys
                                139                 :      * when devising an encoding scheme.  It's possible for a strategy to work
                                140                 :      * better than an alternative strategy with one usage pattern, while the
                                141                 :      * reverse might be true for another usage pattern.  All of these factors
                                142                 :      * must be considered.
                                143                 :      */
                                144                 : 
                                145                 :     /*
                                146                 :      * "abbreviate" concerns whether or not the abbreviated key optimization
                                147                 :      * is applicable in principle (that is, the sortsupport routine needs to
                                148                 :      * know if its dealing with a key where an abbreviated representation can
                                149                 :      * usefully be packed together.  Conventionally, this is the leading
                                150                 :      * attribute key).  Note, however, that in order to determine that
                                151                 :      * abbreviation is not in play, the core code always checks whether or not
                                152                 :      * the opclass has set abbrev_converter.  This is a one way, one time
                                153                 :      * message to the opclass.
                                154                 :      */
                                155                 :     bool        abbreviate;
                                156                 : 
                                157                 :     /*
                                158                 :      * Converter to abbreviated format, from original representation.  Core
                                159                 :      * code uses this callback to convert from a pass-by-reference "original"
                                160                 :      * Datum to a pass-by-value abbreviated key Datum.  Note that original is
                                161                 :      * guaranteed NOT NULL, because it doesn't make sense to factor NULLness
                                162                 :      * into ad-hoc cost model.
                                163                 :      *
                                164                 :      * abbrev_converter is tested to see if abbreviation is in play.  Core
                                165                 :      * code may set it to NULL to indicate abbreviation should not be used
                                166                 :      * (which is something sortsupport routines need not concern themselves
                                167                 :      * with). However, sortsupport routines must not set it when it is
                                168                 :      * immediately established that abbreviation should not proceed (e.g., for
                                169                 :      * !abbreviate calls, or due to platform-specific impediments to using
                                170                 :      * abbreviation).
                                171                 :      */
                                172                 :     Datum       (*abbrev_converter) (Datum original, SortSupport ssup);
                                173                 : 
                                174                 :     /*
                                175                 :      * abbrev_abort callback allows clients to verify that the current
                                176                 :      * strategy is working out, using a sortsupport routine defined ad-hoc
                                177                 :      * cost model. If there is a lot of duplicate abbreviated keys in
                                178                 :      * practice, it's useful to be able to abandon the strategy before paying
                                179                 :      * too high a cost in conversion (perhaps certain opclass-specific
                                180                 :      * adaptations are useful too).
                                181                 :      */
                                182                 :     bool        (*abbrev_abort) (int memtupcount, SortSupport ssup);
                                183                 : 
                                184                 :     /*
                                185                 :      * Full, authoritative comparator for key that an abbreviated
                                186                 :      * representation was generated for, used when an abbreviated comparison
                                187                 :      * was inconclusive (by calling ApplySortAbbrevFullComparator()), or used
                                188                 :      * to replace "comparator" when core system ultimately decides against
                                189                 :      * abbreviation.
                                190                 :      */
                                191                 :     int         (*abbrev_full_comparator) (Datum x, Datum y, SortSupport ssup);
                                192                 : } SortSupportData;
                                193                 : 
                                194                 : 
                                195                 : /*
                                196                 :  * Apply a sort comparator function and return a 3-way comparison result.
                                197                 :  * This takes care of handling reverse-sort and NULLs-ordering properly.
                                198                 :  */
                                199                 : static inline int
 4141 tgl                       200 CBC   978875233 : ApplySortComparator(Datum datum1, bool isNull1,
                                201                 :                     Datum datum2, bool isNull2,
                                202                 :                     SortSupport ssup)
                                203                 : {
                                204                 :     int         compare;
                                205                 : 
                                206       978875233 :     if (isNull1)
                                207                 :     {
                                208          197790 :         if (isNull2)
                                209          181612 :             compare = 0;        /* NULL "=" NULL */
                                210           16178 :         else if (ssup->ssup_nulls_first)
                                211            2503 :             compare = -1;       /* NULL "<" NOT_NULL */
                                212                 :         else
                                213           13675 :             compare = 1;        /* NULL ">" NOT_NULL */
                                214                 :     }
                                215       978677443 :     else if (isNull2)
                                216                 :     {
                                217           35557 :         if (ssup->ssup_nulls_first)
                                218              43 :             compare = 1;        /* NOT_NULL ">" NULL */
                                219                 :         else
                                220           35514 :             compare = -1;       /* NOT_NULL "<" NULL */
                                221                 :     }
                                222                 :     else
                                223                 :     {
 2040 peter_e                   224       978641886 :         compare = ssup->comparator(datum1, datum2, ssup);
 4141 tgl                       225       978641886 :         if (ssup->ssup_reverse)
 1647                           226         4823610 :             INVERT_COMPARE_RESULT(compare);
                                227                 :     }
                                228                 : 
 4141                           229       978875233 :     return compare;
                                230                 : }
                                231                 : 
                                232                 : static inline int
  372 john.naylor               233        22722154 : ApplyUnsignedSortComparator(Datum datum1, bool isNull1,
                                234                 :                             Datum datum2, bool isNull2,
                                235                 :                             SortSupport ssup)
                                236                 : {
                                237                 :     int         compare;
                                238                 : 
                                239        22722154 :     if (isNull1)
                                240                 :     {
                                241            1853 :         if (isNull2)
                                242             268 :             compare = 0;        /* NULL "=" NULL */
                                243            1585 :         else if (ssup->ssup_nulls_first)
                                244             114 :             compare = -1;       /* NULL "<" NOT_NULL */
                                245                 :         else
                                246            1471 :             compare = 1;        /* NULL ">" NOT_NULL */
                                247                 :     }
                                248        22720301 :     else if (isNull2)
                                249                 :     {
                                250             485 :         if (ssup->ssup_nulls_first)
                                251              33 :             compare = 1;        /* NOT_NULL ">" NULL */
                                252                 :         else
                                253             452 :             compare = -1;       /* NOT_NULL "<" NULL */
                                254                 :     }
                                255                 :     else
                                256                 :     {
                                257        22719816 :         compare = datum1 < datum2 ? -1 : datum1 > datum2 ? 1 : 0;
                                258        22719816 :         if (ssup->ssup_reverse)
                                259          492190 :             INVERT_COMPARE_RESULT(compare);
                                260                 :     }
                                261                 : 
                                262        22722154 :     return compare;
                                263                 : }
                                264                 : 
                                265                 : #if SIZEOF_DATUM >= 8
                                266                 : static inline int
                                267         2811465 : ApplySignedSortComparator(Datum datum1, bool isNull1,
                                268                 :                           Datum datum2, bool isNull2,
                                269                 :                           SortSupport ssup)
                                270                 : {
                                271                 :     int         compare;
                                272                 : 
                                273         2811465 :     if (isNull1)
                                274                 :     {
                                275               6 :         if (isNull2)
                                276               6 :             compare = 0;        /* NULL "=" NULL */
  372 john.naylor               277 UBC           0 :         else if (ssup->ssup_nulls_first)
                                278               0 :             compare = -1;       /* NULL "<" NOT_NULL */
                                279                 :         else
                                280               0 :             compare = 1;        /* NULL ">" NOT_NULL */
                                281                 :     }
  372 john.naylor               282 CBC     2811459 :     else if (isNull2)
                                283                 :     {
                                284               6 :         if (ssup->ssup_nulls_first)
  372 john.naylor               285 UBC           0 :             compare = 1;        /* NOT_NULL ">" NULL */
                                286                 :         else
  372 john.naylor               287 CBC           6 :             compare = -1;       /* NOT_NULL "<" NULL */
                                288                 :     }
                                289                 :     else
                                290                 :     {
  332                           291         2811453 :         compare = DatumGetInt64(datum1) < DatumGetInt64(datum2) ? -1 :
                                292          831841 :             DatumGetInt64(datum1) > DatumGetInt64(datum2) ? 1 : 0;
  372                           293         2811453 :         if (ssup->ssup_reverse)
                                294             846 :             INVERT_COMPARE_RESULT(compare);
                                295                 :     }
                                296                 : 
                                297         2811465 :     return compare;
                                298                 : }
                                299                 : #endif
                                300                 : 
                                301                 : static inline int
                                302        26236863 : ApplyInt32SortComparator(Datum datum1, bool isNull1,
                                303                 :                          Datum datum2, bool isNull2,
                                304                 :                          SortSupport ssup)
                                305                 : {
                                306                 :     int         compare;
                                307                 : 
                                308        26236863 :     if (isNull1)
                                309                 :     {
                                310            2484 :         if (isNull2)
                                311             905 :             compare = 0;        /* NULL "=" NULL */
                                312            1579 :         else if (ssup->ssup_nulls_first)
                                313             400 :             compare = -1;       /* NULL "<" NOT_NULL */
                                314                 :         else
                                315            1179 :             compare = 1;        /* NULL ">" NOT_NULL */
                                316                 :     }
                                317        26234379 :     else if (isNull2)
                                318                 :     {
                                319            1175 :         if (ssup->ssup_nulls_first)
                                320             346 :             compare = 1;        /* NOT_NULL ">" NULL */
                                321                 :         else
                                322             829 :             compare = -1;       /* NOT_NULL "<" NULL */
                                323                 :     }
                                324                 :     else
                                325                 :     {
  332                           326        26233204 :         compare = DatumGetInt32(datum1) < DatumGetInt32(datum2) ? -1 :
                                327        15690792 :             DatumGetInt32(datum1) > DatumGetInt32(datum2) ? 1 : 0;
  372                           328        26233204 :         if (ssup->ssup_reverse)
                                329         2545612 :             INVERT_COMPARE_RESULT(compare);
                                330                 :     }
                                331                 : 
                                332        26236863 :     return compare;
                                333                 : }
                                334                 : 
                                335                 : /*
                                336                 :  * Apply a sort comparator function and return a 3-way comparison using full,
                                337                 :  * authoritative comparator.  This takes care of handling reverse-sort and
                                338                 :  * NULLs-ordering properly.
                                339                 :  */
                                340                 : static inline int
 3002 rhaas                     341         2074586 : ApplySortAbbrevFullComparator(Datum datum1, bool isNull1,
                                342                 :                               Datum datum2, bool isNull2,
                                343                 :                               SortSupport ssup)
                                344                 : {
                                345                 :     int         compare;
                                346                 : 
                                347         2074586 :     if (isNull1)
                                348                 :     {
                                349             269 :         if (isNull2)
                                350             269 :             compare = 0;        /* NULL "=" NULL */
 3002 rhaas                     351 UBC           0 :         else if (ssup->ssup_nulls_first)
                                352               0 :             compare = -1;       /* NULL "<" NOT_NULL */
                                353                 :         else
                                354               0 :             compare = 1;        /* NULL ">" NOT_NULL */
                                355                 :     }
 3002 rhaas                     356 CBC     2074317 :     else if (isNull2)
                                357                 :     {
 3002 rhaas                     358 UBC           0 :         if (ssup->ssup_nulls_first)
                                359               0 :             compare = 1;        /* NOT_NULL ">" NULL */
                                360                 :         else
                                361               0 :             compare = -1;       /* NOT_NULL "<" NULL */
                                362                 :     }
                                363                 :     else
                                364                 :     {
 2040 peter_e                   365 CBC     2074317 :         compare = ssup->abbrev_full_comparator(datum1, datum2, ssup);
 3002 rhaas                     366         2074317 :         if (ssup->ssup_reverse)
 1647 tgl                       367          105039 :             INVERT_COMPARE_RESULT(compare);
                                368                 :     }
                                369                 : 
 3002 rhaas                     370         2074586 :     return compare;
                                371                 : }
                                372                 : 
                                373                 : /*
                                374                 :  * Datum comparison functions that we have specialized sort routines for.
                                375                 :  * Datatypes that install these as their comparator or abbrevated comparator
                                376                 :  * are eligible for faster sorting.
                                377                 :  */
                                378                 : extern int  ssup_datum_unsigned_cmp(Datum x, Datum y, SortSupport ssup);
                                379                 : #if SIZEOF_DATUM >= 8
                                380                 : extern int  ssup_datum_signed_cmp(Datum x, Datum y, SortSupport ssup);
                                381                 : #endif
                                382                 : extern int  ssup_datum_int32_cmp(Datum x, Datum y, SortSupport ssup);
                                383                 : 
                                384                 : /* Other functions in utils/sort/sortsupport.c */
                                385                 : extern void PrepareSortSupportComparisonShim(Oid cmpFunc, SortSupport ssup);
                                386                 : extern void PrepareSortSupportFromOrderingOp(Oid orderingOp, SortSupport ssup);
                                387                 : extern void PrepareSortSupportFromIndexRel(Relation indexRel, int16 strategy,
                                388                 :                                            SortSupport ssup);
                                389                 : extern void PrepareSortSupportFromGistIndexRel(Relation indexRel, SortSupport ssup);
                                390                 : 
                                391                 : #endif                          /* SORTSUPPORT_H */
        

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