LCOV - code coverage report
Current view: top level - src/backend/utils/adt - orderedsetaggs.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 432 488 88.5 %
Date: 2025-01-18 05:15:39 Functions: 20 23 87.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * orderedsetaggs.c
       4             :  *      Ordered-set aggregate functions.
       5             :  *
       6             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       7             :  * Portions Copyright (c) 1994, Regents of the University of California
       8             :  *
       9             :  *
      10             :  * IDENTIFICATION
      11             :  *    src/backend/utils/adt/orderedsetaggs.c
      12             :  *
      13             :  *-------------------------------------------------------------------------
      14             :  */
      15             : #include "postgres.h"
      16             : 
      17             : #include <math.h>
      18             : 
      19             : #include "catalog/pg_aggregate.h"
      20             : #include "catalog/pg_operator.h"
      21             : #include "catalog/pg_type.h"
      22             : #include "executor/executor.h"
      23             : #include "miscadmin.h"
      24             : #include "nodes/nodeFuncs.h"
      25             : #include "optimizer/optimizer.h"
      26             : #include "utils/array.h"
      27             : #include "utils/fmgrprotos.h"
      28             : #include "utils/lsyscache.h"
      29             : #include "utils/tuplesort.h"
      30             : 
      31             : 
      32             : /*
      33             :  * Generic support for ordered-set aggregates
      34             :  *
      35             :  * The state for an ordered-set aggregate is divided into a per-group struct
      36             :  * (which is the internal-type transition state datum returned to nodeAgg.c)
      37             :  * and a per-query struct, which contains data and sub-objects that we can
      38             :  * create just once per query because they will not change across groups.
      39             :  * The per-query struct and subsidiary data live in the executor's per-query
      40             :  * memory context, and go away implicitly at ExecutorEnd().
      41             :  *
      42             :  * These structs are set up during the first call of the transition function.
      43             :  * Because we allow nodeAgg.c to merge ordered-set aggregates (but not
      44             :  * hypothetical aggregates) with identical inputs and transition functions,
      45             :  * this info must not depend on the particular aggregate (ie, particular
      46             :  * final-function), nor on the direct argument(s) of the aggregate.
      47             :  */
      48             : 
      49             : typedef struct OSAPerQueryState
      50             : {
      51             :     /* Representative Aggref for this aggregate: */
      52             :     Aggref     *aggref;
      53             :     /* Memory context containing this struct and other per-query data: */
      54             :     MemoryContext qcontext;
      55             :     /* Context for expression evaluation */
      56             :     ExprContext *econtext;
      57             :     /* Do we expect multiple final-function calls within one group? */
      58             :     bool        rescan_needed;
      59             : 
      60             :     /* These fields are used only when accumulating tuples: */
      61             : 
      62             :     /* Tuple descriptor for tuples inserted into sortstate: */
      63             :     TupleDesc   tupdesc;
      64             :     /* Tuple slot we can use for inserting/extracting tuples: */
      65             :     TupleTableSlot *tupslot;
      66             :     /* Per-sort-column sorting information */
      67             :     int         numSortCols;
      68             :     AttrNumber *sortColIdx;
      69             :     Oid        *sortOperators;
      70             :     Oid        *eqOperators;
      71             :     Oid        *sortCollations;
      72             :     bool       *sortNullsFirsts;
      73             :     /* Equality operator call info, created only if needed: */
      74             :     ExprState  *compareTuple;
      75             : 
      76             :     /* These fields are used only when accumulating datums: */
      77             : 
      78             :     /* Info about datatype of datums being sorted: */
      79             :     Oid         sortColType;
      80             :     int16       typLen;
      81             :     bool        typByVal;
      82             :     char        typAlign;
      83             :     /* Info about sort ordering: */
      84             :     Oid         sortOperator;
      85             :     Oid         eqOperator;
      86             :     Oid         sortCollation;
      87             :     bool        sortNullsFirst;
      88             :     /* Equality operator call info, created only if needed: */
      89             :     FmgrInfo    equalfn;
      90             : } OSAPerQueryState;
      91             : 
      92             : typedef struct OSAPerGroupState
      93             : {
      94             :     /* Link to the per-query state for this aggregate: */
      95             :     OSAPerQueryState *qstate;
      96             :     /* Memory context containing per-group data: */
      97             :     MemoryContext gcontext;
      98             :     /* Sort object we're accumulating data in: */
      99             :     Tuplesortstate *sortstate;
     100             :     /* Number of normal rows inserted into sortstate: */
     101             :     int64       number_of_rows;
     102             :     /* Have we already done tuplesort_performsort? */
     103             :     bool        sort_done;
     104             : } OSAPerGroupState;
     105             : 
     106             : static void ordered_set_shutdown(Datum arg);
     107             : 
     108             : 
     109             : /*
     110             :  * Set up working state for an ordered-set aggregate
     111             :  */
     112             : static OSAPerGroupState *
     113         660 : ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples)
     114             : {
     115             :     OSAPerGroupState *osastate;
     116             :     OSAPerQueryState *qstate;
     117             :     MemoryContext gcontext;
     118             :     MemoryContext oldcontext;
     119             :     int         tuplesortopt;
     120             : 
     121             :     /*
     122             :      * Check we're called as aggregate (and not a window function), and get
     123             :      * the Agg node's group-lifespan context (which might change from group to
     124             :      * group, so we shouldn't cache it in the per-query state).
     125             :      */
     126         660 :     if (AggCheckCallContext(fcinfo, &gcontext) != AGG_CONTEXT_AGGREGATE)
     127           0 :         elog(ERROR, "ordered-set aggregate called in non-aggregate context");
     128             : 
     129             :     /*
     130             :      * We keep a link to the per-query state in fn_extra; if it's not there,
     131             :      * create it, and do the per-query setup we need.
     132             :      */
     133         660 :     qstate = (OSAPerQueryState *) fcinfo->flinfo->fn_extra;
     134         660 :     if (qstate == NULL)
     135             :     {
     136             :         Aggref     *aggref;
     137             :         MemoryContext qcontext;
     138             :         List       *sortlist;
     139             :         int         numSortCols;
     140             : 
     141             :         /* Get the Aggref so we can examine aggregate's arguments */
     142         246 :         aggref = AggGetAggref(fcinfo);
     143         246 :         if (!aggref)
     144           0 :             elog(ERROR, "ordered-set aggregate called in non-aggregate context");
     145         246 :         if (!AGGKIND_IS_ORDERED_SET(aggref->aggkind))
     146           0 :             elog(ERROR, "ordered-set aggregate support function called for non-ordered-set aggregate");
     147             : 
     148             :         /*
     149             :          * Prepare per-query structures in the fn_mcxt, which we assume is the
     150             :          * executor's per-query context; in any case it's the right place to
     151             :          * keep anything found via fn_extra.
     152             :          */
     153         246 :         qcontext = fcinfo->flinfo->fn_mcxt;
     154         246 :         oldcontext = MemoryContextSwitchTo(qcontext);
     155             : 
     156         246 :         qstate = (OSAPerQueryState *) palloc0(sizeof(OSAPerQueryState));
     157         246 :         qstate->aggref = aggref;
     158         246 :         qstate->qcontext = qcontext;
     159             : 
     160             :         /* We need to support rescans if the trans state is shared */
     161         246 :         qstate->rescan_needed = AggStateIsShared(fcinfo);
     162             : 
     163             :         /* Extract the sort information */
     164         246 :         sortlist = aggref->aggorder;
     165         246 :         numSortCols = list_length(sortlist);
     166             : 
     167         246 :         if (use_tuples)
     168             :         {
     169         100 :             bool        ishypothetical = (aggref->aggkind == AGGKIND_HYPOTHETICAL);
     170             :             ListCell   *lc;
     171             :             int         i;
     172             : 
     173         100 :             if (ishypothetical)
     174         100 :                 numSortCols++;  /* make space for flag column */
     175         100 :             qstate->numSortCols = numSortCols;
     176         100 :             qstate->sortColIdx = (AttrNumber *) palloc(numSortCols * sizeof(AttrNumber));
     177         100 :             qstate->sortOperators = (Oid *) palloc(numSortCols * sizeof(Oid));
     178         100 :             qstate->eqOperators = (Oid *) palloc(numSortCols * sizeof(Oid));
     179         100 :             qstate->sortCollations = (Oid *) palloc(numSortCols * sizeof(Oid));
     180         100 :             qstate->sortNullsFirsts = (bool *) palloc(numSortCols * sizeof(bool));
     181             : 
     182         100 :             i = 0;
     183         250 :             foreach(lc, sortlist)
     184             :             {
     185         150 :                 SortGroupClause *sortcl = (SortGroupClause *) lfirst(lc);
     186         150 :                 TargetEntry *tle = get_sortgroupclause_tle(sortcl,
     187             :                                                            aggref->args);
     188             : 
     189             :                 /* the parser should have made sure of this */
     190             :                 Assert(OidIsValid(sortcl->sortop));
     191             : 
     192         150 :                 qstate->sortColIdx[i] = tle->resno;
     193         150 :                 qstate->sortOperators[i] = sortcl->sortop;
     194         150 :                 qstate->eqOperators[i] = sortcl->eqop;
     195         150 :                 qstate->sortCollations[i] = exprCollation((Node *) tle->expr);
     196         150 :                 qstate->sortNullsFirsts[i] = sortcl->nulls_first;
     197         150 :                 i++;
     198             :             }
     199             : 
     200         100 :             if (ishypothetical)
     201             :             {
     202             :                 /* Add an integer flag column as the last sort column */
     203         100 :                 qstate->sortColIdx[i] = list_length(aggref->args) + 1;
     204         100 :                 qstate->sortOperators[i] = Int4LessOperator;
     205         100 :                 qstate->eqOperators[i] = Int4EqualOperator;
     206         100 :                 qstate->sortCollations[i] = InvalidOid;
     207         100 :                 qstate->sortNullsFirsts[i] = false;
     208         100 :                 i++;
     209             :             }
     210             : 
     211             :             Assert(i == numSortCols);
     212             : 
     213             :             /*
     214             :              * Get a tupledesc corresponding to the aggregated inputs
     215             :              * (including sort expressions) of the agg.
     216             :              */
     217         100 :             qstate->tupdesc = ExecTypeFromTL(aggref->args);
     218             : 
     219             :             /* If we need a flag column, hack the tupledesc to include that */
     220         100 :             if (ishypothetical)
     221             :             {
     222             :                 TupleDesc   newdesc;
     223         100 :                 int         natts = qstate->tupdesc->natts;
     224             : 
     225         100 :                 newdesc = CreateTemplateTupleDesc(natts + 1);
     226         250 :                 for (i = 1; i <= natts; i++)
     227         150 :                     TupleDescCopyEntry(newdesc, i, qstate->tupdesc, i);
     228             : 
     229         100 :                 TupleDescInitEntry(newdesc,
     230         100 :                                    (AttrNumber) ++natts,
     231             :                                    "flag",
     232             :                                    INT4OID,
     233             :                                    -1,
     234             :                                    0);
     235             : 
     236         100 :                 FreeTupleDesc(qstate->tupdesc);
     237         100 :                 qstate->tupdesc = newdesc;
     238             :             }
     239             : 
     240             :             /* Create slot we'll use to store/retrieve rows */
     241         100 :             qstate->tupslot = MakeSingleTupleTableSlot(qstate->tupdesc,
     242             :                                                        &TTSOpsMinimalTuple);
     243             :         }
     244             :         else
     245             :         {
     246             :             /* Sort single datums */
     247             :             SortGroupClause *sortcl;
     248             :             TargetEntry *tle;
     249             : 
     250         146 :             if (numSortCols != 1 || aggref->aggkind == AGGKIND_HYPOTHETICAL)
     251           0 :                 elog(ERROR, "ordered-set aggregate support function does not support multiple aggregated columns");
     252             : 
     253         146 :             sortcl = (SortGroupClause *) linitial(sortlist);
     254         146 :             tle = get_sortgroupclause_tle(sortcl, aggref->args);
     255             : 
     256             :             /* the parser should have made sure of this */
     257             :             Assert(OidIsValid(sortcl->sortop));
     258             : 
     259             :             /* Save sort ordering info */
     260         146 :             qstate->sortColType = exprType((Node *) tle->expr);
     261         146 :             qstate->sortOperator = sortcl->sortop;
     262         146 :             qstate->eqOperator = sortcl->eqop;
     263         146 :             qstate->sortCollation = exprCollation((Node *) tle->expr);
     264         146 :             qstate->sortNullsFirst = sortcl->nulls_first;
     265             : 
     266             :             /* Save datatype info */
     267         146 :             get_typlenbyvalalign(qstate->sortColType,
     268             :                                  &qstate->typLen,
     269             :                                  &qstate->typByVal,
     270             :                                  &qstate->typAlign);
     271             :         }
     272             : 
     273         246 :         fcinfo->flinfo->fn_extra = qstate;
     274             : 
     275         246 :         MemoryContextSwitchTo(oldcontext);
     276             :     }
     277             : 
     278             :     /* Now build the stuff we need in group-lifespan context */
     279         660 :     oldcontext = MemoryContextSwitchTo(gcontext);
     280             : 
     281         660 :     osastate = (OSAPerGroupState *) palloc(sizeof(OSAPerGroupState));
     282         660 :     osastate->qstate = qstate;
     283         660 :     osastate->gcontext = gcontext;
     284             : 
     285         660 :     tuplesortopt = TUPLESORT_NONE;
     286             : 
     287         660 :     if (qstate->rescan_needed)
     288          24 :         tuplesortopt |= TUPLESORT_RANDOMACCESS;
     289             : 
     290             :     /*
     291             :      * Initialize tuplesort object.
     292             :      */
     293         660 :     if (use_tuples)
     294         274 :         osastate->sortstate = tuplesort_begin_heap(qstate->tupdesc,
     295             :                                                    qstate->numSortCols,
     296             :                                                    qstate->sortColIdx,
     297             :                                                    qstate->sortOperators,
     298             :                                                    qstate->sortCollations,
     299             :                                                    qstate->sortNullsFirsts,
     300             :                                                    work_mem,
     301             :                                                    NULL,
     302             :                                                    tuplesortopt);
     303             :     else
     304         386 :         osastate->sortstate = tuplesort_begin_datum(qstate->sortColType,
     305             :                                                     qstate->sortOperator,
     306             :                                                     qstate->sortCollation,
     307         386 :                                                     qstate->sortNullsFirst,
     308             :                                                     work_mem,
     309             :                                                     NULL,
     310             :                                                     tuplesortopt);
     311             : 
     312         660 :     osastate->number_of_rows = 0;
     313         660 :     osastate->sort_done = false;
     314             : 
     315             :     /* Now register a shutdown callback to clean things up at end of group */
     316         660 :     AggRegisterCallback(fcinfo,
     317             :                         ordered_set_shutdown,
     318             :                         PointerGetDatum(osastate));
     319             : 
     320         660 :     MemoryContextSwitchTo(oldcontext);
     321             : 
     322         660 :     return osastate;
     323             : }
     324             : 
     325             : /*
     326             :  * Clean up when evaluation of an ordered-set aggregate is complete.
     327             :  *
     328             :  * We don't need to bother freeing objects in the per-group memory context,
     329             :  * since that will get reset anyway by nodeAgg.c; nor should we free anything
     330             :  * in the per-query context, which will get cleared (if this was the last
     331             :  * group) by ExecutorEnd.  But we must take care to release any potential
     332             :  * non-memory resources.
     333             :  *
     334             :  * In the case where we're not expecting multiple finalfn calls, we could
     335             :  * arguably rely on the finalfn to clean up; but it's easier and more testable
     336             :  * if we just do it the same way in either case.
     337             :  */
     338             : static void
     339         660 : ordered_set_shutdown(Datum arg)
     340             : {
     341         660 :     OSAPerGroupState *osastate = (OSAPerGroupState *) DatumGetPointer(arg);
     342             : 
     343             :     /* Tuplesort object might have temp files. */
     344         660 :     if (osastate->sortstate)
     345         660 :         tuplesort_end(osastate->sortstate);
     346         660 :     osastate->sortstate = NULL;
     347             :     /* The tupleslot probably can't be holding a pin, but let's be safe. */
     348         660 :     if (osastate->qstate->tupslot)
     349         274 :         ExecClearTuple(osastate->qstate->tupslot);
     350         660 : }
     351             : 
     352             : 
     353             : /*
     354             :  * Generic transition function for ordered-set aggregates
     355             :  * with a single input column in which we want to suppress nulls
     356             :  */
     357             : Datum
     358     1203698 : ordered_set_transition(PG_FUNCTION_ARGS)
     359             : {
     360             :     OSAPerGroupState *osastate;
     361             : 
     362             :     /* If first call, create the transition state workspace */
     363     1203698 :     if (PG_ARGISNULL(0))
     364         386 :         osastate = ordered_set_startup(fcinfo, false);
     365             :     else
     366     1203312 :         osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
     367             : 
     368             :     /* Load the datum into the tuplesort object, but only if it's not null */
     369     1203698 :     if (!PG_ARGISNULL(1))
     370             :     {
     371     1203626 :         tuplesort_putdatum(osastate->sortstate, PG_GETARG_DATUM(1), false);
     372     1203626 :         osastate->number_of_rows++;
     373             :     }
     374             : 
     375     1203698 :     PG_RETURN_POINTER(osastate);
     376             : }
     377             : 
     378             : /*
     379             :  * Generic transition function for ordered-set aggregates
     380             :  * with (potentially) multiple aggregated input columns
     381             :  */
     382             : Datum
     383      302788 : ordered_set_transition_multi(PG_FUNCTION_ARGS)
     384             : {
     385             :     OSAPerGroupState *osastate;
     386             :     TupleTableSlot *slot;
     387             :     int         nargs;
     388             :     int         i;
     389             : 
     390             :     /* If first call, create the transition state workspace */
     391      302788 :     if (PG_ARGISNULL(0))
     392         274 :         osastate = ordered_set_startup(fcinfo, true);
     393             :     else
     394      302514 :         osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
     395             : 
     396             :     /* Form a tuple from all the other inputs besides the transition value */
     397      302788 :     slot = osastate->qstate->tupslot;
     398      302788 :     ExecClearTuple(slot);
     399      302788 :     nargs = PG_NARGS() - 1;
     400     1206226 :     for (i = 0; i < nargs; i++)
     401             :     {
     402      903438 :         slot->tts_values[i] = PG_GETARG_DATUM(i + 1);
     403      903438 :         slot->tts_isnull[i] = PG_ARGISNULL(i + 1);
     404             :     }
     405      302788 :     if (osastate->qstate->aggref->aggkind == AGGKIND_HYPOTHETICAL)
     406             :     {
     407             :         /* Add a zero flag value to mark this row as a normal input row */
     408      302788 :         slot->tts_values[i] = Int32GetDatum(0);
     409      302788 :         slot->tts_isnull[i] = false;
     410      302788 :         i++;
     411             :     }
     412             :     Assert(i == slot->tts_tupleDescriptor->natts);
     413      302788 :     ExecStoreVirtualTuple(slot);
     414             : 
     415             :     /* Load the row into the tuplesort object */
     416      302788 :     tuplesort_puttupleslot(osastate->sortstate, slot);
     417      302788 :     osastate->number_of_rows++;
     418             : 
     419      302788 :     PG_RETURN_POINTER(osastate);
     420             : }
     421             : 
     422             : 
     423             : /*
     424             :  * percentile_disc(float8) within group(anyelement) - discrete percentile
     425             :  */
     426             : Datum
     427         270 : percentile_disc_final(PG_FUNCTION_ARGS)
     428             : {
     429             :     OSAPerGroupState *osastate;
     430             :     double      percentile;
     431             :     Datum       val;
     432             :     bool        isnull;
     433             :     int64       rownum;
     434             : 
     435             :     Assert(AggCheckCallContext(fcinfo, NULL) == AGG_CONTEXT_AGGREGATE);
     436             : 
     437             :     /* Get and check the percentile argument */
     438         270 :     if (PG_ARGISNULL(1))
     439           0 :         PG_RETURN_NULL();
     440             : 
     441         270 :     percentile = PG_GETARG_FLOAT8(1);
     442             : 
     443         270 :     if (percentile < 0 || percentile > 1 || isnan(percentile))
     444           0 :         ereport(ERROR,
     445             :                 (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
     446             :                  errmsg("percentile value %g is not between 0 and 1",
     447             :                         percentile)));
     448             : 
     449             :     /* If there were no regular rows, the result is NULL */
     450         270 :     if (PG_ARGISNULL(0))
     451          54 :         PG_RETURN_NULL();
     452             : 
     453         216 :     osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
     454             : 
     455             :     /* number_of_rows could be zero if we only saw NULL input values */
     456         216 :     if (osastate->number_of_rows == 0)
     457           0 :         PG_RETURN_NULL();
     458             : 
     459             :     /* Finish the sort, or rescan if we already did */
     460         216 :     if (!osastate->sort_done)
     461             :     {
     462         192 :         tuplesort_performsort(osastate->sortstate);
     463         192 :         osastate->sort_done = true;
     464             :     }
     465             :     else
     466          24 :         tuplesort_rescan(osastate->sortstate);
     467             : 
     468             :     /*----------
     469             :      * We need the smallest K such that (K/N) >= percentile.
     470             :      * N>0, therefore K >= N*percentile, therefore K = ceil(N*percentile).
     471             :      * So we skip K-1 rows (if K>0) and return the next row fetched.
     472             :      *----------
     473             :      */
     474         216 :     rownum = (int64) ceil(percentile * osastate->number_of_rows);
     475             :     Assert(rownum <= osastate->number_of_rows);
     476             : 
     477         216 :     if (rownum > 1)
     478             :     {
     479         156 :         if (!tuplesort_skiptuples(osastate->sortstate, rownum - 1, true))
     480           0 :             elog(ERROR, "missing row in percentile_disc");
     481             :     }
     482             : 
     483         216 :     if (!tuplesort_getdatum(osastate->sortstate, true, true, &val, &isnull,
     484             :                             NULL))
     485           0 :         elog(ERROR, "missing row in percentile_disc");
     486             : 
     487             :     /* We shouldn't have stored any nulls, but do the right thing anyway */
     488         216 :     if (isnull)
     489           0 :         PG_RETURN_NULL();
     490             :     else
     491         216 :         PG_RETURN_DATUM(val);
     492             : }
     493             : 
     494             : 
     495             : /*
     496             :  * For percentile_cont, we need a way to interpolate between consecutive
     497             :  * values. Use a helper function for that, so that we can share the rest
     498             :  * of the code between types.
     499             :  */
     500             : typedef Datum (*LerpFunc) (Datum lo, Datum hi, double pct);
     501             : 
     502             : static Datum
     503         132 : float8_lerp(Datum lo, Datum hi, double pct)
     504             : {
     505         132 :     double      loval = DatumGetFloat8(lo);
     506         132 :     double      hival = DatumGetFloat8(hi);
     507             : 
     508         132 :     return Float8GetDatum(loval + (pct * (hival - loval)));
     509             : }
     510             : 
     511             : static Datum
     512           0 : interval_lerp(Datum lo, Datum hi, double pct)
     513             : {
     514           0 :     Datum       diff_result = DirectFunctionCall2(interval_mi, hi, lo);
     515           0 :     Datum       mul_result = DirectFunctionCall2(interval_mul,
     516             :                                                  diff_result,
     517             :                                                  Float8GetDatumFast(pct));
     518             : 
     519           0 :     return DirectFunctionCall2(interval_pl, mul_result, lo);
     520             : }
     521             : 
     522             : /*
     523             :  * Continuous percentile
     524             :  */
     525             : static Datum
     526         104 : percentile_cont_final_common(FunctionCallInfo fcinfo,
     527             :                              Oid expect_type,
     528             :                              LerpFunc lerpfunc)
     529             : {
     530             :     OSAPerGroupState *osastate;
     531             :     double      percentile;
     532         104 :     int64       first_row = 0;
     533         104 :     int64       second_row = 0;
     534             :     Datum       val;
     535             :     Datum       first_val;
     536             :     Datum       second_val;
     537             :     double      proportion;
     538             :     bool        isnull;
     539             : 
     540             :     Assert(AggCheckCallContext(fcinfo, NULL) == AGG_CONTEXT_AGGREGATE);
     541             : 
     542             :     /* Get and check the percentile argument */
     543         104 :     if (PG_ARGISNULL(1))
     544           0 :         PG_RETURN_NULL();
     545             : 
     546         104 :     percentile = PG_GETARG_FLOAT8(1);
     547             : 
     548         104 :     if (percentile < 0 || percentile > 1 || isnan(percentile))
     549           0 :         ereport(ERROR,
     550             :                 (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
     551             :                  errmsg("percentile value %g is not between 0 and 1",
     552             :                         percentile)));
     553             : 
     554             :     /* If there were no regular rows, the result is NULL */
     555         104 :     if (PG_ARGISNULL(0))
     556           0 :         PG_RETURN_NULL();
     557             : 
     558         104 :     osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
     559             : 
     560             :     /* number_of_rows could be zero if we only saw NULL input values */
     561         104 :     if (osastate->number_of_rows == 0)
     562           0 :         PG_RETURN_NULL();
     563             : 
     564             :     Assert(expect_type == osastate->qstate->sortColType);
     565             : 
     566             :     /* Finish the sort, or rescan if we already did */
     567         104 :     if (!osastate->sort_done)
     568             :     {
     569         104 :         tuplesort_performsort(osastate->sortstate);
     570         104 :         osastate->sort_done = true;
     571             :     }
     572             :     else
     573           0 :         tuplesort_rescan(osastate->sortstate);
     574             : 
     575         104 :     first_row = floor(percentile * (osastate->number_of_rows - 1));
     576         104 :     second_row = ceil(percentile * (osastate->number_of_rows - 1));
     577             : 
     578             :     Assert(first_row < osastate->number_of_rows);
     579             : 
     580         104 :     if (!tuplesort_skiptuples(osastate->sortstate, first_row, true))
     581           0 :         elog(ERROR, "missing row in percentile_cont");
     582             : 
     583         104 :     if (!tuplesort_getdatum(osastate->sortstate, true, true, &first_val,
     584             :                             &isnull, NULL))
     585           0 :         elog(ERROR, "missing row in percentile_cont");
     586         104 :     if (isnull)
     587           0 :         PG_RETURN_NULL();
     588             : 
     589         104 :     if (first_row == second_row)
     590             :     {
     591          32 :         val = first_val;
     592             :     }
     593             :     else
     594             :     {
     595          72 :         if (!tuplesort_getdatum(osastate->sortstate, true, true, &second_val,
     596             :                                 &isnull, NULL))
     597           0 :             elog(ERROR, "missing row in percentile_cont");
     598             : 
     599          72 :         if (isnull)
     600           0 :             PG_RETURN_NULL();
     601             : 
     602          72 :         proportion = (percentile * (osastate->number_of_rows - 1)) - first_row;
     603          72 :         val = lerpfunc(first_val, second_val, proportion);
     604             :     }
     605             : 
     606         104 :     PG_RETURN_DATUM(val);
     607             : }
     608             : 
     609             : /*
     610             :  * percentile_cont(float8) within group (float8)    - continuous percentile
     611             :  */
     612             : Datum
     613         104 : percentile_cont_float8_final(PG_FUNCTION_ARGS)
     614             : {
     615         104 :     return percentile_cont_final_common(fcinfo, FLOAT8OID, float8_lerp);
     616             : }
     617             : 
     618             : /*
     619             :  * percentile_cont(float8) within group (interval)  - continuous percentile
     620             :  */
     621             : Datum
     622           0 : percentile_cont_interval_final(PG_FUNCTION_ARGS)
     623             : {
     624           0 :     return percentile_cont_final_common(fcinfo, INTERVALOID, interval_lerp);
     625             : }
     626             : 
     627             : 
     628             : /*
     629             :  * Support code for handling arrays of percentiles
     630             :  *
     631             :  * Note: in each pct_info entry, second_row should be equal to or
     632             :  * exactly one more than first_row.
     633             :  */
     634             : struct pct_info
     635             : {
     636             :     int64       first_row;      /* first row to sample */
     637             :     int64       second_row;     /* possible second row to sample */
     638             :     double      proportion;     /* interpolation fraction */
     639             :     int         idx;            /* index of this item in original array */
     640             : };
     641             : 
     642             : /*
     643             :  * Sort comparator to sort pct_infos by first_row then second_row
     644             :  */
     645             : static int
     646         300 : pct_info_cmp(const void *pa, const void *pb)
     647             : {
     648         300 :     const struct pct_info *a = (const struct pct_info *) pa;
     649         300 :     const struct pct_info *b = (const struct pct_info *) pb;
     650             : 
     651         300 :     if (a->first_row != b->first_row)
     652         258 :         return (a->first_row < b->first_row) ? -1 : 1;
     653          42 :     if (a->second_row != b->second_row)
     654           6 :         return (a->second_row < b->second_row) ? -1 : 1;
     655          36 :     return 0;
     656             : }
     657             : 
     658             : /*
     659             :  * Construct array showing which rows to sample for percentiles.
     660             :  */
     661             : static struct pct_info *
     662          30 : setup_pct_info(int num_percentiles,
     663             :                Datum *percentiles_datum,
     664             :                bool *percentiles_null,
     665             :                int64 rowcount,
     666             :                bool continuous)
     667             : {
     668             :     struct pct_info *pct_info;
     669             :     int         i;
     670             : 
     671          30 :     pct_info = (struct pct_info *) palloc(num_percentiles * sizeof(struct pct_info));
     672             : 
     673         222 :     for (i = 0; i < num_percentiles; i++)
     674             :     {
     675         192 :         pct_info[i].idx = i;
     676             : 
     677         192 :         if (percentiles_null[i])
     678             :         {
     679             :             /* dummy entry for any NULL in array */
     680          12 :             pct_info[i].first_row = 0;
     681          12 :             pct_info[i].second_row = 0;
     682          12 :             pct_info[i].proportion = 0;
     683             :         }
     684             :         else
     685             :         {
     686         180 :             double      p = DatumGetFloat8(percentiles_datum[i]);
     687             : 
     688         180 :             if (p < 0 || p > 1 || isnan(p))
     689           0 :                 ereport(ERROR,
     690             :                         (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
     691             :                          errmsg("percentile value %g is not between 0 and 1",
     692             :                                 p)));
     693             : 
     694         180 :             if (continuous)
     695             :             {
     696          96 :                 pct_info[i].first_row = 1 + floor(p * (rowcount - 1));
     697          96 :                 pct_info[i].second_row = 1 + ceil(p * (rowcount - 1));
     698          96 :                 pct_info[i].proportion = (p * (rowcount - 1)) - floor(p * (rowcount - 1));
     699             :             }
     700             :             else
     701             :             {
     702             :                 /*----------
     703             :                  * We need the smallest K such that (K/N) >= percentile.
     704             :                  * N>0, therefore K >= N*percentile, therefore
     705             :                  * K = ceil(N*percentile); but not less than 1.
     706             :                  *----------
     707             :                  */
     708          84 :                 int64       row = (int64) ceil(p * rowcount);
     709             : 
     710          84 :                 row = Max(1, row);
     711          84 :                 pct_info[i].first_row = row;
     712          84 :                 pct_info[i].second_row = row;
     713          84 :                 pct_info[i].proportion = 0;
     714             :             }
     715             :         }
     716             :     }
     717             : 
     718             :     /*
     719             :      * The parameter array wasn't necessarily in sorted order, but we need to
     720             :      * visit the rows in order, so sort by first_row/second_row.
     721             :      */
     722          30 :     qsort(pct_info, num_percentiles, sizeof(struct pct_info), pct_info_cmp);
     723             : 
     724          30 :     return pct_info;
     725             : }
     726             : 
     727             : /*
     728             :  * percentile_disc(float8[]) within group (anyelement)  - discrete percentiles
     729             :  */
     730             : Datum
     731          18 : percentile_disc_multi_final(PG_FUNCTION_ARGS)
     732             : {
     733             :     OSAPerGroupState *osastate;
     734             :     ArrayType  *param;
     735             :     Datum      *percentiles_datum;
     736             :     bool       *percentiles_null;
     737             :     int         num_percentiles;
     738             :     struct pct_info *pct_info;
     739             :     Datum      *result_datum;
     740             :     bool       *result_isnull;
     741          18 :     int64       rownum = 0;
     742          18 :     Datum       val = (Datum) 0;
     743          18 :     bool        isnull = true;
     744             :     int         i;
     745             : 
     746             :     Assert(AggCheckCallContext(fcinfo, NULL) == AGG_CONTEXT_AGGREGATE);
     747             : 
     748             :     /* If there were no regular rows, the result is NULL */
     749          18 :     if (PG_ARGISNULL(0))
     750           0 :         PG_RETURN_NULL();
     751             : 
     752          18 :     osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
     753             : 
     754             :     /* number_of_rows could be zero if we only saw NULL input values */
     755          18 :     if (osastate->number_of_rows == 0)
     756           0 :         PG_RETURN_NULL();
     757             : 
     758             :     /* Deconstruct the percentile-array input */
     759          18 :     if (PG_ARGISNULL(1))
     760           0 :         PG_RETURN_NULL();
     761          18 :     param = PG_GETARG_ARRAYTYPE_P(1);
     762             : 
     763          18 :     deconstruct_array_builtin(param, FLOAT8OID,
     764             :                               &percentiles_datum,
     765             :                               &percentiles_null,
     766             :                               &num_percentiles);
     767             : 
     768          18 :     if (num_percentiles == 0)
     769           0 :         PG_RETURN_POINTER(construct_empty_array(osastate->qstate->sortColType));
     770             : 
     771          18 :     pct_info = setup_pct_info(num_percentiles,
     772             :                               percentiles_datum,
     773             :                               percentiles_null,
     774             :                               osastate->number_of_rows,
     775             :                               false);
     776             : 
     777          18 :     result_datum = (Datum *) palloc(num_percentiles * sizeof(Datum));
     778          18 :     result_isnull = (bool *) palloc(num_percentiles * sizeof(bool));
     779             : 
     780             :     /*
     781             :      * Start by dealing with any nulls in the param array - those are sorted
     782             :      * to the front on row=0, so set the corresponding result indexes to null
     783             :      */
     784          30 :     for (i = 0; i < num_percentiles; i++)
     785             :     {
     786          30 :         int         idx = pct_info[i].idx;
     787             : 
     788          30 :         if (pct_info[i].first_row > 0)
     789          18 :             break;
     790             : 
     791          12 :         result_datum[idx] = (Datum) 0;
     792          12 :         result_isnull[idx] = true;
     793             :     }
     794             : 
     795             :     /*
     796             :      * If there's anything left after doing the nulls, then grind the input
     797             :      * and extract the needed values
     798             :      */
     799          18 :     if (i < num_percentiles)
     800             :     {
     801             :         /* Finish the sort, or rescan if we already did */
     802          18 :         if (!osastate->sort_done)
     803             :         {
     804          18 :             tuplesort_performsort(osastate->sortstate);
     805          18 :             osastate->sort_done = true;
     806             :         }
     807             :         else
     808           0 :             tuplesort_rescan(osastate->sortstate);
     809             : 
     810         102 :         for (; i < num_percentiles; i++)
     811             :         {
     812          84 :             int64       target_row = pct_info[i].first_row;
     813          84 :             int         idx = pct_info[i].idx;
     814             : 
     815             :             /* Advance to target row, if not already there */
     816          84 :             if (target_row > rownum)
     817             :             {
     818          84 :                 if (!tuplesort_skiptuples(osastate->sortstate, target_row - rownum - 1, true))
     819           0 :                     elog(ERROR, "missing row in percentile_disc");
     820             : 
     821          84 :                 if (!tuplesort_getdatum(osastate->sortstate, true, true, &val,
     822             :                                         &isnull, NULL))
     823           0 :                     elog(ERROR, "missing row in percentile_disc");
     824             : 
     825          84 :                 rownum = target_row;
     826             :             }
     827             : 
     828          84 :             result_datum[idx] = val;
     829          84 :             result_isnull[idx] = isnull;
     830             :         }
     831             :     }
     832             : 
     833             :     /* We make the output array the same shape as the input */
     834          18 :     PG_RETURN_POINTER(construct_md_array(result_datum, result_isnull,
     835             :                                          ARR_NDIM(param),
     836             :                                          ARR_DIMS(param),
     837             :                                          ARR_LBOUND(param),
     838             :                                          osastate->qstate->sortColType,
     839             :                                          osastate->qstate->typLen,
     840             :                                          osastate->qstate->typByVal,
     841             :                                          osastate->qstate->typAlign));
     842             : }
     843             : 
     844             : /*
     845             :  * percentile_cont(float8[]) within group ()    - continuous percentiles
     846             :  */
     847             : static Datum
     848          12 : percentile_cont_multi_final_common(FunctionCallInfo fcinfo,
     849             :                                    Oid expect_type,
     850             :                                    int16 typLen, bool typByVal, char typAlign,
     851             :                                    LerpFunc lerpfunc)
     852             : {
     853             :     OSAPerGroupState *osastate;
     854             :     ArrayType  *param;
     855             :     Datum      *percentiles_datum;
     856             :     bool       *percentiles_null;
     857             :     int         num_percentiles;
     858             :     struct pct_info *pct_info;
     859             :     Datum      *result_datum;
     860             :     bool       *result_isnull;
     861          12 :     int64       rownum = 0;
     862          12 :     Datum       first_val = (Datum) 0;
     863          12 :     Datum       second_val = (Datum) 0;
     864             :     bool        isnull;
     865             :     int         i;
     866             : 
     867             :     Assert(AggCheckCallContext(fcinfo, NULL) == AGG_CONTEXT_AGGREGATE);
     868             : 
     869             :     /* If there were no regular rows, the result is NULL */
     870          12 :     if (PG_ARGISNULL(0))
     871           0 :         PG_RETURN_NULL();
     872             : 
     873          12 :     osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
     874             : 
     875             :     /* number_of_rows could be zero if we only saw NULL input values */
     876          12 :     if (osastate->number_of_rows == 0)
     877           0 :         PG_RETURN_NULL();
     878             : 
     879             :     Assert(expect_type == osastate->qstate->sortColType);
     880             : 
     881             :     /* Deconstruct the percentile-array input */
     882          12 :     if (PG_ARGISNULL(1))
     883           0 :         PG_RETURN_NULL();
     884          12 :     param = PG_GETARG_ARRAYTYPE_P(1);
     885             : 
     886          12 :     deconstruct_array_builtin(param, FLOAT8OID,
     887             :                               &percentiles_datum,
     888             :                               &percentiles_null,
     889             :                               &num_percentiles);
     890             : 
     891          12 :     if (num_percentiles == 0)
     892           0 :         PG_RETURN_POINTER(construct_empty_array(osastate->qstate->sortColType));
     893             : 
     894          12 :     pct_info = setup_pct_info(num_percentiles,
     895             :                               percentiles_datum,
     896             :                               percentiles_null,
     897             :                               osastate->number_of_rows,
     898             :                               true);
     899             : 
     900          12 :     result_datum = (Datum *) palloc(num_percentiles * sizeof(Datum));
     901          12 :     result_isnull = (bool *) palloc(num_percentiles * sizeof(bool));
     902             : 
     903             :     /*
     904             :      * Start by dealing with any nulls in the param array - those are sorted
     905             :      * to the front on row=0, so set the corresponding result indexes to null
     906             :      */
     907          12 :     for (i = 0; i < num_percentiles; i++)
     908             :     {
     909          12 :         int         idx = pct_info[i].idx;
     910             : 
     911          12 :         if (pct_info[i].first_row > 0)
     912          12 :             break;
     913             : 
     914           0 :         result_datum[idx] = (Datum) 0;
     915           0 :         result_isnull[idx] = true;
     916             :     }
     917             : 
     918             :     /*
     919             :      * If there's anything left after doing the nulls, then grind the input
     920             :      * and extract the needed values
     921             :      */
     922          12 :     if (i < num_percentiles)
     923             :     {
     924             :         /* Finish the sort, or rescan if we already did */
     925          12 :         if (!osastate->sort_done)
     926             :         {
     927          12 :             tuplesort_performsort(osastate->sortstate);
     928          12 :             osastate->sort_done = true;
     929             :         }
     930             :         else
     931           0 :             tuplesort_rescan(osastate->sortstate);
     932             : 
     933         108 :         for (; i < num_percentiles; i++)
     934             :         {
     935          96 :             int64       first_row = pct_info[i].first_row;
     936          96 :             int64       second_row = pct_info[i].second_row;
     937          96 :             int         idx = pct_info[i].idx;
     938             : 
     939             :             /*
     940             :              * Advance to first_row, if not already there.  Note that we might
     941             :              * already have rownum beyond first_row, in which case first_val
     942             :              * is already correct.  (This occurs when interpolating between
     943             :              * the same two input rows as for the previous percentile.)
     944             :              */
     945          96 :             if (first_row > rownum)
     946             :             {
     947          48 :                 if (!tuplesort_skiptuples(osastate->sortstate, first_row - rownum - 1, true))
     948           0 :                     elog(ERROR, "missing row in percentile_cont");
     949             : 
     950          48 :                 if (!tuplesort_getdatum(osastate->sortstate, true, true,
     951          48 :                                         &first_val, &isnull, NULL) || isnull)
     952           0 :                     elog(ERROR, "missing row in percentile_cont");
     953             : 
     954          48 :                 rownum = first_row;
     955             :                 /* Always advance second_val to be latest input value */
     956          48 :                 second_val = first_val;
     957             :             }
     958          48 :             else if (first_row == rownum)
     959             :             {
     960             :                 /*
     961             :                  * We are already at the desired row, so we must previously
     962             :                  * have read its value into second_val (and perhaps first_val
     963             :                  * as well, but this assignment is harmless in that case).
     964             :                  */
     965          24 :                 first_val = second_val;
     966             :             }
     967             : 
     968             :             /* Fetch second_row if needed */
     969          96 :             if (second_row > rownum)
     970             :             {
     971          36 :                 if (!tuplesort_getdatum(osastate->sortstate, true, true,
     972          36 :                                         &second_val, &isnull, NULL) || isnull)
     973           0 :                     elog(ERROR, "missing row in percentile_cont");
     974          36 :                 rownum++;
     975             :             }
     976             :             /* We should now certainly be on second_row exactly */
     977             :             Assert(second_row == rownum);
     978             : 
     979             :             /* Compute appropriate result */
     980          96 :             if (second_row > first_row)
     981          60 :                 result_datum[idx] = lerpfunc(first_val, second_val,
     982          60 :                                              pct_info[i].proportion);
     983             :             else
     984          36 :                 result_datum[idx] = first_val;
     985             : 
     986          96 :             result_isnull[idx] = false;
     987             :         }
     988             :     }
     989             : 
     990             :     /* We make the output array the same shape as the input */
     991          12 :     PG_RETURN_POINTER(construct_md_array(result_datum, result_isnull,
     992             :                                          ARR_NDIM(param),
     993             :                                          ARR_DIMS(param), ARR_LBOUND(param),
     994             :                                          expect_type,
     995             :                                          typLen,
     996             :                                          typByVal,
     997             :                                          typAlign));
     998             : }
     999             : 
    1000             : /*
    1001             :  * percentile_cont(float8[]) within group (float8)  - continuous percentiles
    1002             :  */
    1003             : Datum
    1004          12 : percentile_cont_float8_multi_final(PG_FUNCTION_ARGS)
    1005             : {
    1006          12 :     return percentile_cont_multi_final_common(fcinfo,
    1007             :                                               FLOAT8OID,
    1008             :     /* hard-wired info on type float8 */
    1009             :                                               sizeof(float8),
    1010             :                                               FLOAT8PASSBYVAL,
    1011             :                                               TYPALIGN_DOUBLE,
    1012             :                                               float8_lerp);
    1013             : }
    1014             : 
    1015             : /*
    1016             :  * percentile_cont(float8[]) within group (interval)  - continuous percentiles
    1017             :  */
    1018             : Datum
    1019           0 : percentile_cont_interval_multi_final(PG_FUNCTION_ARGS)
    1020             : {
    1021           0 :     return percentile_cont_multi_final_common(fcinfo,
    1022             :                                               INTERVALOID,
    1023             :     /* hard-wired info on type interval */
    1024             :                                               16, false, TYPALIGN_DOUBLE,
    1025             :                                               interval_lerp);
    1026             : }
    1027             : 
    1028             : 
    1029             : /*
    1030             :  * mode() within group (anyelement) - most common value
    1031             :  */
    1032             : Datum
    1033          60 : mode_final(PG_FUNCTION_ARGS)
    1034             : {
    1035             :     OSAPerGroupState *osastate;
    1036             :     Datum       val;
    1037             :     bool        isnull;
    1038          60 :     Datum       mode_val = (Datum) 0;
    1039          60 :     int64       mode_freq = 0;
    1040          60 :     Datum       last_val = (Datum) 0;
    1041          60 :     int64       last_val_freq = 0;
    1042          60 :     bool        last_val_is_mode = false;
    1043             :     FmgrInfo   *equalfn;
    1044          60 :     Datum       abbrev_val = (Datum) 0;
    1045          60 :     Datum       last_abbrev_val = (Datum) 0;
    1046             :     bool        shouldfree;
    1047             : 
    1048             :     Assert(AggCheckCallContext(fcinfo, NULL) == AGG_CONTEXT_AGGREGATE);
    1049             : 
    1050             :     /* If there were no regular rows, the result is NULL */
    1051          60 :     if (PG_ARGISNULL(0))
    1052           0 :         PG_RETURN_NULL();
    1053             : 
    1054          60 :     osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
    1055             : 
    1056             :     /* number_of_rows could be zero if we only saw NULL input values */
    1057          60 :     if (osastate->number_of_rows == 0)
    1058           0 :         PG_RETURN_NULL();
    1059             : 
    1060             :     /* Look up the equality function for the datatype, if we didn't already */
    1061          60 :     equalfn = &(osastate->qstate->equalfn);
    1062          60 :     if (!OidIsValid(equalfn->fn_oid))
    1063           6 :         fmgr_info_cxt(get_opcode(osastate->qstate->eqOperator), equalfn,
    1064           6 :                       osastate->qstate->qcontext);
    1065             : 
    1066          60 :     shouldfree = !(osastate->qstate->typByVal);
    1067             : 
    1068             :     /* Finish the sort, or rescan if we already did */
    1069          60 :     if (!osastate->sort_done)
    1070             :     {
    1071          60 :         tuplesort_performsort(osastate->sortstate);
    1072          60 :         osastate->sort_done = true;
    1073             :     }
    1074             :     else
    1075           0 :         tuplesort_rescan(osastate->sortstate);
    1076             : 
    1077             :     /* Scan tuples and count frequencies */
    1078       60060 :     while (tuplesort_getdatum(osastate->sortstate, true, true, &val, &isnull,
    1079             :                               &abbrev_val))
    1080             :     {
    1081             :         /* we don't expect any nulls, but ignore them if found */
    1082       60000 :         if (isnull)
    1083           0 :             continue;
    1084             : 
    1085       60000 :         if (last_val_freq == 0)
    1086             :         {
    1087             :             /* first nonnull value - it's the mode for now */
    1088          60 :             mode_val = last_val = val;
    1089          60 :             mode_freq = last_val_freq = 1;
    1090          60 :             last_val_is_mode = true;
    1091          60 :             last_abbrev_val = abbrev_val;
    1092             :         }
    1093      119880 :         else if (abbrev_val == last_abbrev_val &&
    1094       59940 :                  DatumGetBool(FunctionCall2Coll(equalfn, PG_GET_COLLATION(), val, last_val)))
    1095             :         {
    1096             :             /* value equal to previous value, count it */
    1097       59760 :             if (last_val_is_mode)
    1098       15924 :                 mode_freq++;    /* needn't maintain last_val_freq */
    1099       43836 :             else if (++last_val_freq > mode_freq)
    1100             :             {
    1101             :                 /* last_val becomes new mode */
    1102          66 :                 if (shouldfree)
    1103          66 :                     pfree(DatumGetPointer(mode_val));
    1104          66 :                 mode_val = last_val;
    1105          66 :                 mode_freq = last_val_freq;
    1106          66 :                 last_val_is_mode = true;
    1107             :             }
    1108       59760 :             if (shouldfree)
    1109       59760 :                 pfree(DatumGetPointer(val));
    1110             :         }
    1111             :         else
    1112             :         {
    1113             :             /* val should replace last_val */
    1114         180 :             if (shouldfree && !last_val_is_mode)
    1115          72 :                 pfree(DatumGetPointer(last_val));
    1116         180 :             last_val = val;
    1117             :             /* avoid equality function calls by reusing abbreviated keys */
    1118         180 :             last_abbrev_val = abbrev_val;
    1119         180 :             last_val_freq = 1;
    1120         180 :             last_val_is_mode = false;
    1121             :         }
    1122             : 
    1123       60000 :         CHECK_FOR_INTERRUPTS();
    1124             :     }
    1125             : 
    1126          60 :     if (shouldfree && !last_val_is_mode)
    1127          42 :         pfree(DatumGetPointer(last_val));
    1128             : 
    1129          60 :     if (mode_freq)
    1130          60 :         PG_RETURN_DATUM(mode_val);
    1131             :     else
    1132           0 :         PG_RETURN_NULL();
    1133             : }
    1134             : 
    1135             : 
    1136             : /*
    1137             :  * Common code to sanity-check args for hypothetical-set functions. No need
    1138             :  * for friendly errors, these can only happen if someone's messing up the
    1139             :  * aggregate definitions. The checks are needed for security, however.
    1140             :  */
    1141             : static void
    1142         274 : hypothetical_check_argtypes(FunctionCallInfo fcinfo, int nargs,
    1143             :                             TupleDesc tupdesc)
    1144             : {
    1145             :     int         i;
    1146             : 
    1147             :     /* check that we have an int4 flag column */
    1148         274 :     if (!tupdesc ||
    1149         274 :         (nargs + 1) != tupdesc->natts ||
    1150         274 :         TupleDescAttr(tupdesc, nargs)->atttypid != INT4OID)
    1151           0 :         elog(ERROR, "type mismatch in hypothetical-set function");
    1152             : 
    1153             :     /* check that direct args match in type with aggregated args */
    1154         838 :     for (i = 0; i < nargs; i++)
    1155             :     {
    1156         564 :         Form_pg_attribute attr = TupleDescAttr(tupdesc, i);
    1157             : 
    1158         564 :         if (get_fn_expr_argtype(fcinfo->flinfo, i + 1) != attr->atttypid)
    1159           0 :             elog(ERROR, "type mismatch in hypothetical-set function");
    1160             :     }
    1161         274 : }
    1162             : 
    1163             : /*
    1164             :  * compute rank of hypothetical row
    1165             :  *
    1166             :  * flag should be -1 to sort hypothetical row ahead of its peers, or +1
    1167             :  * to sort behind.
    1168             :  * total number of regular rows is returned into *number_of_rows.
    1169             :  */
    1170             : static int64
    1171         250 : hypothetical_rank_common(FunctionCallInfo fcinfo, int flag,
    1172             :                          int64 *number_of_rows)
    1173             : {
    1174         250 :     int         nargs = PG_NARGS() - 1;
    1175         250 :     int64       rank = 1;
    1176             :     OSAPerGroupState *osastate;
    1177             :     TupleTableSlot *slot;
    1178             :     int         i;
    1179             : 
    1180             :     Assert(AggCheckCallContext(fcinfo, NULL) == AGG_CONTEXT_AGGREGATE);
    1181             : 
    1182             :     /* If there were no regular rows, the rank is always 1 */
    1183         250 :     if (PG_ARGISNULL(0))
    1184             :     {
    1185           6 :         *number_of_rows = 0;
    1186           6 :         return 1;
    1187             :     }
    1188             : 
    1189         244 :     osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
    1190         244 :     *number_of_rows = osastate->number_of_rows;
    1191             : 
    1192             :     /* Adjust nargs to be the number of direct (or aggregated) args */
    1193         244 :     if (nargs % 2 != 0)
    1194           0 :         elog(ERROR, "wrong number of arguments in hypothetical-set function");
    1195         244 :     nargs /= 2;
    1196             : 
    1197         244 :     hypothetical_check_argtypes(fcinfo, nargs, osastate->qstate->tupdesc);
    1198             : 
    1199             :     /* because we need a hypothetical row, we can't share transition state */
    1200             :     Assert(!osastate->sort_done);
    1201             : 
    1202             :     /* insert the hypothetical row into the sort */
    1203         244 :     slot = osastate->qstate->tupslot;
    1204         244 :     ExecClearTuple(slot);
    1205         778 :     for (i = 0; i < nargs; i++)
    1206             :     {
    1207         534 :         slot->tts_values[i] = PG_GETARG_DATUM(i + 1);
    1208         534 :         slot->tts_isnull[i] = PG_ARGISNULL(i + 1);
    1209             :     }
    1210         244 :     slot->tts_values[i] = Int32GetDatum(flag);
    1211         244 :     slot->tts_isnull[i] = false;
    1212         244 :     ExecStoreVirtualTuple(slot);
    1213             : 
    1214         244 :     tuplesort_puttupleslot(osastate->sortstate, slot);
    1215             : 
    1216             :     /* finish the sort */
    1217         244 :     tuplesort_performsort(osastate->sortstate);
    1218         244 :     osastate->sort_done = true;
    1219             : 
    1220             :     /* iterate till we find the hypothetical row */
    1221        4274 :     while (tuplesort_gettupleslot(osastate->sortstate, true, true, slot, NULL))
    1222             :     {
    1223             :         bool        isnull;
    1224        4274 :         Datum       d = slot_getattr(slot, nargs + 1, &isnull);
    1225             : 
    1226        4274 :         if (!isnull && DatumGetInt32(d) != 0)
    1227         244 :             break;
    1228             : 
    1229        4030 :         rank++;
    1230             : 
    1231        4030 :         CHECK_FOR_INTERRUPTS();
    1232             :     }
    1233             : 
    1234         244 :     ExecClearTuple(slot);
    1235             : 
    1236         244 :     return rank;
    1237             : }
    1238             : 
    1239             : 
    1240             : /*
    1241             :  * rank()  - rank of hypothetical row
    1242             :  */
    1243             : Datum
    1244         232 : hypothetical_rank_final(PG_FUNCTION_ARGS)
    1245             : {
    1246             :     int64       rank;
    1247             :     int64       rowcount;
    1248             : 
    1249         232 :     rank = hypothetical_rank_common(fcinfo, -1, &rowcount);
    1250             : 
    1251         232 :     PG_RETURN_INT64(rank);
    1252             : }
    1253             : 
    1254             : /*
    1255             :  * percent_rank()   - percentile rank of hypothetical row
    1256             :  */
    1257             : Datum
    1258          12 : hypothetical_percent_rank_final(PG_FUNCTION_ARGS)
    1259             : {
    1260             :     int64       rank;
    1261             :     int64       rowcount;
    1262             :     double      result_val;
    1263             : 
    1264          12 :     rank = hypothetical_rank_common(fcinfo, -1, &rowcount);
    1265             : 
    1266          12 :     if (rowcount == 0)
    1267           6 :         PG_RETURN_FLOAT8(0);
    1268             : 
    1269           6 :     result_val = (double) (rank - 1) / (double) (rowcount);
    1270             : 
    1271           6 :     PG_RETURN_FLOAT8(result_val);
    1272             : }
    1273             : 
    1274             : /*
    1275             :  * cume_dist()  - cumulative distribution of hypothetical row
    1276             :  */
    1277             : Datum
    1278           6 : hypothetical_cume_dist_final(PG_FUNCTION_ARGS)
    1279             : {
    1280             :     int64       rank;
    1281             :     int64       rowcount;
    1282             :     double      result_val;
    1283             : 
    1284           6 :     rank = hypothetical_rank_common(fcinfo, 1, &rowcount);
    1285             : 
    1286           6 :     result_val = (double) (rank) / (double) (rowcount + 1);
    1287             : 
    1288           6 :     PG_RETURN_FLOAT8(result_val);
    1289             : }
    1290             : 
    1291             : /*
    1292             :  * dense_rank() - rank of hypothetical row without gaps in ranking
    1293             :  */
    1294             : Datum
    1295          30 : hypothetical_dense_rank_final(PG_FUNCTION_ARGS)
    1296             : {
    1297             :     ExprContext *econtext;
    1298             :     ExprState  *compareTuple;
    1299          30 :     int         nargs = PG_NARGS() - 1;
    1300          30 :     int64       rank = 1;
    1301          30 :     int64       duplicate_count = 0;
    1302             :     OSAPerGroupState *osastate;
    1303             :     int         numDistinctCols;
    1304          30 :     Datum       abbrevVal = (Datum) 0;
    1305          30 :     Datum       abbrevOld = (Datum) 0;
    1306             :     TupleTableSlot *slot;
    1307             :     TupleTableSlot *extraslot;
    1308             :     TupleTableSlot *slot2;
    1309             :     int         i;
    1310             : 
    1311             :     Assert(AggCheckCallContext(fcinfo, NULL) == AGG_CONTEXT_AGGREGATE);
    1312             : 
    1313             :     /* If there were no regular rows, the rank is always 1 */
    1314          30 :     if (PG_ARGISNULL(0))
    1315           0 :         PG_RETURN_INT64(rank);
    1316             : 
    1317          30 :     osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
    1318          30 :     econtext = osastate->qstate->econtext;
    1319          30 :     if (!econtext)
    1320             :     {
    1321             :         MemoryContext oldcontext;
    1322             : 
    1323             :         /* Make sure to we create econtext under correct parent context. */
    1324          18 :         oldcontext = MemoryContextSwitchTo(osastate->qstate->qcontext);
    1325          18 :         osastate->qstate->econtext = CreateStandaloneExprContext();
    1326          18 :         econtext = osastate->qstate->econtext;
    1327          18 :         MemoryContextSwitchTo(oldcontext);
    1328             :     }
    1329             : 
    1330             :     /* Adjust nargs to be the number of direct (or aggregated) args */
    1331          30 :     if (nargs % 2 != 0)
    1332           0 :         elog(ERROR, "wrong number of arguments in hypothetical-set function");
    1333          30 :     nargs /= 2;
    1334             : 
    1335          30 :     hypothetical_check_argtypes(fcinfo, nargs, osastate->qstate->tupdesc);
    1336             : 
    1337             :     /*
    1338             :      * When comparing tuples, we can omit the flag column since we will only
    1339             :      * compare rows with flag == 0.
    1340             :      */
    1341          30 :     numDistinctCols = osastate->qstate->numSortCols - 1;
    1342             : 
    1343             :     /* Build tuple comparator, if we didn't already */
    1344          30 :     compareTuple = osastate->qstate->compareTuple;
    1345          30 :     if (compareTuple == NULL)
    1346             :     {
    1347          18 :         AttrNumber *sortColIdx = osastate->qstate->sortColIdx;
    1348             :         MemoryContext oldContext;
    1349             : 
    1350          18 :         oldContext = MemoryContextSwitchTo(osastate->qstate->qcontext);
    1351          18 :         compareTuple = execTuplesMatchPrepare(osastate->qstate->tupdesc,
    1352             :                                               numDistinctCols,
    1353             :                                               sortColIdx,
    1354          18 :                                               osastate->qstate->eqOperators,
    1355          18 :                                               osastate->qstate->sortCollations,
    1356             :                                               NULL);
    1357          18 :         MemoryContextSwitchTo(oldContext);
    1358          18 :         osastate->qstate->compareTuple = compareTuple;
    1359             :     }
    1360             : 
    1361             :     /* because we need a hypothetical row, we can't share transition state */
    1362             :     Assert(!osastate->sort_done);
    1363             : 
    1364             :     /* insert the hypothetical row into the sort */
    1365          30 :     slot = osastate->qstate->tupslot;
    1366          30 :     ExecClearTuple(slot);
    1367          60 :     for (i = 0; i < nargs; i++)
    1368             :     {
    1369          30 :         slot->tts_values[i] = PG_GETARG_DATUM(i + 1);
    1370          30 :         slot->tts_isnull[i] = PG_ARGISNULL(i + 1);
    1371             :     }
    1372          30 :     slot->tts_values[i] = Int32GetDatum(-1);
    1373          30 :     slot->tts_isnull[i] = false;
    1374          30 :     ExecStoreVirtualTuple(slot);
    1375             : 
    1376          30 :     tuplesort_puttupleslot(osastate->sortstate, slot);
    1377             : 
    1378             :     /* finish the sort */
    1379          30 :     tuplesort_performsort(osastate->sortstate);
    1380          30 :     osastate->sort_done = true;
    1381             : 
    1382             :     /*
    1383             :      * We alternate fetching into tupslot and extraslot so that we have the
    1384             :      * previous row available for comparisons.  This is accomplished by
    1385             :      * swapping the slot pointer variables after each row.
    1386             :      */
    1387          30 :     extraslot = MakeSingleTupleTableSlot(osastate->qstate->tupdesc,
    1388             :                                          &TTSOpsMinimalTuple);
    1389          30 :     slot2 = extraslot;
    1390             : 
    1391             :     /* iterate till we find the hypothetical row */
    1392          66 :     while (tuplesort_gettupleslot(osastate->sortstate, true, true, slot,
    1393             :                                   &abbrevVal))
    1394             :     {
    1395             :         bool        isnull;
    1396          66 :         Datum       d = slot_getattr(slot, nargs + 1, &isnull);
    1397             :         TupleTableSlot *tmpslot;
    1398             : 
    1399          66 :         if (!isnull && DatumGetInt32(d) != 0)
    1400          30 :             break;
    1401             : 
    1402             :         /* count non-distinct tuples */
    1403          36 :         econtext->ecxt_outertuple = slot;
    1404          36 :         econtext->ecxt_innertuple = slot2;
    1405             : 
    1406          36 :         if (!TupIsNull(slot2) &&
    1407          48 :             abbrevVal == abbrevOld &&
    1408          24 :             ExecQualAndReset(compareTuple, econtext))
    1409          12 :             duplicate_count++;
    1410             : 
    1411          36 :         tmpslot = slot2;
    1412          36 :         slot2 = slot;
    1413          36 :         slot = tmpslot;
    1414             :         /* avoid ExecQual() calls by reusing abbreviated keys */
    1415          36 :         abbrevOld = abbrevVal;
    1416             : 
    1417          36 :         rank++;
    1418             : 
    1419          36 :         CHECK_FOR_INTERRUPTS();
    1420             :     }
    1421             : 
    1422          30 :     ExecClearTuple(slot);
    1423          30 :     ExecClearTuple(slot2);
    1424             : 
    1425          30 :     ExecDropSingleTupleTableSlot(extraslot);
    1426             : 
    1427          30 :     rank = rank - duplicate_count;
    1428             : 
    1429          30 :     PG_RETURN_INT64(rank);
    1430             : }

Generated by: LCOV version 1.14