LCOV - code coverage report
Current view: top level - src/backend/utils/sort - tuplesortvariants.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 665 695 95.7 %
Date: 2025-01-18 04:15:08 Functions: 42 44 95.5 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * tuplesortvariants.c
       4             :  *    Implementation of tuple sorting variants.
       5             :  *
       6             :  * This module handles the sorting of heap tuples, index tuples, or single
       7             :  * Datums.  The implementation is based on the generalized tuple sorting
       8             :  * facility given in tuplesort.c.  Support other kinds of sortable objects
       9             :  * could be easily added here, another module, or even an extension.
      10             :  *
      11             :  *
      12             :  * Copyright (c) 2022-2025, PostgreSQL Global Development Group
      13             :  *
      14             :  * IDENTIFICATION
      15             :  *    src/backend/utils/sort/tuplesortvariants.c
      16             :  *
      17             :  *-------------------------------------------------------------------------
      18             :  */
      19             : 
      20             : #include "postgres.h"
      21             : 
      22             : #include "access/brin_tuple.h"
      23             : #include "access/hash.h"
      24             : #include "access/htup_details.h"
      25             : #include "access/nbtree.h"
      26             : #include "catalog/index.h"
      27             : #include "executor/executor.h"
      28             : #include "pg_trace.h"
      29             : #include "utils/datum.h"
      30             : #include "utils/guc.h"
      31             : #include "utils/lsyscache.h"
      32             : #include "utils/tuplesort.h"
      33             : 
      34             : 
      35             : /* sort-type codes for sort__start probes */
      36             : #define HEAP_SORT       0
      37             : #define INDEX_SORT      1
      38             : #define DATUM_SORT      2
      39             : #define CLUSTER_SORT    3
      40             : 
      41             : static void removeabbrev_heap(Tuplesortstate *state, SortTuple *stups,
      42             :                               int count);
      43             : static void removeabbrev_cluster(Tuplesortstate *state, SortTuple *stups,
      44             :                                  int count);
      45             : static void removeabbrev_index(Tuplesortstate *state, SortTuple *stups,
      46             :                                int count);
      47             : static void removeabbrev_index_brin(Tuplesortstate *state, SortTuple *stups,
      48             :                                     int count);
      49             : static void removeabbrev_datum(Tuplesortstate *state, SortTuple *stups,
      50             :                                int count);
      51             : static int  comparetup_heap(const SortTuple *a, const SortTuple *b,
      52             :                             Tuplesortstate *state);
      53             : static int  comparetup_heap_tiebreak(const SortTuple *a, const SortTuple *b,
      54             :                                      Tuplesortstate *state);
      55             : static void writetup_heap(Tuplesortstate *state, LogicalTape *tape,
      56             :                           SortTuple *stup);
      57             : static void readtup_heap(Tuplesortstate *state, SortTuple *stup,
      58             :                          LogicalTape *tape, unsigned int len);
      59             : static int  comparetup_cluster(const SortTuple *a, const SortTuple *b,
      60             :                                Tuplesortstate *state);
      61             : static int  comparetup_cluster_tiebreak(const SortTuple *a, const SortTuple *b,
      62             :                                         Tuplesortstate *state);
      63             : static void writetup_cluster(Tuplesortstate *state, LogicalTape *tape,
      64             :                              SortTuple *stup);
      65             : static void readtup_cluster(Tuplesortstate *state, SortTuple *stup,
      66             :                             LogicalTape *tape, unsigned int tuplen);
      67             : static int  comparetup_index_btree(const SortTuple *a, const SortTuple *b,
      68             :                                    Tuplesortstate *state);
      69             : static int  comparetup_index_btree_tiebreak(const SortTuple *a, const SortTuple *b,
      70             :                                             Tuplesortstate *state);
      71             : static int  comparetup_index_hash(const SortTuple *a, const SortTuple *b,
      72             :                                   Tuplesortstate *state);
      73             : static int  comparetup_index_hash_tiebreak(const SortTuple *a, const SortTuple *b,
      74             :                                            Tuplesortstate *state);
      75             : static int  comparetup_index_brin(const SortTuple *a, const SortTuple *b,
      76             :                                   Tuplesortstate *state);
      77             : static void writetup_index(Tuplesortstate *state, LogicalTape *tape,
      78             :                            SortTuple *stup);
      79             : static void readtup_index(Tuplesortstate *state, SortTuple *stup,
      80             :                           LogicalTape *tape, unsigned int len);
      81             : static void writetup_index_brin(Tuplesortstate *state, LogicalTape *tape,
      82             :                                 SortTuple *stup);
      83             : static void readtup_index_brin(Tuplesortstate *state, SortTuple *stup,
      84             :                                LogicalTape *tape, unsigned int len);
      85             : static int  comparetup_datum(const SortTuple *a, const SortTuple *b,
      86             :                              Tuplesortstate *state);
      87             : static int  comparetup_datum_tiebreak(const SortTuple *a, const SortTuple *b,
      88             :                                       Tuplesortstate *state);
      89             : static void writetup_datum(Tuplesortstate *state, LogicalTape *tape,
      90             :                            SortTuple *stup);
      91             : static void readtup_datum(Tuplesortstate *state, SortTuple *stup,
      92             :                           LogicalTape *tape, unsigned int len);
      93             : static void freestate_cluster(Tuplesortstate *state);
      94             : 
      95             : /*
      96             :  * Data structure pointed by "TuplesortPublic.arg" for the CLUSTER case.  Set by
      97             :  * the tuplesort_begin_cluster.
      98             :  */
      99             : typedef struct
     100             : {
     101             :     TupleDesc   tupDesc;
     102             : 
     103             :     IndexInfo  *indexInfo;      /* info about index being used for reference */
     104             :     EState     *estate;         /* for evaluating index expressions */
     105             : } TuplesortClusterArg;
     106             : 
     107             : /*
     108             :  * Data structure pointed by "TuplesortPublic.arg" for the IndexTuple case.
     109             :  * Set by tuplesort_begin_index_xxx and used only by the IndexTuple routines.
     110             :  */
     111             : typedef struct
     112             : {
     113             :     Relation    heapRel;        /* table the index is being built on */
     114             :     Relation    indexRel;       /* index being built */
     115             : } TuplesortIndexArg;
     116             : 
     117             : /*
     118             :  * Data structure pointed by "TuplesortPublic.arg" for the index_btree subcase.
     119             :  */
     120             : typedef struct
     121             : {
     122             :     TuplesortIndexArg index;
     123             : 
     124             :     bool        enforceUnique;  /* complain if we find duplicate tuples */
     125             :     bool        uniqueNullsNotDistinct; /* unique constraint null treatment */
     126             : } TuplesortIndexBTreeArg;
     127             : 
     128             : /*
     129             :  * Data structure pointed by "TuplesortPublic.arg" for the index_hash subcase.
     130             :  */
     131             : typedef struct
     132             : {
     133             :     TuplesortIndexArg index;
     134             : 
     135             :     uint32      high_mask;      /* masks for sortable part of hash code */
     136             :     uint32      low_mask;
     137             :     uint32      max_buckets;
     138             : } TuplesortIndexHashArg;
     139             : 
     140             : /*
     141             :  * Data structure pointed by "TuplesortPublic.arg" for the Datum case.
     142             :  * Set by tuplesort_begin_datum and used only by the DatumTuple routines.
     143             :  */
     144             : typedef struct
     145             : {
     146             :     /* the datatype oid of Datum's to be sorted */
     147             :     Oid         datumType;
     148             :     /* we need typelen in order to know how to copy the Datums. */
     149             :     int         datumTypeLen;
     150             : } TuplesortDatumArg;
     151             : 
     152             : /*
     153             :  * Computing BrinTuple size with only the tuple is difficult, so we want to track
     154             :  * the length referenced by the SortTuple. That's what BrinSortTuple is meant
     155             :  * to do - it's essentially a BrinTuple prefixed by its length.
     156             :  */
     157             : typedef struct BrinSortTuple
     158             : {
     159             :     Size        tuplen;
     160             :     BrinTuple   tuple;
     161             : } BrinSortTuple;
     162             : 
     163             : /* Size of the BrinSortTuple, given length of the BrinTuple. */
     164             : #define BRINSORTTUPLE_SIZE(len)     (offsetof(BrinSortTuple, tuple) + (len))
     165             : 
     166             : 
     167             : Tuplesortstate *
     168      105050 : tuplesort_begin_heap(TupleDesc tupDesc,
     169             :                      int nkeys, AttrNumber *attNums,
     170             :                      Oid *sortOperators, Oid *sortCollations,
     171             :                      bool *nullsFirstFlags,
     172             :                      int workMem, SortCoordinate coordinate, int sortopt)
     173             : {
     174      105050 :     Tuplesortstate *state = tuplesort_begin_common(workMem, coordinate,
     175             :                                                    sortopt);
     176      105050 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     177             :     MemoryContext oldcontext;
     178             :     int         i;
     179             : 
     180      105050 :     oldcontext = MemoryContextSwitchTo(base->maincontext);
     181             : 
     182             :     Assert(nkeys > 0);
     183             : 
     184      105050 :     if (trace_sort)
     185           0 :         elog(LOG,
     186             :              "begin tuple sort: nkeys = %d, workMem = %d, randomAccess = %c",
     187             :              nkeys, workMem, sortopt & TUPLESORT_RANDOMACCESS ? 't' : 'f');
     188             : 
     189      105050 :     base->nKeys = nkeys;
     190             : 
     191             :     TRACE_POSTGRESQL_SORT_START(HEAP_SORT,
     192             :                                 false,  /* no unique check */
     193             :                                 nkeys,
     194             :                                 workMem,
     195             :                                 sortopt & TUPLESORT_RANDOMACCESS,
     196             :                                 PARALLEL_SORT(coordinate));
     197             : 
     198      105050 :     base->removeabbrev = removeabbrev_heap;
     199      105050 :     base->comparetup = comparetup_heap;
     200      105050 :     base->comparetup_tiebreak = comparetup_heap_tiebreak;
     201      105050 :     base->writetup = writetup_heap;
     202      105050 :     base->readtup = readtup_heap;
     203      105050 :     base->haveDatum1 = true;
     204      105050 :     base->arg = tupDesc;     /* assume we need not copy tupDesc */
     205             : 
     206             :     /* Prepare SortSupport data for each column */
     207      105050 :     base->sortKeys = (SortSupport) palloc0(nkeys * sizeof(SortSupportData));
     208             : 
     209      262672 :     for (i = 0; i < nkeys; i++)
     210             :     {
     211      157634 :         SortSupport sortKey = base->sortKeys + i;
     212             : 
     213             :         Assert(attNums[i] != 0);
     214             :         Assert(sortOperators[i] != 0);
     215             : 
     216      157634 :         sortKey->ssup_cxt = CurrentMemoryContext;
     217      157634 :         sortKey->ssup_collation = sortCollations[i];
     218      157634 :         sortKey->ssup_nulls_first = nullsFirstFlags[i];
     219      157634 :         sortKey->ssup_attno = attNums[i];
     220             :         /* Convey if abbreviation optimization is applicable in principle */
     221      157634 :         sortKey->abbreviate = (i == 0 && base->haveDatum1);
     222             : 
     223      157634 :         PrepareSortSupportFromOrderingOp(sortOperators[i], sortKey);
     224             :     }
     225             : 
     226             :     /*
     227             :      * The "onlyKey" optimization cannot be used with abbreviated keys, since
     228             :      * tie-breaker comparisons may be required.  Typically, the optimization
     229             :      * is only of value to pass-by-value types anyway, whereas abbreviated
     230             :      * keys are typically only of value to pass-by-reference types.
     231             :      */
     232      105038 :     if (nkeys == 1 && !base->sortKeys->abbrev_converter)
     233       58902 :         base->onlyKey = base->sortKeys;
     234             : 
     235      105038 :     MemoryContextSwitchTo(oldcontext);
     236             : 
     237      105038 :     return state;
     238             : }
     239             : 
     240             : Tuplesortstate *
     241         112 : tuplesort_begin_cluster(TupleDesc tupDesc,
     242             :                         Relation indexRel,
     243             :                         int workMem,
     244             :                         SortCoordinate coordinate, int sortopt)
     245             : {
     246         112 :     Tuplesortstate *state = tuplesort_begin_common(workMem, coordinate,
     247             :                                                    sortopt);
     248         112 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     249             :     BTScanInsert indexScanKey;
     250             :     MemoryContext oldcontext;
     251             :     TuplesortClusterArg *arg;
     252             :     int         i;
     253             : 
     254             :     Assert(indexRel->rd_rel->relam == BTREE_AM_OID);
     255             : 
     256         112 :     oldcontext = MemoryContextSwitchTo(base->maincontext);
     257         112 :     arg = (TuplesortClusterArg *) palloc0(sizeof(TuplesortClusterArg));
     258             : 
     259         112 :     if (trace_sort)
     260           0 :         elog(LOG,
     261             :              "begin tuple sort: nkeys = %d, workMem = %d, randomAccess = %c",
     262             :              RelationGetNumberOfAttributes(indexRel),
     263             :              workMem, sortopt & TUPLESORT_RANDOMACCESS ? 't' : 'f');
     264             : 
     265         112 :     base->nKeys = IndexRelationGetNumberOfKeyAttributes(indexRel);
     266             : 
     267             :     TRACE_POSTGRESQL_SORT_START(CLUSTER_SORT,
     268             :                                 false,  /* no unique check */
     269             :                                 base->nKeys,
     270             :                                 workMem,
     271             :                                 sortopt & TUPLESORT_RANDOMACCESS,
     272             :                                 PARALLEL_SORT(coordinate));
     273             : 
     274         112 :     base->removeabbrev = removeabbrev_cluster;
     275         112 :     base->comparetup = comparetup_cluster;
     276         112 :     base->comparetup_tiebreak = comparetup_cluster_tiebreak;
     277         112 :     base->writetup = writetup_cluster;
     278         112 :     base->readtup = readtup_cluster;
     279         112 :     base->freestate = freestate_cluster;
     280         112 :     base->arg = arg;
     281             : 
     282         112 :     arg->indexInfo = BuildIndexInfo(indexRel);
     283             : 
     284             :     /*
     285             :      * If we don't have a simple leading attribute, we don't currently
     286             :      * initialize datum1, so disable optimizations that require it.
     287             :      */
     288         112 :     if (arg->indexInfo->ii_IndexAttrNumbers[0] == 0)
     289          24 :         base->haveDatum1 = false;
     290             :     else
     291          88 :         base->haveDatum1 = true;
     292             : 
     293         112 :     arg->tupDesc = tupDesc;      /* assume we need not copy tupDesc */
     294             : 
     295         112 :     indexScanKey = _bt_mkscankey(indexRel, NULL);
     296             : 
     297         112 :     if (arg->indexInfo->ii_Expressions != NULL)
     298             :     {
     299             :         TupleTableSlot *slot;
     300             :         ExprContext *econtext;
     301             : 
     302             :         /*
     303             :          * We will need to use FormIndexDatum to evaluate the index
     304             :          * expressions.  To do that, we need an EState, as well as a
     305             :          * TupleTableSlot to put the table tuples into.  The econtext's
     306             :          * scantuple has to point to that slot, too.
     307             :          */
     308          24 :         arg->estate = CreateExecutorState();
     309          24 :         slot = MakeSingleTupleTableSlot(tupDesc, &TTSOpsHeapTuple);
     310          24 :         econtext = GetPerTupleExprContext(arg->estate);
     311          24 :         econtext->ecxt_scantuple = slot;
     312             :     }
     313             : 
     314             :     /* Prepare SortSupport data for each column */
     315         112 :     base->sortKeys = (SortSupport) palloc0(base->nKeys *
     316             :                                            sizeof(SortSupportData));
     317             : 
     318         248 :     for (i = 0; i < base->nKeys; i++)
     319             :     {
     320         136 :         SortSupport sortKey = base->sortKeys + i;
     321         136 :         ScanKey     scanKey = indexScanKey->scankeys + i;
     322             :         int16       strategy;
     323             : 
     324         136 :         sortKey->ssup_cxt = CurrentMemoryContext;
     325         136 :         sortKey->ssup_collation = scanKey->sk_collation;
     326         136 :         sortKey->ssup_nulls_first =
     327         136 :             (scanKey->sk_flags & SK_BT_NULLS_FIRST) != 0;
     328         136 :         sortKey->ssup_attno = scanKey->sk_attno;
     329             :         /* Convey if abbreviation optimization is applicable in principle */
     330         136 :         sortKey->abbreviate = (i == 0 && base->haveDatum1);
     331             : 
     332             :         Assert(sortKey->ssup_attno != 0);
     333             : 
     334         136 :         strategy = (scanKey->sk_flags & SK_BT_DESC) != 0 ?
     335             :             BTGreaterStrategyNumber : BTLessStrategyNumber;
     336             : 
     337         136 :         PrepareSortSupportFromIndexRel(indexRel, strategy, sortKey);
     338             :     }
     339             : 
     340         112 :     pfree(indexScanKey);
     341             : 
     342         112 :     MemoryContextSwitchTo(oldcontext);
     343             : 
     344         112 :     return state;
     345             : }
     346             : 
     347             : Tuplesortstate *
     348       87950 : tuplesort_begin_index_btree(Relation heapRel,
     349             :                             Relation indexRel,
     350             :                             bool enforceUnique,
     351             :                             bool uniqueNullsNotDistinct,
     352             :                             int workMem,
     353             :                             SortCoordinate coordinate,
     354             :                             int sortopt)
     355             : {
     356       87950 :     Tuplesortstate *state = tuplesort_begin_common(workMem, coordinate,
     357             :                                                    sortopt);
     358       87950 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     359             :     BTScanInsert indexScanKey;
     360             :     TuplesortIndexBTreeArg *arg;
     361             :     MemoryContext oldcontext;
     362             :     int         i;
     363             : 
     364       87950 :     oldcontext = MemoryContextSwitchTo(base->maincontext);
     365       87950 :     arg = (TuplesortIndexBTreeArg *) palloc(sizeof(TuplesortIndexBTreeArg));
     366             : 
     367       87950 :     if (trace_sort)
     368           0 :         elog(LOG,
     369             :              "begin index sort: unique = %c, workMem = %d, randomAccess = %c",
     370             :              enforceUnique ? 't' : 'f',
     371             :              workMem, sortopt & TUPLESORT_RANDOMACCESS ? 't' : 'f');
     372             : 
     373       87950 :     base->nKeys = IndexRelationGetNumberOfKeyAttributes(indexRel);
     374             : 
     375             :     TRACE_POSTGRESQL_SORT_START(INDEX_SORT,
     376             :                                 enforceUnique,
     377             :                                 base->nKeys,
     378             :                                 workMem,
     379             :                                 sortopt & TUPLESORT_RANDOMACCESS,
     380             :                                 PARALLEL_SORT(coordinate));
     381             : 
     382       87950 :     base->removeabbrev = removeabbrev_index;
     383       87950 :     base->comparetup = comparetup_index_btree;
     384       87950 :     base->comparetup_tiebreak = comparetup_index_btree_tiebreak;
     385       87950 :     base->writetup = writetup_index;
     386       87950 :     base->readtup = readtup_index;
     387       87950 :     base->haveDatum1 = true;
     388       87950 :     base->arg = arg;
     389             : 
     390       87950 :     arg->index.heapRel = heapRel;
     391       87950 :     arg->index.indexRel = indexRel;
     392       87950 :     arg->enforceUnique = enforceUnique;
     393       87950 :     arg->uniqueNullsNotDistinct = uniqueNullsNotDistinct;
     394             : 
     395       87950 :     indexScanKey = _bt_mkscankey(indexRel, NULL);
     396             : 
     397             :     /* Prepare SortSupport data for each column */
     398       87950 :     base->sortKeys = (SortSupport) palloc0(base->nKeys *
     399             :                                            sizeof(SortSupportData));
     400             : 
     401      233774 :     for (i = 0; i < base->nKeys; i++)
     402             :     {
     403      145824 :         SortSupport sortKey = base->sortKeys + i;
     404      145824 :         ScanKey     scanKey = indexScanKey->scankeys + i;
     405             :         int16       strategy;
     406             : 
     407      145824 :         sortKey->ssup_cxt = CurrentMemoryContext;
     408      145824 :         sortKey->ssup_collation = scanKey->sk_collation;
     409      145824 :         sortKey->ssup_nulls_first =
     410      145824 :             (scanKey->sk_flags & SK_BT_NULLS_FIRST) != 0;
     411      145824 :         sortKey->ssup_attno = scanKey->sk_attno;
     412             :         /* Convey if abbreviation optimization is applicable in principle */
     413      145824 :         sortKey->abbreviate = (i == 0 && base->haveDatum1);
     414             : 
     415             :         Assert(sortKey->ssup_attno != 0);
     416             : 
     417      145824 :         strategy = (scanKey->sk_flags & SK_BT_DESC) != 0 ?
     418             :             BTGreaterStrategyNumber : BTLessStrategyNumber;
     419             : 
     420      145824 :         PrepareSortSupportFromIndexRel(indexRel, strategy, sortKey);
     421             :     }
     422             : 
     423       87950 :     pfree(indexScanKey);
     424             : 
     425       87950 :     MemoryContextSwitchTo(oldcontext);
     426             : 
     427       87950 :     return state;
     428             : }
     429             : 
     430             : Tuplesortstate *
     431           8 : tuplesort_begin_index_hash(Relation heapRel,
     432             :                            Relation indexRel,
     433             :                            uint32 high_mask,
     434             :                            uint32 low_mask,
     435             :                            uint32 max_buckets,
     436             :                            int workMem,
     437             :                            SortCoordinate coordinate,
     438             :                            int sortopt)
     439             : {
     440           8 :     Tuplesortstate *state = tuplesort_begin_common(workMem, coordinate,
     441             :                                                    sortopt);
     442           8 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     443             :     MemoryContext oldcontext;
     444             :     TuplesortIndexHashArg *arg;
     445             : 
     446           8 :     oldcontext = MemoryContextSwitchTo(base->maincontext);
     447           8 :     arg = (TuplesortIndexHashArg *) palloc(sizeof(TuplesortIndexHashArg));
     448             : 
     449           8 :     if (trace_sort)
     450           0 :         elog(LOG,
     451             :              "begin index sort: high_mask = 0x%x, low_mask = 0x%x, "
     452             :              "max_buckets = 0x%x, workMem = %d, randomAccess = %c",
     453             :              high_mask,
     454             :              low_mask,
     455             :              max_buckets,
     456             :              workMem,
     457             :              sortopt & TUPLESORT_RANDOMACCESS ? 't' : 'f');
     458             : 
     459           8 :     base->nKeys = 1;         /* Only one sort column, the hash code */
     460             : 
     461           8 :     base->removeabbrev = removeabbrev_index;
     462           8 :     base->comparetup = comparetup_index_hash;
     463           8 :     base->comparetup_tiebreak = comparetup_index_hash_tiebreak;
     464           8 :     base->writetup = writetup_index;
     465           8 :     base->readtup = readtup_index;
     466           8 :     base->haveDatum1 = true;
     467           8 :     base->arg = arg;
     468             : 
     469           8 :     arg->index.heapRel = heapRel;
     470           8 :     arg->index.indexRel = indexRel;
     471             : 
     472           8 :     arg->high_mask = high_mask;
     473           8 :     arg->low_mask = low_mask;
     474           8 :     arg->max_buckets = max_buckets;
     475             : 
     476           8 :     MemoryContextSwitchTo(oldcontext);
     477             : 
     478           8 :     return state;
     479             : }
     480             : 
     481             : Tuplesortstate *
     482         150 : tuplesort_begin_index_gist(Relation heapRel,
     483             :                            Relation indexRel,
     484             :                            int workMem,
     485             :                            SortCoordinate coordinate,
     486             :                            int sortopt)
     487             : {
     488         150 :     Tuplesortstate *state = tuplesort_begin_common(workMem, coordinate,
     489             :                                                    sortopt);
     490         150 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     491             :     MemoryContext oldcontext;
     492             :     TuplesortIndexBTreeArg *arg;
     493             :     int         i;
     494             : 
     495         150 :     oldcontext = MemoryContextSwitchTo(base->maincontext);
     496         150 :     arg = (TuplesortIndexBTreeArg *) palloc(sizeof(TuplesortIndexBTreeArg));
     497             : 
     498         150 :     if (trace_sort)
     499           0 :         elog(LOG,
     500             :              "begin index sort: workMem = %d, randomAccess = %c",
     501             :              workMem, sortopt & TUPLESORT_RANDOMACCESS ? 't' : 'f');
     502             : 
     503         150 :     base->nKeys = IndexRelationGetNumberOfKeyAttributes(indexRel);
     504             : 
     505         150 :     base->removeabbrev = removeabbrev_index;
     506         150 :     base->comparetup = comparetup_index_btree;
     507         150 :     base->comparetup_tiebreak = comparetup_index_btree_tiebreak;
     508         150 :     base->writetup = writetup_index;
     509         150 :     base->readtup = readtup_index;
     510         150 :     base->haveDatum1 = true;
     511         150 :     base->arg = arg;
     512             : 
     513         150 :     arg->index.heapRel = heapRel;
     514         150 :     arg->index.indexRel = indexRel;
     515         150 :     arg->enforceUnique = false;
     516         150 :     arg->uniqueNullsNotDistinct = false;
     517             : 
     518             :     /* Prepare SortSupport data for each column */
     519         150 :     base->sortKeys = (SortSupport) palloc0(base->nKeys *
     520             :                                            sizeof(SortSupportData));
     521             : 
     522         300 :     for (i = 0; i < base->nKeys; i++)
     523             :     {
     524         150 :         SortSupport sortKey = base->sortKeys + i;
     525             : 
     526         150 :         sortKey->ssup_cxt = CurrentMemoryContext;
     527         150 :         sortKey->ssup_collation = indexRel->rd_indcollation[i];
     528         150 :         sortKey->ssup_nulls_first = false;
     529         150 :         sortKey->ssup_attno = i + 1;
     530             :         /* Convey if abbreviation optimization is applicable in principle */
     531         150 :         sortKey->abbreviate = (i == 0 && base->haveDatum1);
     532             : 
     533             :         Assert(sortKey->ssup_attno != 0);
     534             : 
     535             :         /* Look for a sort support function */
     536         150 :         PrepareSortSupportFromGistIndexRel(indexRel, sortKey);
     537             :     }
     538             : 
     539         150 :     MemoryContextSwitchTo(oldcontext);
     540             : 
     541         150 :     return state;
     542             : }
     543             : 
     544             : Tuplesortstate *
     545          28 : tuplesort_begin_index_brin(int workMem,
     546             :                            SortCoordinate coordinate,
     547             :                            int sortopt)
     548             : {
     549          28 :     Tuplesortstate *state = tuplesort_begin_common(workMem, coordinate,
     550             :                                                    sortopt);
     551          28 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     552             : 
     553          28 :     if (trace_sort)
     554           0 :         elog(LOG,
     555             :              "begin index sort: workMem = %d, randomAccess = %c",
     556             :              workMem,
     557             :              sortopt & TUPLESORT_RANDOMACCESS ? 't' : 'f');
     558             : 
     559          28 :     base->nKeys = 1;         /* Only one sort column, the block number */
     560             : 
     561          28 :     base->removeabbrev = removeabbrev_index_brin;
     562          28 :     base->comparetup = comparetup_index_brin;
     563          28 :     base->writetup = writetup_index_brin;
     564          28 :     base->readtup = readtup_index_brin;
     565          28 :     base->haveDatum1 = true;
     566          28 :     base->arg = NULL;
     567             : 
     568          28 :     return state;
     569             : }
     570             : 
     571             : Tuplesortstate *
     572       59804 : tuplesort_begin_datum(Oid datumType, Oid sortOperator, Oid sortCollation,
     573             :                       bool nullsFirstFlag, int workMem,
     574             :                       SortCoordinate coordinate, int sortopt)
     575             : {
     576       59804 :     Tuplesortstate *state = tuplesort_begin_common(workMem, coordinate,
     577             :                                                    sortopt);
     578       59804 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     579             :     TuplesortDatumArg *arg;
     580             :     MemoryContext oldcontext;
     581             :     int16       typlen;
     582             :     bool        typbyval;
     583             : 
     584       59804 :     oldcontext = MemoryContextSwitchTo(base->maincontext);
     585       59804 :     arg = (TuplesortDatumArg *) palloc(sizeof(TuplesortDatumArg));
     586             : 
     587       59804 :     if (trace_sort)
     588           0 :         elog(LOG,
     589             :              "begin datum sort: workMem = %d, randomAccess = %c",
     590             :              workMem, sortopt & TUPLESORT_RANDOMACCESS ? 't' : 'f');
     591             : 
     592       59804 :     base->nKeys = 1;         /* always a one-column sort */
     593             : 
     594             :     TRACE_POSTGRESQL_SORT_START(DATUM_SORT,
     595             :                                 false,  /* no unique check */
     596             :                                 1,
     597             :                                 workMem,
     598             :                                 sortopt & TUPLESORT_RANDOMACCESS,
     599             :                                 PARALLEL_SORT(coordinate));
     600             : 
     601       59804 :     base->removeabbrev = removeabbrev_datum;
     602       59804 :     base->comparetup = comparetup_datum;
     603       59804 :     base->comparetup_tiebreak = comparetup_datum_tiebreak;
     604       59804 :     base->writetup = writetup_datum;
     605       59804 :     base->readtup = readtup_datum;
     606       59804 :     base->haveDatum1 = true;
     607       59804 :     base->arg = arg;
     608             : 
     609       59804 :     arg->datumType = datumType;
     610             : 
     611             :     /* lookup necessary attributes of the datum type */
     612       59804 :     get_typlenbyval(datumType, &typlen, &typbyval);
     613       59804 :     arg->datumTypeLen = typlen;
     614       59804 :     base->tuples = !typbyval;
     615             : 
     616             :     /* Prepare SortSupport data */
     617       59804 :     base->sortKeys = (SortSupport) palloc0(sizeof(SortSupportData));
     618             : 
     619       59804 :     base->sortKeys->ssup_cxt = CurrentMemoryContext;
     620       59804 :     base->sortKeys->ssup_collation = sortCollation;
     621       59804 :     base->sortKeys->ssup_nulls_first = nullsFirstFlag;
     622             : 
     623             :     /*
     624             :      * Abbreviation is possible here only for by-reference types.  In theory,
     625             :      * a pass-by-value datatype could have an abbreviated form that is cheaper
     626             :      * to compare.  In a tuple sort, we could support that, because we can
     627             :      * always extract the original datum from the tuple as needed.  Here, we
     628             :      * can't, because a datum sort only stores a single copy of the datum; the
     629             :      * "tuple" field of each SortTuple is NULL.
     630             :      */
     631       59804 :     base->sortKeys->abbreviate = !typbyval;
     632             : 
     633       59804 :     PrepareSortSupportFromOrderingOp(sortOperator, base->sortKeys);
     634             : 
     635             :     /*
     636             :      * The "onlyKey" optimization cannot be used with abbreviated keys, since
     637             :      * tie-breaker comparisons may be required.  Typically, the optimization
     638             :      * is only of value to pass-by-value types anyway, whereas abbreviated
     639             :      * keys are typically only of value to pass-by-reference types.
     640             :      */
     641       59804 :     if (!base->sortKeys->abbrev_converter)
     642       59092 :         base->onlyKey = base->sortKeys;
     643             : 
     644       59804 :     MemoryContextSwitchTo(oldcontext);
     645             : 
     646       59804 :     return state;
     647             : }
     648             : 
     649             : /*
     650             :  * Accept one tuple while collecting input data for sort.
     651             :  *
     652             :  * Note that the input data is always copied; the caller need not save it.
     653             :  */
     654             : void
     655    11378690 : tuplesort_puttupleslot(Tuplesortstate *state, TupleTableSlot *slot)
     656             : {
     657    11378690 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     658    11378690 :     MemoryContext oldcontext = MemoryContextSwitchTo(base->tuplecontext);
     659    11378690 :     TupleDesc   tupDesc = (TupleDesc) base->arg;
     660             :     SortTuple   stup;
     661             :     MinimalTuple tuple;
     662             :     HeapTupleData htup;
     663             :     Size        tuplen;
     664             : 
     665             :     /* copy the tuple into sort storage */
     666    11378690 :     tuple = ExecCopySlotMinimalTuple(slot);
     667    11378690 :     stup.tuple = tuple;
     668             :     /* set up first-column key value */
     669    11378690 :     htup.t_len = tuple->t_len + MINIMAL_TUPLE_OFFSET;
     670    11378690 :     htup.t_data = (HeapTupleHeader) ((char *) tuple - MINIMAL_TUPLE_OFFSET);
     671    22757380 :     stup.datum1 = heap_getattr(&htup,
     672    11378690 :                                base->sortKeys[0].ssup_attno,
     673             :                                tupDesc,
     674             :                                &stup.isnull1);
     675             : 
     676             :     /* GetMemoryChunkSpace is not supported for bump contexts */
     677    11378690 :     if (TupleSortUseBumpTupleCxt(base->sortopt))
     678     7876286 :         tuplen = MAXALIGN(tuple->t_len);
     679             :     else
     680     3502404 :         tuplen = GetMemoryChunkSpace(tuple);
     681             : 
     682    11378690 :     tuplesort_puttuple_common(state, &stup,
     683    12492916 :                               base->sortKeys->abbrev_converter &&
     684     1114226 :                               !stup.isnull1, tuplen);
     685             : 
     686    11378690 :     MemoryContextSwitchTo(oldcontext);
     687    11378690 : }
     688             : 
     689             : /*
     690             :  * Accept one tuple while collecting input data for sort.
     691             :  *
     692             :  * Note that the input data is always copied; the caller need not save it.
     693             :  */
     694             : void
     695      547388 : tuplesort_putheaptuple(Tuplesortstate *state, HeapTuple tup)
     696             : {
     697             :     SortTuple   stup;
     698      547388 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     699      547388 :     MemoryContext oldcontext = MemoryContextSwitchTo(base->tuplecontext);
     700      547388 :     TuplesortClusterArg *arg = (TuplesortClusterArg *) base->arg;
     701             :     Size        tuplen;
     702             : 
     703             :     /* copy the tuple into sort storage */
     704      547388 :     tup = heap_copytuple(tup);
     705      547388 :     stup.tuple = tup;
     706             : 
     707             :     /*
     708             :      * set up first-column key value, and potentially abbreviate, if it's a
     709             :      * simple column
     710             :      */
     711      547388 :     if (base->haveDatum1)
     712             :     {
     713      545762 :         stup.datum1 = heap_getattr(tup,
     714      545762 :                                    arg->indexInfo->ii_IndexAttrNumbers[0],
     715             :                                    arg->tupDesc,
     716             :                                    &stup.isnull1);
     717             :     }
     718             : 
     719             :     /* GetMemoryChunkSpace is not supported for bump contexts */
     720      547388 :     if (TupleSortUseBumpTupleCxt(base->sortopt))
     721      547388 :         tuplen = MAXALIGN(HEAPTUPLESIZE + tup->t_len);
     722             :     else
     723           0 :         tuplen = GetMemoryChunkSpace(tup);
     724             : 
     725      547388 :     tuplesort_puttuple_common(state, &stup,
     726     1093150 :                               base->haveDatum1 &&
     727      910476 :                               base->sortKeys->abbrev_converter &&
     728      363088 :                               !stup.isnull1, tuplen);
     729             : 
     730      547388 :     MemoryContextSwitchTo(oldcontext);
     731      547388 : }
     732             : 
     733             : /*
     734             :  * Collect one index tuple while collecting input data for sort, building
     735             :  * it from caller-supplied values.
     736             :  */
     737             : void
     738    12886506 : tuplesort_putindextuplevalues(Tuplesortstate *state, Relation rel,
     739             :                               ItemPointer self, const Datum *values,
     740             :                               const bool *isnull)
     741             : {
     742             :     SortTuple   stup;
     743             :     IndexTuple  tuple;
     744    12886506 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     745    12886506 :     TuplesortIndexArg *arg = (TuplesortIndexArg *) base->arg;
     746             :     Size        tuplen;
     747             : 
     748    12886506 :     stup.tuple = index_form_tuple_context(RelationGetDescr(rel), values,
     749             :                                           isnull, base->tuplecontext);
     750    12886506 :     tuple = ((IndexTuple) stup.tuple);
     751    12886506 :     tuple->t_tid = *self;
     752             :     /* set up first-column key value */
     753    25773012 :     stup.datum1 = index_getattr(tuple,
     754             :                                 1,
     755    12886506 :                                 RelationGetDescr(arg->indexRel),
     756             :                                 &stup.isnull1);
     757             : 
     758             :     /* GetMemoryChunkSpace is not supported for bump contexts */
     759    12886506 :     if (TupleSortUseBumpTupleCxt(base->sortopt))
     760    12886506 :         tuplen = MAXALIGN(tuple->t_info & INDEX_SIZE_MASK);
     761             :     else
     762           0 :         tuplen = GetMemoryChunkSpace(tuple);
     763             : 
     764    12886506 :     tuplesort_puttuple_common(state, &stup,
     765    25652012 :                               base->sortKeys &&
     766    15008536 :                               base->sortKeys->abbrev_converter &&
     767     2122030 :                               !stup.isnull1, tuplen);
     768    12886506 : }
     769             : 
     770             : /*
     771             :  * Collect one BRIN tuple while collecting input data for sort.
     772             :  */
     773             : void
     774          40 : tuplesort_putbrintuple(Tuplesortstate *state, BrinTuple *tuple, Size size)
     775             : {
     776             :     SortTuple   stup;
     777             :     BrinSortTuple *bstup;
     778          40 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     779          40 :     MemoryContext oldcontext = MemoryContextSwitchTo(base->tuplecontext);
     780             :     Size        tuplen;
     781             : 
     782             :     /* allocate space for the whole BRIN sort tuple */
     783          40 :     bstup = palloc(BRINSORTTUPLE_SIZE(size));
     784             : 
     785          40 :     bstup->tuplen = size;
     786          40 :     memcpy(&bstup->tuple, tuple, size);
     787             : 
     788          40 :     stup.tuple = bstup;
     789          40 :     stup.datum1 = tuple->bt_blkno;
     790          40 :     stup.isnull1 = false;
     791             : 
     792             :     /* GetMemoryChunkSpace is not supported for bump contexts */
     793          40 :     if (TupleSortUseBumpTupleCxt(base->sortopt))
     794          40 :         tuplen = MAXALIGN(BRINSORTTUPLE_SIZE(size));
     795             :     else
     796           0 :         tuplen = GetMemoryChunkSpace(bstup);
     797             : 
     798          40 :     tuplesort_puttuple_common(state, &stup,
     799          40 :                               base->sortKeys &&
     800          40 :                               base->sortKeys->abbrev_converter &&
     801           0 :                               !stup.isnull1, tuplen);
     802             : 
     803          40 :     MemoryContextSwitchTo(oldcontext);
     804          40 : }
     805             : 
     806             : /*
     807             :  * Accept one Datum while collecting input data for sort.
     808             :  *
     809             :  * If the Datum is pass-by-ref type, the value will be copied.
     810             :  */
     811             : void
     812     3803450 : tuplesort_putdatum(Tuplesortstate *state, Datum val, bool isNull)
     813             : {
     814     3803450 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     815     3803450 :     MemoryContext oldcontext = MemoryContextSwitchTo(base->tuplecontext);
     816     3803450 :     TuplesortDatumArg *arg = (TuplesortDatumArg *) base->arg;
     817             :     SortTuple   stup;
     818             : 
     819             :     /*
     820             :      * Pass-by-value types or null values are just stored directly in
     821             :      * stup.datum1 (and stup.tuple is not used and set to NULL).
     822             :      *
     823             :      * Non-null pass-by-reference values need to be copied into memory we
     824             :      * control, and possibly abbreviated. The copied value is pointed to by
     825             :      * stup.tuple and is treated as the canonical copy (e.g. to return via
     826             :      * tuplesort_getdatum or when writing to tape); stup.datum1 gets the
     827             :      * abbreviated value if abbreviation is happening, otherwise it's
     828             :      * identical to stup.tuple.
     829             :      */
     830             : 
     831     3803450 :     if (isNull || !base->tuples)
     832             :     {
     833             :         /*
     834             :          * Set datum1 to zeroed representation for NULLs (to be consistent,
     835             :          * and to support cheap inequality tests for NULL abbreviated keys).
     836             :          */
     837     2067036 :         stup.datum1 = !isNull ? val : (Datum) 0;
     838     2067036 :         stup.isnull1 = isNull;
     839     2067036 :         stup.tuple = NULL;      /* no separate storage */
     840             :     }
     841             :     else
     842             :     {
     843     1736414 :         stup.isnull1 = false;
     844     1736414 :         stup.datum1 = datumCopy(val, false, arg->datumTypeLen);
     845     1736414 :         stup.tuple = DatumGetPointer(stup.datum1);
     846             :     }
     847             : 
     848     3803450 :     tuplesort_puttuple_common(state, &stup,
     849     5540618 :                               base->tuples &&
     850     3803450 :                               base->sortKeys->abbrev_converter && !isNull, 0);
     851             : 
     852     3803450 :     MemoryContextSwitchTo(oldcontext);
     853     3803450 : }
     854             : 
     855             : /*
     856             :  * Fetch the next tuple in either forward or back direction.
     857             :  * If successful, put tuple in slot and return true; else, clear the slot
     858             :  * and return false.
     859             :  *
     860             :  * Caller may optionally be passed back abbreviated value (on true return
     861             :  * value) when abbreviation was used, which can be used to cheaply avoid
     862             :  * equality checks that might otherwise be required.  Caller can safely make a
     863             :  * determination of "non-equal tuple" based on simple binary inequality.  A
     864             :  * NULL value in leading attribute will set abbreviated value to zeroed
     865             :  * representation, which caller may rely on in abbreviated inequality check.
     866             :  *
     867             :  * If copy is true, the slot receives a tuple that's been copied into the
     868             :  * caller's memory context, so that it will stay valid regardless of future
     869             :  * manipulations of the tuplesort's state (up to and including deleting the
     870             :  * tuplesort).  If copy is false, the slot will just receive a pointer to a
     871             :  * tuple held within the tuplesort, which is more efficient, but only safe for
     872             :  * callers that are prepared to have any subsequent manipulation of the
     873             :  * tuplesort's state invalidate slot contents.
     874             :  */
     875             : bool
     876     9625144 : tuplesort_gettupleslot(Tuplesortstate *state, bool forward, bool copy,
     877             :                        TupleTableSlot *slot, Datum *abbrev)
     878             : {
     879     9625144 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     880     9625144 :     MemoryContext oldcontext = MemoryContextSwitchTo(base->sortcontext);
     881             :     SortTuple   stup;
     882             : 
     883     9625144 :     if (!tuplesort_gettuple_common(state, forward, &stup))
     884      107238 :         stup.tuple = NULL;
     885             : 
     886     9625144 :     MemoryContextSwitchTo(oldcontext);
     887             : 
     888     9625144 :     if (stup.tuple)
     889             :     {
     890             :         /* Record abbreviated key for caller */
     891     9517906 :         if (base->sortKeys->abbrev_converter && abbrev)
     892           0 :             *abbrev = stup.datum1;
     893             : 
     894     9517906 :         if (copy)
     895        4520 :             stup.tuple = heap_copy_minimal_tuple((MinimalTuple) stup.tuple);
     896             : 
     897     9517906 :         ExecStoreMinimalTuple((MinimalTuple) stup.tuple, slot, copy);
     898     9517906 :         return true;
     899             :     }
     900             :     else
     901             :     {
     902      107238 :         ExecClearTuple(slot);
     903      107238 :         return false;
     904             :     }
     905             : }
     906             : 
     907             : /*
     908             :  * Fetch the next tuple in either forward or back direction.
     909             :  * Returns NULL if no more tuples.  Returned tuple belongs to tuplesort memory
     910             :  * context, and must not be freed by caller.  Caller may not rely on tuple
     911             :  * remaining valid after any further manipulation of tuplesort.
     912             :  */
     913             : HeapTuple
     914      547500 : tuplesort_getheaptuple(Tuplesortstate *state, bool forward)
     915             : {
     916      547500 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     917      547500 :     MemoryContext oldcontext = MemoryContextSwitchTo(base->sortcontext);
     918             :     SortTuple   stup;
     919             : 
     920      547500 :     if (!tuplesort_gettuple_common(state, forward, &stup))
     921         112 :         stup.tuple = NULL;
     922             : 
     923      547500 :     MemoryContextSwitchTo(oldcontext);
     924             : 
     925      547500 :     return stup.tuple;
     926             : }
     927             : 
     928             : /*
     929             :  * Fetch the next index tuple in either forward or back direction.
     930             :  * Returns NULL if no more tuples.  Returned tuple belongs to tuplesort memory
     931             :  * context, and must not be freed by caller.  Caller may not rely on tuple
     932             :  * remaining valid after any further manipulation of tuplesort.
     933             :  */
     934             : IndexTuple
     935    12934412 : tuplesort_getindextuple(Tuplesortstate *state, bool forward)
     936             : {
     937    12934412 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     938    12934412 :     MemoryContext oldcontext = MemoryContextSwitchTo(base->sortcontext);
     939             :     SortTuple   stup;
     940             : 
     941    12934412 :     if (!tuplesort_gettuple_common(state, forward, &stup))
     942       48284 :         stup.tuple = NULL;
     943             : 
     944    12934412 :     MemoryContextSwitchTo(oldcontext);
     945             : 
     946    12934412 :     return (IndexTuple) stup.tuple;
     947             : }
     948             : 
     949             : /*
     950             :  * Fetch the next BRIN tuple in either forward or back direction.
     951             :  * Returns NULL if no more tuples.  Returned tuple belongs to tuplesort memory
     952             :  * context, and must not be freed by caller.  Caller may not rely on tuple
     953             :  * remaining valid after any further manipulation of tuplesort.
     954             :  */
     955             : BrinTuple *
     956          48 : tuplesort_getbrintuple(Tuplesortstate *state, Size *len, bool forward)
     957             : {
     958          48 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
     959          48 :     MemoryContext oldcontext = MemoryContextSwitchTo(base->sortcontext);
     960             :     SortTuple   stup;
     961             :     BrinSortTuple *btup;
     962             : 
     963          48 :     if (!tuplesort_gettuple_common(state, forward, &stup))
     964           8 :         stup.tuple = NULL;
     965             : 
     966          48 :     MemoryContextSwitchTo(oldcontext);
     967             : 
     968          48 :     if (!stup.tuple)
     969           8 :         return NULL;
     970             : 
     971          40 :     btup = (BrinSortTuple *) stup.tuple;
     972             : 
     973          40 :     *len = btup->tuplen;
     974             : 
     975          40 :     return &btup->tuple;
     976             : }
     977             : 
     978             : /*
     979             :  * Fetch the next Datum in either forward or back direction.
     980             :  * Returns false if no more datums.
     981             :  *
     982             :  * If the Datum is pass-by-ref type, the returned value is freshly palloc'd
     983             :  * in caller's context, and is now owned by the caller (this differs from
     984             :  * similar routines for other types of tuplesorts).
     985             :  *
     986             :  * Caller may optionally be passed back abbreviated value (on true return
     987             :  * value) when abbreviation was used, which can be used to cheaply avoid
     988             :  * equality checks that might otherwise be required.  Caller can safely make a
     989             :  * determination of "non-equal tuple" based on simple binary inequality.  A
     990             :  * NULL value will have a zeroed abbreviated value representation, which caller
     991             :  * may rely on in abbreviated inequality check.
     992             :  *
     993             :  * For byref Datums, if copy is true, *val is set to a copy of the Datum
     994             :  * copied into the caller's memory context, so that it will stay valid
     995             :  * regardless of future manipulations of the tuplesort's state (up to and
     996             :  * including deleting the tuplesort).  If copy is false, *val will just be
     997             :  * set to a pointer to the Datum held within the tuplesort, which is more
     998             :  * efficient, but only safe for callers that are prepared to have any
     999             :  * subsequent manipulation of the tuplesort's state invalidate slot contents.
    1000             :  * For byval Datums, the value of the 'copy' parameter has no effect.
    1001             : 
    1002             :  */
    1003             : bool
    1004     2433968 : tuplesort_getdatum(Tuplesortstate *state, bool forward, bool copy,
    1005             :                    Datum *val, bool *isNull, Datum *abbrev)
    1006             : {
    1007     2433968 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1008     2433968 :     MemoryContext oldcontext = MemoryContextSwitchTo(base->sortcontext);
    1009     2433968 :     TuplesortDatumArg *arg = (TuplesortDatumArg *) base->arg;
    1010             :     SortTuple   stup;
    1011             : 
    1012     2433968 :     if (!tuplesort_gettuple_common(state, forward, &stup))
    1013             :     {
    1014       58654 :         MemoryContextSwitchTo(oldcontext);
    1015       58654 :         return false;
    1016             :     }
    1017             : 
    1018             :     /* Ensure we copy into caller's memory context */
    1019     2375314 :     MemoryContextSwitchTo(oldcontext);
    1020             : 
    1021             :     /* Record abbreviated key for caller */
    1022     2375314 :     if (base->sortKeys->abbrev_converter && abbrev)
    1023       55024 :         *abbrev = stup.datum1;
    1024             : 
    1025     2375314 :     if (stup.isnull1 || !base->tuples)
    1026             :     {
    1027     1239086 :         *val = stup.datum1;
    1028     1239086 :         *isNull = stup.isnull1;
    1029             :     }
    1030             :     else
    1031             :     {
    1032             :         /* use stup.tuple because stup.datum1 may be an abbreviation */
    1033     1136228 :         if (copy)
    1034       60048 :             *val = datumCopy(PointerGetDatum(stup.tuple), false,
    1035             :                              arg->datumTypeLen);
    1036             :         else
    1037     1076180 :             *val = PointerGetDatum(stup.tuple);
    1038     1136228 :         *isNull = false;
    1039             :     }
    1040             : 
    1041     2375314 :     return true;
    1042             : }
    1043             : 
    1044             : 
    1045             : /*
    1046             :  * Routines specialized for HeapTuple (actually MinimalTuple) case
    1047             :  */
    1048             : 
    1049             : static void
    1050          12 : removeabbrev_heap(Tuplesortstate *state, SortTuple *stups, int count)
    1051             : {
    1052             :     int         i;
    1053          12 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1054             : 
    1055      122892 :     for (i = 0; i < count; i++)
    1056             :     {
    1057             :         HeapTupleData htup;
    1058             : 
    1059      122880 :         htup.t_len = ((MinimalTuple) stups[i].tuple)->t_len +
    1060             :             MINIMAL_TUPLE_OFFSET;
    1061      122880 :         htup.t_data = (HeapTupleHeader) ((char *) stups[i].tuple -
    1062             :                                          MINIMAL_TUPLE_OFFSET);
    1063      122880 :         stups[i].datum1 = heap_getattr(&htup,
    1064      122880 :                                        base->sortKeys[0].ssup_attno,
    1065      122880 :                                        (TupleDesc) base->arg,
    1066      122880 :                                        &stups[i].isnull1);
    1067             :     }
    1068          12 : }
    1069             : 
    1070             : static int
    1071    41091484 : comparetup_heap(const SortTuple *a, const SortTuple *b, Tuplesortstate *state)
    1072             : {
    1073    41091484 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1074    41091484 :     SortSupport sortKey = base->sortKeys;
    1075             :     int32       compare;
    1076             : 
    1077             : 
    1078             :     /* Compare the leading sort key */
    1079    41091484 :     compare = ApplySortComparator(a->datum1, a->isnull1,
    1080    41091484 :                                   b->datum1, b->isnull1,
    1081             :                                   sortKey);
    1082    41091484 :     if (compare != 0)
    1083    16891654 :         return compare;
    1084             : 
    1085             :     /* Compare additional sort keys */
    1086    24199830 :     return comparetup_heap_tiebreak(a, b, state);
    1087             : }
    1088             : 
    1089             : static int
    1090    27327522 : comparetup_heap_tiebreak(const SortTuple *a, const SortTuple *b, Tuplesortstate *state)
    1091             : {
    1092    27327522 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1093    27327522 :     SortSupport sortKey = base->sortKeys;
    1094             :     HeapTupleData ltup;
    1095             :     HeapTupleData rtup;
    1096             :     TupleDesc   tupDesc;
    1097             :     int         nkey;
    1098             :     int32       compare;
    1099             :     AttrNumber  attno;
    1100             :     Datum       datum1,
    1101             :                 datum2;
    1102             :     bool        isnull1,
    1103             :                 isnull2;
    1104             : 
    1105    27327522 :     ltup.t_len = ((MinimalTuple) a->tuple)->t_len + MINIMAL_TUPLE_OFFSET;
    1106    27327522 :     ltup.t_data = (HeapTupleHeader) ((char *) a->tuple - MINIMAL_TUPLE_OFFSET);
    1107    27327522 :     rtup.t_len = ((MinimalTuple) b->tuple)->t_len + MINIMAL_TUPLE_OFFSET;
    1108    27327522 :     rtup.t_data = (HeapTupleHeader) ((char *) b->tuple - MINIMAL_TUPLE_OFFSET);
    1109    27327522 :     tupDesc = (TupleDesc) base->arg;
    1110             : 
    1111    27327522 :     if (sortKey->abbrev_converter)
    1112             :     {
    1113     1058984 :         attno = sortKey->ssup_attno;
    1114             : 
    1115     1058984 :         datum1 = heap_getattr(&ltup, attno, tupDesc, &isnull1);
    1116     1058984 :         datum2 = heap_getattr(&rtup, attno, tupDesc, &isnull2);
    1117             : 
    1118     1058984 :         compare = ApplySortAbbrevFullComparator(datum1, isnull1,
    1119             :                                                 datum2, isnull2,
    1120             :                                                 sortKey);
    1121     1058984 :         if (compare != 0)
    1122      887300 :             return compare;
    1123             :     }
    1124             : 
    1125    26440222 :     sortKey++;
    1126    28121714 :     for (nkey = 1; nkey < base->nKeys; nkey++, sortKey++)
    1127             :     {
    1128    26659902 :         attno = sortKey->ssup_attno;
    1129             : 
    1130    26659902 :         datum1 = heap_getattr(&ltup, attno, tupDesc, &isnull1);
    1131    26659902 :         datum2 = heap_getattr(&rtup, attno, tupDesc, &isnull2);
    1132             : 
    1133    26659902 :         compare = ApplySortComparator(datum1, isnull1,
    1134             :                                       datum2, isnull2,
    1135             :                                       sortKey);
    1136    26659902 :         if (compare != 0)
    1137    24978410 :             return compare;
    1138             :     }
    1139             : 
    1140     1461812 :     return 0;
    1141             : }
    1142             : 
    1143             : static void
    1144     1077450 : writetup_heap(Tuplesortstate *state, LogicalTape *tape, SortTuple *stup)
    1145             : {
    1146     1077450 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1147     1077450 :     MinimalTuple tuple = (MinimalTuple) stup->tuple;
    1148             : 
    1149             :     /* the part of the MinimalTuple we'll write: */
    1150     1077450 :     char       *tupbody = (char *) tuple + MINIMAL_TUPLE_DATA_OFFSET;
    1151     1077450 :     unsigned int tupbodylen = tuple->t_len - MINIMAL_TUPLE_DATA_OFFSET;
    1152             : 
    1153             :     /* total on-disk footprint: */
    1154     1077450 :     unsigned int tuplen = tupbodylen + sizeof(int);
    1155             : 
    1156     1077450 :     LogicalTapeWrite(tape, &tuplen, sizeof(tuplen));
    1157     1077450 :     LogicalTapeWrite(tape, tupbody, tupbodylen);
    1158     1077450 :     if (base->sortopt & TUPLESORT_RANDOMACCESS) /* need trailing length word? */
    1159       30000 :         LogicalTapeWrite(tape, &tuplen, sizeof(tuplen));
    1160     1077450 : }
    1161             : 
    1162             : static void
    1163      975180 : readtup_heap(Tuplesortstate *state, SortTuple *stup,
    1164             :              LogicalTape *tape, unsigned int len)
    1165             : {
    1166      975180 :     unsigned int tupbodylen = len - sizeof(int);
    1167      975180 :     unsigned int tuplen = tupbodylen + MINIMAL_TUPLE_DATA_OFFSET;
    1168      975180 :     MinimalTuple tuple = (MinimalTuple) tuplesort_readtup_alloc(state, tuplen);
    1169      975180 :     char       *tupbody = (char *) tuple + MINIMAL_TUPLE_DATA_OFFSET;
    1170      975180 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1171             :     HeapTupleData htup;
    1172             : 
    1173             :     /* read in the tuple proper */
    1174      975180 :     tuple->t_len = tuplen;
    1175      975180 :     LogicalTapeReadExact(tape, tupbody, tupbodylen);
    1176      975180 :     if (base->sortopt & TUPLESORT_RANDOMACCESS) /* need trailing length word? */
    1177       47784 :         LogicalTapeReadExact(tape, &tuplen, sizeof(tuplen));
    1178      975180 :     stup->tuple = tuple;
    1179             :     /* set up first-column key value */
    1180      975180 :     htup.t_len = tuple->t_len + MINIMAL_TUPLE_OFFSET;
    1181      975180 :     htup.t_data = (HeapTupleHeader) ((char *) tuple - MINIMAL_TUPLE_OFFSET);
    1182     1950360 :     stup->datum1 = heap_getattr(&htup,
    1183      975180 :                                 base->sortKeys[0].ssup_attno,
    1184      975180 :                                 (TupleDesc) base->arg,
    1185             :                                 &stup->isnull1);
    1186      975180 : }
    1187             : 
    1188             : /*
    1189             :  * Routines specialized for the CLUSTER case (HeapTuple data, with
    1190             :  * comparisons per a btree index definition)
    1191             :  */
    1192             : 
    1193             : static void
    1194          12 : removeabbrev_cluster(Tuplesortstate *state, SortTuple *stups, int count)
    1195             : {
    1196             :     int         i;
    1197          12 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1198          12 :     TuplesortClusterArg *arg = (TuplesortClusterArg *) base->arg;
    1199             : 
    1200      122892 :     for (i = 0; i < count; i++)
    1201             :     {
    1202             :         HeapTuple   tup;
    1203             : 
    1204      122880 :         tup = (HeapTuple) stups[i].tuple;
    1205      122880 :         stups[i].datum1 = heap_getattr(tup,
    1206      122880 :                                        arg->indexInfo->ii_IndexAttrNumbers[0],
    1207             :                                        arg->tupDesc,
    1208      122880 :                                        &stups[i].isnull1);
    1209             :     }
    1210          12 : }
    1211             : 
    1212             : static int
    1213     5997234 : comparetup_cluster(const SortTuple *a, const SortTuple *b,
    1214             :                    Tuplesortstate *state)
    1215             : {
    1216     5997234 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1217     5997234 :     SortSupport sortKey = base->sortKeys;
    1218             :     int32       compare;
    1219             : 
    1220             :     /* Compare the leading sort key, if it's simple */
    1221     5997234 :     if (base->haveDatum1)
    1222             :     {
    1223     5985444 :         compare = ApplySortComparator(a->datum1, a->isnull1,
    1224     5985444 :                                       b->datum1, b->isnull1,
    1225             :                                       sortKey);
    1226     5985444 :         if (compare != 0)
    1227     5844206 :             return compare;
    1228             :     }
    1229             : 
    1230      153028 :     return comparetup_cluster_tiebreak(a, b, state);
    1231             : }
    1232             : 
    1233             : static int
    1234      587268 : comparetup_cluster_tiebreak(const SortTuple *a, const SortTuple *b,
    1235             :                             Tuplesortstate *state)
    1236             : {
    1237      587268 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1238      587268 :     TuplesortClusterArg *arg = (TuplesortClusterArg *) base->arg;
    1239      587268 :     SortSupport sortKey = base->sortKeys;
    1240             :     HeapTuple   ltup;
    1241             :     HeapTuple   rtup;
    1242             :     TupleDesc   tupDesc;
    1243             :     int         nkey;
    1244      587268 :     int32       compare = 0;
    1245             :     Datum       datum1,
    1246             :                 datum2;
    1247             :     bool        isnull1,
    1248             :                 isnull2;
    1249             : 
    1250      587268 :     ltup = (HeapTuple) a->tuple;
    1251      587268 :     rtup = (HeapTuple) b->tuple;
    1252      587268 :     tupDesc = arg->tupDesc;
    1253             : 
    1254             :     /* Compare the leading sort key, if it's simple */
    1255      587268 :     if (base->haveDatum1)
    1256             :     {
    1257      575478 :         if (sortKey->abbrev_converter)
    1258             :         {
    1259      144452 :             AttrNumber  leading = arg->indexInfo->ii_IndexAttrNumbers[0];
    1260             : 
    1261      144452 :             datum1 = heap_getattr(ltup, leading, tupDesc, &isnull1);
    1262      144452 :             datum2 = heap_getattr(rtup, leading, tupDesc, &isnull2);
    1263             : 
    1264      144452 :             compare = ApplySortAbbrevFullComparator(datum1, isnull1,
    1265             :                                                     datum2, isnull2,
    1266             :                                                     sortKey);
    1267             :         }
    1268      575478 :         if (compare != 0 || base->nKeys == 1)
    1269      147918 :             return compare;
    1270             :         /* Compare additional columns the hard way */
    1271      427560 :         sortKey++;
    1272      427560 :         nkey = 1;
    1273             :     }
    1274             :     else
    1275             :     {
    1276             :         /* Must compare all keys the hard way */
    1277       11790 :         nkey = 0;
    1278             :     }
    1279             : 
    1280      439350 :     if (arg->indexInfo->ii_Expressions == NULL)
    1281             :     {
    1282             :         /* If not expression index, just compare the proper heap attrs */
    1283             : 
    1284      593520 :         for (; nkey < base->nKeys; nkey++, sortKey++)
    1285             :         {
    1286      593520 :             AttrNumber  attno = arg->indexInfo->ii_IndexAttrNumbers[nkey];
    1287             : 
    1288      593520 :             datum1 = heap_getattr(ltup, attno, tupDesc, &isnull1);
    1289      593520 :             datum2 = heap_getattr(rtup, attno, tupDesc, &isnull2);
    1290             : 
    1291      593520 :             compare = ApplySortComparator(datum1, isnull1,
    1292             :                                           datum2, isnull2,
    1293             :                                           sortKey);
    1294      593520 :             if (compare != 0)
    1295      427560 :                 return compare;
    1296             :         }
    1297             :     }
    1298             :     else
    1299             :     {
    1300             :         /*
    1301             :          * In the expression index case, compute the whole index tuple and
    1302             :          * then compare values.  It would perhaps be faster to compute only as
    1303             :          * many columns as we need to compare, but that would require
    1304             :          * duplicating all the logic in FormIndexDatum.
    1305             :          */
    1306             :         Datum       l_index_values[INDEX_MAX_KEYS];
    1307             :         bool        l_index_isnull[INDEX_MAX_KEYS];
    1308             :         Datum       r_index_values[INDEX_MAX_KEYS];
    1309             :         bool        r_index_isnull[INDEX_MAX_KEYS];
    1310             :         TupleTableSlot *ecxt_scantuple;
    1311             : 
    1312             :         /* Reset context each time to prevent memory leakage */
    1313       11790 :         ResetPerTupleExprContext(arg->estate);
    1314             : 
    1315       11790 :         ecxt_scantuple = GetPerTupleExprContext(arg->estate)->ecxt_scantuple;
    1316             : 
    1317       11790 :         ExecStoreHeapTuple(ltup, ecxt_scantuple, false);
    1318       11790 :         FormIndexDatum(arg->indexInfo, ecxt_scantuple, arg->estate,
    1319             :                        l_index_values, l_index_isnull);
    1320             : 
    1321       11790 :         ExecStoreHeapTuple(rtup, ecxt_scantuple, false);
    1322       11790 :         FormIndexDatum(arg->indexInfo, ecxt_scantuple, arg->estate,
    1323             :                        r_index_values, r_index_isnull);
    1324             : 
    1325       12714 :         for (; nkey < base->nKeys; nkey++, sortKey++)
    1326             :         {
    1327       12714 :             compare = ApplySortComparator(l_index_values[nkey],
    1328       12714 :                                           l_index_isnull[nkey],
    1329             :                                           r_index_values[nkey],
    1330       12714 :                                           r_index_isnull[nkey],
    1331             :                                           sortKey);
    1332       12714 :             if (compare != 0)
    1333       11790 :                 return compare;
    1334             :         }
    1335             :     }
    1336             : 
    1337           0 :     return 0;
    1338             : }
    1339             : 
    1340             : static void
    1341       60000 : writetup_cluster(Tuplesortstate *state, LogicalTape *tape, SortTuple *stup)
    1342             : {
    1343       60000 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1344       60000 :     HeapTuple   tuple = (HeapTuple) stup->tuple;
    1345       60000 :     unsigned int tuplen = tuple->t_len + sizeof(ItemPointerData) + sizeof(int);
    1346             : 
    1347             :     /* We need to store t_self, but not other fields of HeapTupleData */
    1348       60000 :     LogicalTapeWrite(tape, &tuplen, sizeof(tuplen));
    1349       60000 :     LogicalTapeWrite(tape, &tuple->t_self, sizeof(ItemPointerData));
    1350       60000 :     LogicalTapeWrite(tape, tuple->t_data, tuple->t_len);
    1351       60000 :     if (base->sortopt & TUPLESORT_RANDOMACCESS) /* need trailing length word? */
    1352           0 :         LogicalTapeWrite(tape, &tuplen, sizeof(tuplen));
    1353       60000 : }
    1354             : 
    1355             : static void
    1356       60000 : readtup_cluster(Tuplesortstate *state, SortTuple *stup,
    1357             :                 LogicalTape *tape, unsigned int tuplen)
    1358             : {
    1359       60000 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1360       60000 :     TuplesortClusterArg *arg = (TuplesortClusterArg *) base->arg;
    1361       60000 :     unsigned int t_len = tuplen - sizeof(ItemPointerData) - sizeof(int);
    1362       60000 :     HeapTuple   tuple = (HeapTuple) tuplesort_readtup_alloc(state,
    1363             :                                                             t_len + HEAPTUPLESIZE);
    1364             : 
    1365             :     /* Reconstruct the HeapTupleData header */
    1366       60000 :     tuple->t_data = (HeapTupleHeader) ((char *) tuple + HEAPTUPLESIZE);
    1367       60000 :     tuple->t_len = t_len;
    1368       60000 :     LogicalTapeReadExact(tape, &tuple->t_self, sizeof(ItemPointerData));
    1369             :     /* We don't currently bother to reconstruct t_tableOid */
    1370       60000 :     tuple->t_tableOid = InvalidOid;
    1371             :     /* Read in the tuple body */
    1372       60000 :     LogicalTapeReadExact(tape, tuple->t_data, tuple->t_len);
    1373       60000 :     if (base->sortopt & TUPLESORT_RANDOMACCESS) /* need trailing length word? */
    1374           0 :         LogicalTapeReadExact(tape, &tuplen, sizeof(tuplen));
    1375       60000 :     stup->tuple = tuple;
    1376             :     /* set up first-column key value, if it's a simple column */
    1377       60000 :     if (base->haveDatum1)
    1378       60000 :         stup->datum1 = heap_getattr(tuple,
    1379       60000 :                                     arg->indexInfo->ii_IndexAttrNumbers[0],
    1380             :                                     arg->tupDesc,
    1381             :                                     &stup->isnull1);
    1382       60000 : }
    1383             : 
    1384             : static void
    1385         112 : freestate_cluster(Tuplesortstate *state)
    1386             : {
    1387         112 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1388         112 :     TuplesortClusterArg *arg = (TuplesortClusterArg *) base->arg;
    1389             : 
    1390             :     /* Free any execution state created for CLUSTER case */
    1391         112 :     if (arg->estate != NULL)
    1392             :     {
    1393          24 :         ExprContext *econtext = GetPerTupleExprContext(arg->estate);
    1394             : 
    1395          24 :         ExecDropSingleTupleTableSlot(econtext->ecxt_scantuple);
    1396          24 :         FreeExecutorState(arg->estate);
    1397             :     }
    1398         112 : }
    1399             : 
    1400             : /*
    1401             :  * Routines specialized for IndexTuple case
    1402             :  *
    1403             :  * The btree and hash cases require separate comparison functions, but the
    1404             :  * IndexTuple representation is the same so the copy/write/read support
    1405             :  * functions can be shared.
    1406             :  */
    1407             : 
    1408             : static void
    1409          60 : removeabbrev_index(Tuplesortstate *state, SortTuple *stups, int count)
    1410             : {
    1411          60 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1412          60 :     TuplesortIndexArg *arg = (TuplesortIndexArg *) base->arg;
    1413             :     int         i;
    1414             : 
    1415      614460 :     for (i = 0; i < count; i++)
    1416             :     {
    1417             :         IndexTuple  tuple;
    1418             : 
    1419      614400 :         tuple = stups[i].tuple;
    1420      614400 :         stups[i].datum1 = index_getattr(tuple,
    1421             :                                         1,
    1422      614400 :                                         RelationGetDescr(arg->indexRel),
    1423      614400 :                                         &stups[i].isnull1);
    1424             :     }
    1425          60 : }
    1426             : 
    1427             : static int
    1428    55377154 : comparetup_index_btree(const SortTuple *a, const SortTuple *b,
    1429             :                        Tuplesortstate *state)
    1430             : {
    1431             :     /*
    1432             :      * This is similar to comparetup_heap(), but expects index tuples.  There
    1433             :      * is also special handling for enforcing uniqueness, and special
    1434             :      * treatment for equal keys at the end.
    1435             :      */
    1436    55377154 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1437    55377154 :     SortSupport sortKey = base->sortKeys;
    1438             :     int32       compare;
    1439             : 
    1440             :     /* Compare the leading sort key */
    1441    55377154 :     compare = ApplySortComparator(a->datum1, a->isnull1,
    1442    55377154 :                                   b->datum1, b->isnull1,
    1443             :                                   sortKey);
    1444    55377154 :     if (compare != 0)
    1445    50187272 :         return compare;
    1446             : 
    1447             :     /* Compare additional sort keys */
    1448     5189882 :     return comparetup_index_btree_tiebreak(a, b, state);
    1449             : }
    1450             : 
    1451             : static int
    1452    17558072 : comparetup_index_btree_tiebreak(const SortTuple *a, const SortTuple *b,
    1453             :                                 Tuplesortstate *state)
    1454             : {
    1455    17558072 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1456    17558072 :     TuplesortIndexBTreeArg *arg = (TuplesortIndexBTreeArg *) base->arg;
    1457    17558072 :     SortSupport sortKey = base->sortKeys;
    1458             :     IndexTuple  tuple1;
    1459             :     IndexTuple  tuple2;
    1460             :     int         keysz;
    1461             :     TupleDesc   tupDes;
    1462    17558072 :     bool        equal_hasnull = false;
    1463             :     int         nkey;
    1464             :     int32       compare;
    1465             :     Datum       datum1,
    1466             :                 datum2;
    1467             :     bool        isnull1,
    1468             :                 isnull2;
    1469             : 
    1470    17558072 :     tuple1 = (IndexTuple) a->tuple;
    1471    17558072 :     tuple2 = (IndexTuple) b->tuple;
    1472    17558072 :     keysz = base->nKeys;
    1473    17558072 :     tupDes = RelationGetDescr(arg->index.indexRel);
    1474             : 
    1475    17558072 :     if (sortKey->abbrev_converter)
    1476             :     {
    1477      859806 :         datum1 = index_getattr(tuple1, 1, tupDes, &isnull1);
    1478      859806 :         datum2 = index_getattr(tuple2, 1, tupDes, &isnull2);
    1479             : 
    1480      859806 :         compare = ApplySortAbbrevFullComparator(datum1, isnull1,
    1481             :                                                 datum2, isnull2,
    1482             :                                                 sortKey);
    1483      859806 :         if (compare != 0)
    1484      802606 :             return compare;
    1485             :     }
    1486             : 
    1487             :     /* they are equal, so we only need to examine one null flag */
    1488    16755466 :     if (a->isnull1)
    1489         426 :         equal_hasnull = true;
    1490             : 
    1491    16755466 :     sortKey++;
    1492    19227606 :     for (nkey = 2; nkey <= keysz; nkey++, sortKey++)
    1493             :     {
    1494     7110322 :         datum1 = index_getattr(tuple1, nkey, tupDes, &isnull1);
    1495     7110322 :         datum2 = index_getattr(tuple2, nkey, tupDes, &isnull2);
    1496             : 
    1497     7110322 :         compare = ApplySortComparator(datum1, isnull1,
    1498             :                                       datum2, isnull2,
    1499             :                                       sortKey);
    1500     7110322 :         if (compare != 0)
    1501     4638182 :             return compare;     /* done when we find unequal attributes */
    1502             : 
    1503             :         /* they are equal, so we only need to examine one null flag */
    1504     2472140 :         if (isnull1)
    1505       25992 :             equal_hasnull = true;
    1506             :     }
    1507             : 
    1508             :     /*
    1509             :      * If btree has asked us to enforce uniqueness, complain if two equal
    1510             :      * tuples are detected (unless there was at least one NULL field and NULLS
    1511             :      * NOT DISTINCT was not set).
    1512             :      *
    1513             :      * It is sufficient to make the test here, because if two tuples are equal
    1514             :      * they *must* get compared at some stage of the sort --- otherwise the
    1515             :      * sort algorithm wouldn't have checked whether one must appear before the
    1516             :      * other.
    1517             :      */
    1518    12117284 :     if (arg->enforceUnique && !(!arg->uniqueNullsNotDistinct && equal_hasnull))
    1519             :     {
    1520             :         Datum       values[INDEX_MAX_KEYS];
    1521             :         bool        isnull[INDEX_MAX_KEYS];
    1522             :         char       *key_desc;
    1523             : 
    1524             :         /*
    1525             :          * Some rather brain-dead implementations of qsort (such as the one in
    1526             :          * QNX 4) will sometimes call the comparison routine to compare a
    1527             :          * value to itself, but we always use our own implementation, which
    1528             :          * does not.
    1529             :          */
    1530             :         Assert(tuple1 != tuple2);
    1531             : 
    1532          84 :         index_deform_tuple(tuple1, tupDes, values, isnull);
    1533             : 
    1534          84 :         key_desc = BuildIndexValueDescription(arg->index.indexRel, values, isnull);
    1535             : 
    1536          84 :         ereport(ERROR,
    1537             :                 (errcode(ERRCODE_UNIQUE_VIOLATION),
    1538             :                  errmsg("could not create unique index \"%s\"",
    1539             :                         RelationGetRelationName(arg->index.indexRel)),
    1540             :                  key_desc ? errdetail("Key %s is duplicated.", key_desc) :
    1541             :                  errdetail("Duplicate keys exist."),
    1542             :                  errtableconstraint(arg->index.heapRel,
    1543             :                                     RelationGetRelationName(arg->index.indexRel))));
    1544             :     }
    1545             : 
    1546             :     /*
    1547             :      * If key values are equal, we sort on ItemPointer.  This is required for
    1548             :      * btree indexes, since heap TID is treated as an implicit last key
    1549             :      * attribute in order to ensure that all keys in the index are physically
    1550             :      * unique.
    1551             :      */
    1552             :     {
    1553    12117200 :         BlockNumber blk1 = ItemPointerGetBlockNumber(&tuple1->t_tid);
    1554    12117200 :         BlockNumber blk2 = ItemPointerGetBlockNumber(&tuple2->t_tid);
    1555             : 
    1556    12117200 :         if (blk1 != blk2)
    1557    11295982 :             return (blk1 < blk2) ? -1 : 1;
    1558             :     }
    1559             :     {
    1560      821218 :         OffsetNumber pos1 = ItemPointerGetOffsetNumber(&tuple1->t_tid);
    1561      821218 :         OffsetNumber pos2 = ItemPointerGetOffsetNumber(&tuple2->t_tid);
    1562             : 
    1563      821218 :         if (pos1 != pos2)
    1564      821218 :             return (pos1 < pos2) ? -1 : 1;
    1565             :     }
    1566             : 
    1567             :     /* ItemPointer values should never be equal */
    1568             :     Assert(false);
    1569             : 
    1570           0 :     return 0;
    1571             : }
    1572             : 
    1573             : static int
    1574     1827924 : comparetup_index_hash(const SortTuple *a, const SortTuple *b,
    1575             :                       Tuplesortstate *state)
    1576             : {
    1577             :     Bucket      bucket1;
    1578             :     Bucket      bucket2;
    1579             :     uint32      hash1;
    1580             :     uint32      hash2;
    1581             :     IndexTuple  tuple1;
    1582             :     IndexTuple  tuple2;
    1583     1827924 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1584     1827924 :     TuplesortIndexHashArg *arg = (TuplesortIndexHashArg *) base->arg;
    1585             : 
    1586             :     /*
    1587             :      * Fetch hash keys and mask off bits we don't want to sort by, so that the
    1588             :      * initial sort is just on the bucket number.  We know that the first
    1589             :      * column of the index tuple is the hash key.
    1590             :      */
    1591             :     Assert(!a->isnull1);
    1592     1827924 :     bucket1 = _hash_hashkey2bucket(DatumGetUInt32(a->datum1),
    1593             :                                    arg->max_buckets, arg->high_mask,
    1594             :                                    arg->low_mask);
    1595             :     Assert(!b->isnull1);
    1596     1827924 :     bucket2 = _hash_hashkey2bucket(DatumGetUInt32(b->datum1),
    1597             :                                    arg->max_buckets, arg->high_mask,
    1598             :                                    arg->low_mask);
    1599     1827924 :     if (bucket1 > bucket2)
    1600      540974 :         return 1;
    1601     1286950 :     else if (bucket1 < bucket2)
    1602      507834 :         return -1;
    1603             : 
    1604             :     /*
    1605             :      * If bucket values are equal, sort by hash values.  This allows us to
    1606             :      * insert directly onto bucket/overflow pages, where the index tuples are
    1607             :      * stored in hash order to allow fast binary search within each page.
    1608             :      */
    1609      779116 :     hash1 = DatumGetUInt32(a->datum1);
    1610      779116 :     hash2 = DatumGetUInt32(b->datum1);
    1611      779116 :     if (hash1 > hash2)
    1612      195016 :         return 1;
    1613      584100 :     else if (hash1 < hash2)
    1614      174182 :         return -1;
    1615             : 
    1616             :     /*
    1617             :      * If hash values are equal, we sort on ItemPointer.  This does not affect
    1618             :      * validity of the finished index, but it may be useful to have index
    1619             :      * scans in physical order.
    1620             :      */
    1621      409918 :     tuple1 = (IndexTuple) a->tuple;
    1622      409918 :     tuple2 = (IndexTuple) b->tuple;
    1623             : 
    1624             :     {
    1625      409918 :         BlockNumber blk1 = ItemPointerGetBlockNumber(&tuple1->t_tid);
    1626      409918 :         BlockNumber blk2 = ItemPointerGetBlockNumber(&tuple2->t_tid);
    1627             : 
    1628      409918 :         if (blk1 != blk2)
    1629      273086 :             return (blk1 < blk2) ? -1 : 1;
    1630             :     }
    1631             :     {
    1632      136832 :         OffsetNumber pos1 = ItemPointerGetOffsetNumber(&tuple1->t_tid);
    1633      136832 :         OffsetNumber pos2 = ItemPointerGetOffsetNumber(&tuple2->t_tid);
    1634             : 
    1635      136832 :         if (pos1 != pos2)
    1636      136832 :             return (pos1 < pos2) ? -1 : 1;
    1637             :     }
    1638             : 
    1639             :     /* ItemPointer values should never be equal */
    1640             :     Assert(false);
    1641             : 
    1642           0 :     return 0;
    1643             : }
    1644             : 
    1645             : /*
    1646             :  * Sorting for hash indexes only uses one sort key, so this shouldn't ever be
    1647             :  * called. It's only here for consistency.
    1648             :  */
    1649             : static int
    1650           0 : comparetup_index_hash_tiebreak(const SortTuple *a, const SortTuple *b,
    1651             :                                Tuplesortstate *state)
    1652             : {
    1653             :     Assert(false);
    1654             : 
    1655           0 :     return 0;
    1656             : }
    1657             : 
    1658             : static void
    1659     3700012 : writetup_index(Tuplesortstate *state, LogicalTape *tape, SortTuple *stup)
    1660             : {
    1661     3700012 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1662     3700012 :     IndexTuple  tuple = (IndexTuple) stup->tuple;
    1663             :     unsigned int tuplen;
    1664             : 
    1665     3700012 :     tuplen = IndexTupleSize(tuple) + sizeof(tuplen);
    1666     3700012 :     LogicalTapeWrite(tape, &tuplen, sizeof(tuplen));
    1667     3700012 :     LogicalTapeWrite(tape, tuple, IndexTupleSize(tuple));
    1668     3700012 :     if (base->sortopt & TUPLESORT_RANDOMACCESS) /* need trailing length word? */
    1669           0 :         LogicalTapeWrite(tape, &tuplen, sizeof(tuplen));
    1670     3700012 : }
    1671             : 
    1672             : static void
    1673     3700012 : readtup_index(Tuplesortstate *state, SortTuple *stup,
    1674             :               LogicalTape *tape, unsigned int len)
    1675             : {
    1676     3700012 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1677     3700012 :     TuplesortIndexArg *arg = (TuplesortIndexArg *) base->arg;
    1678     3700012 :     unsigned int tuplen = len - sizeof(unsigned int);
    1679     3700012 :     IndexTuple  tuple = (IndexTuple) tuplesort_readtup_alloc(state, tuplen);
    1680             : 
    1681     3700012 :     LogicalTapeReadExact(tape, tuple, tuplen);
    1682     3700012 :     if (base->sortopt & TUPLESORT_RANDOMACCESS) /* need trailing length word? */
    1683           0 :         LogicalTapeReadExact(tape, &tuplen, sizeof(tuplen));
    1684     3700012 :     stup->tuple = tuple;
    1685             :     /* set up first-column key value */
    1686     7400024 :     stup->datum1 = index_getattr(tuple,
    1687             :                                  1,
    1688     3700012 :                                  RelationGetDescr(arg->indexRel),
    1689             :                                  &stup->isnull1);
    1690     3700012 : }
    1691             : 
    1692             : /*
    1693             :  * Routines specialized for BrinTuple case
    1694             :  */
    1695             : 
    1696             : static void
    1697           0 : removeabbrev_index_brin(Tuplesortstate *state, SortTuple *stups, int count)
    1698             : {
    1699             :     int         i;
    1700             : 
    1701           0 :     for (i = 0; i < count; i++)
    1702             :     {
    1703             :         BrinSortTuple *tuple;
    1704             : 
    1705           0 :         tuple = stups[i].tuple;
    1706           0 :         stups[i].datum1 = tuple->tuple.bt_blkno;
    1707             :     }
    1708           0 : }
    1709             : 
    1710             : static int
    1711          38 : comparetup_index_brin(const SortTuple *a, const SortTuple *b,
    1712             :                       Tuplesortstate *state)
    1713             : {
    1714             :     Assert(TuplesortstateGetPublic(state)->haveDatum1);
    1715             : 
    1716          38 :     if (DatumGetUInt32(a->datum1) > DatumGetUInt32(b->datum1))
    1717           0 :         return 1;
    1718             : 
    1719          38 :     if (DatumGetUInt32(a->datum1) < DatumGetUInt32(b->datum1))
    1720          38 :         return -1;
    1721             : 
    1722             :     /* silence compilers */
    1723           0 :     return 0;
    1724             : }
    1725             : 
    1726             : static void
    1727          40 : writetup_index_brin(Tuplesortstate *state, LogicalTape *tape, SortTuple *stup)
    1728             : {
    1729          40 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1730          40 :     BrinSortTuple *tuple = (BrinSortTuple *) stup->tuple;
    1731          40 :     unsigned int tuplen = tuple->tuplen;
    1732             : 
    1733          40 :     tuplen = tuplen + sizeof(tuplen);
    1734          40 :     LogicalTapeWrite(tape, &tuplen, sizeof(tuplen));
    1735          40 :     LogicalTapeWrite(tape, &tuple->tuple, tuple->tuplen);
    1736          40 :     if (base->sortopt & TUPLESORT_RANDOMACCESS) /* need trailing length word? */
    1737           0 :         LogicalTapeWrite(tape, &tuplen, sizeof(tuplen));
    1738          40 : }
    1739             : 
    1740             : static void
    1741          40 : readtup_index_brin(Tuplesortstate *state, SortTuple *stup,
    1742             :                    LogicalTape *tape, unsigned int len)
    1743             : {
    1744             :     BrinSortTuple *tuple;
    1745          40 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1746          40 :     unsigned int tuplen = len - sizeof(unsigned int);
    1747             : 
    1748             :     /*
    1749             :      * Allocate space for the BRIN sort tuple, which is BrinTuple with an
    1750             :      * extra length field.
    1751             :      */
    1752          40 :     tuple = (BrinSortTuple *) tuplesort_readtup_alloc(state,
    1753             :                                                       BRINSORTTUPLE_SIZE(tuplen));
    1754             : 
    1755          40 :     tuple->tuplen = tuplen;
    1756             : 
    1757          40 :     LogicalTapeReadExact(tape, &tuple->tuple, tuplen);
    1758          40 :     if (base->sortopt & TUPLESORT_RANDOMACCESS) /* need trailing length word? */
    1759           0 :         LogicalTapeReadExact(tape, &tuplen, sizeof(tuplen));
    1760          40 :     stup->tuple = tuple;
    1761             : 
    1762             :     /* set up first-column key value, which is block number */
    1763          40 :     stup->datum1 = tuple->tuple.bt_blkno;
    1764          40 : }
    1765             : 
    1766             : /*
    1767             :  * Routines specialized for DatumTuple case
    1768             :  */
    1769             : 
    1770             : static void
    1771          12 : removeabbrev_datum(Tuplesortstate *state, SortTuple *stups, int count)
    1772             : {
    1773             :     int         i;
    1774             : 
    1775      122892 :     for (i = 0; i < count; i++)
    1776      122880 :         stups[i].datum1 = PointerGetDatum(stups[i].tuple);
    1777          12 : }
    1778             : 
    1779             : static int
    1780     6792562 : comparetup_datum(const SortTuple *a, const SortTuple *b, Tuplesortstate *state)
    1781             : {
    1782     6792562 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1783             :     int         compare;
    1784             : 
    1785     6792562 :     compare = ApplySortComparator(a->datum1, a->isnull1,
    1786     6792562 :                                   b->datum1, b->isnull1,
    1787             :                                   base->sortKeys);
    1788     6792562 :     if (compare != 0)
    1789     6592280 :         return compare;
    1790             : 
    1791      200282 :     return comparetup_datum_tiebreak(a, b, state);
    1792             : }
    1793             : 
    1794             : static int
    1795     2698292 : comparetup_datum_tiebreak(const SortTuple *a, const SortTuple *b, Tuplesortstate *state)
    1796             : {
    1797     2698292 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1798     2698292 :     int32       compare = 0;
    1799             : 
    1800             :     /* if we have abbreviations, then "tuple" has the original value */
    1801     2698292 :     if (base->sortKeys->abbrev_converter)
    1802     2498040 :         compare = ApplySortAbbrevFullComparator(PointerGetDatum(a->tuple), a->isnull1,
    1803     2498040 :                                                 PointerGetDatum(b->tuple), b->isnull1,
    1804             :                                                 base->sortKeys);
    1805             : 
    1806     2698292 :     return compare;
    1807             : }
    1808             : 
    1809             : static void
    1810     1080522 : writetup_datum(Tuplesortstate *state, LogicalTape *tape, SortTuple *stup)
    1811             : {
    1812     1080522 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1813     1080522 :     TuplesortDatumArg *arg = (TuplesortDatumArg *) base->arg;
    1814             :     void       *waddr;
    1815             :     unsigned int tuplen;
    1816             :     unsigned int writtenlen;
    1817             : 
    1818     1080522 :     if (stup->isnull1)
    1819             :     {
    1820          66 :         waddr = NULL;
    1821          66 :         tuplen = 0;
    1822             :     }
    1823     1080456 :     else if (!base->tuples)
    1824             :     {
    1825      300132 :         waddr = &stup->datum1;
    1826      300132 :         tuplen = sizeof(Datum);
    1827             :     }
    1828             :     else
    1829             :     {
    1830      780324 :         waddr = stup->tuple;
    1831      780324 :         tuplen = datumGetSize(PointerGetDatum(stup->tuple), false, arg->datumTypeLen);
    1832             :         Assert(tuplen != 0);
    1833             :     }
    1834             : 
    1835     1080522 :     writtenlen = tuplen + sizeof(unsigned int);
    1836             : 
    1837     1080522 :     LogicalTapeWrite(tape, &writtenlen, sizeof(writtenlen));
    1838     1080522 :     LogicalTapeWrite(tape, waddr, tuplen);
    1839     1080522 :     if (base->sortopt & TUPLESORT_RANDOMACCESS) /* need trailing length word? */
    1840      480264 :         LogicalTapeWrite(tape, &writtenlen, sizeof(writtenlen));
    1841     1080522 : }
    1842             : 
    1843             : static void
    1844      960552 : readtup_datum(Tuplesortstate *state, SortTuple *stup,
    1845             :               LogicalTape *tape, unsigned int len)
    1846             : {
    1847      960552 :     TuplesortPublic *base = TuplesortstateGetPublic(state);
    1848      960552 :     unsigned int tuplen = len - sizeof(unsigned int);
    1849             : 
    1850      960552 :     if (tuplen == 0)
    1851             :     {
    1852             :         /* it's NULL */
    1853          90 :         stup->datum1 = (Datum) 0;
    1854          90 :         stup->isnull1 = true;
    1855          90 :         stup->tuple = NULL;
    1856             :     }
    1857      960462 :     else if (!base->tuples)
    1858             :     {
    1859             :         Assert(tuplen == sizeof(Datum));
    1860      300138 :         LogicalTapeReadExact(tape, &stup->datum1, tuplen);
    1861      300138 :         stup->isnull1 = false;
    1862      300138 :         stup->tuple = NULL;
    1863             :     }
    1864             :     else
    1865             :     {
    1866      660324 :         void       *raddr = tuplesort_readtup_alloc(state, tuplen);
    1867             : 
    1868      660324 :         LogicalTapeReadExact(tape, raddr, tuplen);
    1869      660324 :         stup->datum1 = PointerGetDatum(raddr);
    1870      660324 :         stup->isnull1 = false;
    1871      660324 :         stup->tuple = raddr;
    1872             :     }
    1873             : 
    1874      960552 :     if (base->sortopt & TUPLESORT_RANDOMACCESS) /* need trailing length word? */
    1875      480312 :         LogicalTapeReadExact(tape, &tuplen, sizeof(tuplen));
    1876      960552 : }

Generated by: LCOV version 1.14