Commit 8b49a604 authored by Tom Lane's avatar Tom Lane

Cache catalog lookup data across groups in ordered-set aggregates.

The initial commit of ordered-set aggregates just did all the setup work
afresh each time the aggregate function is started up.  But in a GROUP BY
query, the catalog lookups need not be repeated for each group, since the
column datatypes and sort information won't change.  When there are many
small groups, this makes for a useful, though not huge, performance
improvement.  Per suggestion from Andrew Gierth.

Profiling of these cases suggests that it might be profitable to avoid
duplicate lookups within tuplesort startup as well; but changing the
tuplesort APIs would have much broader impact, so I left that for
another day.
parent 92459e7a
...@@ -32,16 +32,25 @@ ...@@ -32,16 +32,25 @@
/* /*
* Generic support for ordered-set aggregates * Generic support for ordered-set aggregates
*
* The state for an ordered-set aggregate is divided into a per-group struct
* (which is the internal-type transition state datum returned to nodeAgg.c)
* and a per-query struct, which contains data and sub-objects that we can
* create just once per query because they will not change across groups.
* The per-query struct and subsidiary data live in the executor's per-query
* memory context, and go away implicitly at ExecutorEnd().
*/ */
typedef struct OrderedSetAggState typedef struct OSAPerQueryState
{ {
/* Aggref for this aggregate: */ /* Aggref for this aggregate: */
Aggref *aggref; Aggref *aggref;
/* Sort object we're accumulating data in: */ /* Memory context containing this struct and other per-query data: */
Tuplesortstate *sortstate; MemoryContext qcontext;
/* Number of normal rows inserted into sortstate: */ /* Memory context containing per-group data: */
int64 number_of_rows; MemoryContext gcontext;
/* Agg plan node's output econtext: */
ExprContext *peraggecontext;
/* These fields are used only when accumulating tuples: */ /* These fields are used only when accumulating tuples: */
...@@ -49,17 +58,41 @@ typedef struct OrderedSetAggState ...@@ -49,17 +58,41 @@ typedef struct OrderedSetAggState
TupleDesc tupdesc; TupleDesc tupdesc;
/* Tuple slot we can use for inserting/extracting tuples: */ /* Tuple slot we can use for inserting/extracting tuples: */
TupleTableSlot *tupslot; TupleTableSlot *tupslot;
/* Per-sort-column sorting information */
int numSortCols;
AttrNumber *sortColIdx;
Oid *sortOperators;
Oid *eqOperators;
Oid *sortCollations;
bool *sortNullsFirsts;
/* Equality operator call info, created only if needed: */
FmgrInfo *equalfns;
/* These fields are used only when accumulating datums: */ /* These fields are used only when accumulating datums: */
/* Info about datatype of datums being sorted: */ /* Info about datatype of datums being sorted: */
Oid datumtype; Oid sortColType;
int16 typLen; int16 typLen;
bool typByVal; bool typByVal;
char typAlign; char typAlign;
/* Info about equality operator associated with sort operator: */ /* Info about sort ordering: */
Oid sortOperator;
Oid eqOperator; Oid eqOperator;
} OrderedSetAggState; Oid sortCollation;
bool sortNullsFirst;
/* Equality operator call info, created only if needed: */
FmgrInfo equalfn;
} OSAPerQueryState;
typedef struct OSAPerGroupState
{
/* Link to the per-query state for this aggregate: */
OSAPerQueryState *qstate;
/* Sort object we're accumulating data in: */
Tuplesortstate *sortstate;
/* Number of normal rows inserted into sortstate: */
int64 number_of_rows;
} OSAPerGroupState;
static void ordered_set_shutdown(Datum arg); static void ordered_set_shutdown(Datum arg);
...@@ -67,19 +100,32 @@ static void ordered_set_shutdown(Datum arg); ...@@ -67,19 +100,32 @@ static void ordered_set_shutdown(Datum arg);
/* /*
* Set up working state for an ordered-set aggregate * Set up working state for an ordered-set aggregate
*/ */
static OrderedSetAggState * static OSAPerGroupState *
ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples) ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples)
{ {
OrderedSetAggState *osastate; OSAPerGroupState *osastate;
OSAPerQueryState *qstate;
MemoryContext oldcontext;
/*
* We keep a link to the per-query state in fn_extra; if it's not there,
* create it, and do the per-query setup we need.
*/
qstate = (OSAPerQueryState *) fcinfo->flinfo->fn_extra;
if (qstate == NULL)
{
Aggref *aggref; Aggref *aggref;
MemoryContext qcontext;
MemoryContext gcontext;
ExprContext *peraggecontext; ExprContext *peraggecontext;
MemoryContext aggcontext;
MemoryContext oldcontext;
List *sortlist; List *sortlist;
int numSortCols; int numSortCols;
/* Must be called as aggregate; get the Agg node's query-lifespan context */ /*
if (AggCheckCallContext(fcinfo, &aggcontext) != AGG_CONTEXT_AGGREGATE) * Check we're called as aggregate, and get the Agg node's
* group-lifespan context
*/
if (AggCheckCallContext(fcinfo, &gcontext) != AGG_CONTEXT_AGGREGATE)
elog(ERROR, "ordered-set aggregate called in non-aggregate context"); elog(ERROR, "ordered-set aggregate called in non-aggregate context");
/* Need the Aggref as well */ /* Need the Aggref as well */
aggref = AggGetAggref(fcinfo); aggref = AggGetAggref(fcinfo);
...@@ -92,10 +138,19 @@ ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples) ...@@ -92,10 +138,19 @@ ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples)
if (!peraggecontext) if (!peraggecontext)
elog(ERROR, "ordered-set aggregate called in non-aggregate context"); elog(ERROR, "ordered-set aggregate called in non-aggregate context");
/* Initialize working-state object in the aggregate-lifespan context */ /*
osastate = (OrderedSetAggState *) * Prepare per-query structures in the fn_mcxt, which we assume is the
MemoryContextAllocZero(aggcontext, sizeof(OrderedSetAggState)); * executor's per-query context; in any case it's the right place to
osastate->aggref = aggref; * keep anything found via fn_extra.
*/
qcontext = fcinfo->flinfo->fn_mcxt;
oldcontext = MemoryContextSwitchTo(qcontext);
qstate = (OSAPerQueryState *) palloc0(sizeof(OSAPerQueryState));
qstate->aggref = aggref;
qstate->qcontext = qcontext;
qstate->gcontext = gcontext;
qstate->peraggecontext = peraggecontext;
/* Extract the sort information */ /* Extract the sort information */
sortlist = aggref->aggorder; sortlist = aggref->aggorder;
...@@ -106,32 +161,41 @@ ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples) ...@@ -106,32 +161,41 @@ ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples)
bool ishypothetical = (aggref->aggkind == AGGKIND_HYPOTHETICAL); bool ishypothetical = (aggref->aggkind == AGGKIND_HYPOTHETICAL);
AttrNumber *sortColIdx; AttrNumber *sortColIdx;
Oid *sortOperators; Oid *sortOperators;
Oid *eqOperators;
Oid *sortCollations; Oid *sortCollations;
bool *sortNullsFirst; bool *sortNullsFirsts;
ListCell *lc; ListCell *lc;
int i; int i;
if (ishypothetical) if (ishypothetical)
numSortCols++; /* make space for flag column */ numSortCols++; /* make space for flag column */
/* these arrays are made in short-lived context */ qstate->numSortCols = numSortCols;
sortColIdx = (AttrNumber *) palloc(numSortCols * sizeof(AttrNumber)); qstate->sortColIdx = sortColIdx =
sortOperators = (Oid *) palloc(numSortCols * sizeof(Oid)); (AttrNumber *) palloc(numSortCols * sizeof(AttrNumber));
sortCollations = (Oid *) palloc(numSortCols * sizeof(Oid)); qstate->sortOperators = sortOperators =
sortNullsFirst = (bool *) palloc(numSortCols * sizeof(bool)); (Oid *) palloc(numSortCols * sizeof(Oid));
qstate->eqOperators = eqOperators =
(Oid *) palloc(numSortCols * sizeof(Oid));
qstate->sortCollations = sortCollations =
(Oid *) palloc(numSortCols * sizeof(Oid));
qstate->sortNullsFirsts = sortNullsFirsts =
(bool *) palloc(numSortCols * sizeof(bool));
i = 0; i = 0;
foreach(lc, sortlist) foreach(lc, sortlist)
{ {
SortGroupClause *sortcl = (SortGroupClause *) lfirst(lc); SortGroupClause *sortcl = (SortGroupClause *) lfirst(lc);
TargetEntry *tle = get_sortgroupclause_tle(sortcl, aggref->args); TargetEntry *tle = get_sortgroupclause_tle(sortcl,
aggref->args);
/* the parser should have made sure of this */ /* the parser should have made sure of this */
Assert(OidIsValid(sortcl->sortop)); Assert(OidIsValid(sortcl->sortop));
sortColIdx[i] = tle->resno; sortColIdx[i] = tle->resno;
sortOperators[i] = sortcl->sortop; sortOperators[i] = sortcl->sortop;
eqOperators[i] = sortcl->eqop;
sortCollations[i] = exprCollation((Node *) tle->expr); sortCollations[i] = exprCollation((Node *) tle->expr);
sortNullsFirst[i] = sortcl->nulls_first; sortNullsFirsts[i] = sortcl->nulls_first;
i++; i++;
} }
...@@ -140,31 +204,29 @@ ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples) ...@@ -140,31 +204,29 @@ ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples)
/* Add an integer flag column as the last sort column */ /* Add an integer flag column as the last sort column */
sortColIdx[i] = list_length(aggref->args) + 1; sortColIdx[i] = list_length(aggref->args) + 1;
sortOperators[i] = Int4LessOperator; sortOperators[i] = Int4LessOperator;
eqOperators[i] = Int4EqualOperator;
sortCollations[i] = InvalidOid; sortCollations[i] = InvalidOid;
sortNullsFirst[i] = false; sortNullsFirsts[i] = false;
i++; i++;
} }
Assert(i == numSortCols); Assert(i == numSortCols);
/* Now build the stuff we need in aggregate-lifespan context */
oldcontext = MemoryContextSwitchTo(aggcontext);
/* /*
* Get a tupledesc corresponding to the aggregated inputs (including * Get a tupledesc corresponding to the aggregated inputs
* sort expressions) of the agg. * (including sort expressions) of the agg.
*/ */
osastate->tupdesc = ExecTypeFromTL(aggref->args, false); qstate->tupdesc = ExecTypeFromTL(aggref->args, false);
/* If we need a flag column, hack the tupledesc to include that */ /* If we need a flag column, hack the tupledesc to include that */
if (ishypothetical) if (ishypothetical)
{ {
TupleDesc newdesc; TupleDesc newdesc;
int natts = osastate->tupdesc->natts; int natts = qstate->tupdesc->natts;
newdesc = CreateTemplateTupleDesc(natts + 1, false); newdesc = CreateTemplateTupleDesc(natts + 1, false);
for (i = 1; i <= natts; i++) for (i = 1; i <= natts; i++)
TupleDescCopyEntry(newdesc, i, osastate->tupdesc, i); TupleDescCopyEntry(newdesc, i, qstate->tupdesc, i);
TupleDescInitEntry(newdesc, TupleDescInitEntry(newdesc,
(AttrNumber) ++natts, (AttrNumber) ++natts,
...@@ -173,32 +235,18 @@ ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples) ...@@ -173,32 +235,18 @@ ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples)
-1, -1,
0); 0);
FreeTupleDesc(osastate->tupdesc); FreeTupleDesc(qstate->tupdesc);
osastate->tupdesc = newdesc; qstate->tupdesc = newdesc;
} }
/* Initialize tuplesort object */
osastate->sortstate = tuplesort_begin_heap(osastate->tupdesc,
numSortCols,
sortColIdx,
sortOperators,
sortCollations,
sortNullsFirst,
work_mem, false);
/* Create slot we'll use to store/retrieve rows */ /* Create slot we'll use to store/retrieve rows */
osastate->tupslot = MakeSingleTupleTableSlot(osastate->tupdesc); qstate->tupslot = MakeSingleTupleTableSlot(qstate->tupdesc);
} }
else else
{ {
/* Sort single datums */ /* Sort single datums */
SortGroupClause *sortcl; SortGroupClause *sortcl;
TargetEntry *tle; TargetEntry *tle;
Oid sortColType;
Oid sortOperator;
Oid eqOperator;
Oid sortCollation;
bool sortNullsFirst;
if (numSortCols != 1 || aggref->aggkind == AGGKIND_HYPOTHETICAL) if (numSortCols != 1 || aggref->aggkind == AGGKIND_HYPOTHETICAL)
elog(ERROR, "ordered-set aggregate support function does not support multiple aggregated columns"); elog(ERROR, "ordered-set aggregate support function does not support multiple aggregated columns");
...@@ -209,33 +257,51 @@ ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples) ...@@ -209,33 +257,51 @@ ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples)
/* the parser should have made sure of this */ /* the parser should have made sure of this */
Assert(OidIsValid(sortcl->sortop)); Assert(OidIsValid(sortcl->sortop));
sortColType = exprType((Node *) tle->expr); /* Save sort ordering info */
sortOperator = sortcl->sortop; qstate->sortColType = exprType((Node *) tle->expr);
eqOperator = sortcl->eqop; qstate->sortOperator = sortcl->sortop;
sortCollation = exprCollation((Node *) tle->expr); qstate->eqOperator = sortcl->eqop;
sortNullsFirst = sortcl->nulls_first; qstate->sortCollation = exprCollation((Node *) tle->expr);
qstate->sortNullsFirst = sortcl->nulls_first;
/* Save datatype info */ /* Save datatype info */
osastate->datumtype = sortColType; get_typlenbyvalalign(qstate->sortColType,
get_typlenbyvalalign(sortColType, &qstate->typLen,
&osastate->typLen, &qstate->typByVal,
&osastate->typByVal, &qstate->typAlign);
&osastate->typAlign); }
osastate->eqOperator = eqOperator;
fcinfo->flinfo->fn_extra = (void *) qstate;
MemoryContextSwitchTo(oldcontext);
}
/* Now build the stuff we need in group-lifespan context */
oldcontext = MemoryContextSwitchTo(qstate->gcontext);
/* Now build the stuff we need in aggregate-lifespan context */ osastate = (OSAPerGroupState *) palloc(sizeof(OSAPerGroupState));
oldcontext = MemoryContextSwitchTo(aggcontext); osastate->qstate = qstate;
/* Initialize tuplesort object */ /* Initialize tuplesort object */
osastate->sortstate = tuplesort_begin_datum(sortColType, if (use_tuples)
sortOperator, osastate->sortstate = tuplesort_begin_heap(qstate->tupdesc,
sortCollation, qstate->numSortCols,
sortNullsFirst, qstate->sortColIdx,
qstate->sortOperators,
qstate->sortCollations,
qstate->sortNullsFirsts,
work_mem, false);
else
osastate->sortstate = tuplesort_begin_datum(qstate->sortColType,
qstate->sortOperator,
qstate->sortCollation,
qstate->sortNullsFirst,
work_mem, false); work_mem, false);
}
/* Now register a shutdown callback to clean it all up */ osastate->number_of_rows = 0;
RegisterExprContextCallback(peraggecontext,
/* Now register a shutdown callback to clean things up */
RegisterExprContextCallback(qstate->peraggecontext,
ordered_set_shutdown, ordered_set_shutdown,
PointerGetDatum(osastate)); PointerGetDatum(osastate));
...@@ -247,9 +313,11 @@ ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples) ...@@ -247,9 +313,11 @@ ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples)
/* /*
* Clean up when evaluation of an ordered-set aggregate is complete. * Clean up when evaluation of an ordered-set aggregate is complete.
* *
* We don't need to bother freeing objects in the aggcontext memory context, * We don't need to bother freeing objects in the per-group memory context,
* since that will get reset anyway by nodeAgg.c, but we should take care to * since that will get reset anyway by nodeAgg.c; nor should we free anything
* release any potential non-memory resources. * in the per-query context, which will get cleared (if this was the last
* group) by ExecutorEnd. But we must take care to release any potential
* non-memory resources.
* *
* This callback is arguably unnecessary, since we don't support use of * This callback is arguably unnecessary, since we don't support use of
* ordered-set aggs in AGG_HASHED mode and there is currently no non-error * ordered-set aggs in AGG_HASHED mode and there is currently no non-error
...@@ -263,16 +331,15 @@ ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples) ...@@ -263,16 +331,15 @@ ordered_set_startup(FunctionCallInfo fcinfo, bool use_tuples)
static void static void
ordered_set_shutdown(Datum arg) ordered_set_shutdown(Datum arg)
{ {
OrderedSetAggState *osastate = (OrderedSetAggState *) DatumGetPointer(arg); OSAPerGroupState *osastate = (OSAPerGroupState *) DatumGetPointer(arg);
/* Tuplesort object might have temp files. */ /* Tuplesort object might have temp files. */
if (osastate->sortstate) if (osastate->sortstate)
tuplesort_end(osastate->sortstate); tuplesort_end(osastate->sortstate);
osastate->sortstate = NULL; osastate->sortstate = NULL;
/* The tupleslot probably can't be holding a pin, but let's be safe. */ /* The tupleslot probably can't be holding a pin, but let's be safe. */
if (osastate->tupslot) if (osastate->qstate->tupslot)
ExecDropSingleTupleTableSlot(osastate->tupslot); ExecClearTuple(osastate->qstate->tupslot);
osastate->tupslot = NULL;
} }
...@@ -283,7 +350,7 @@ ordered_set_shutdown(Datum arg) ...@@ -283,7 +350,7 @@ ordered_set_shutdown(Datum arg)
Datum Datum
ordered_set_transition(PG_FUNCTION_ARGS) ordered_set_transition(PG_FUNCTION_ARGS)
{ {
OrderedSetAggState *osastate; OSAPerGroupState *osastate;
/* If first call, create the transition state workspace */ /* If first call, create the transition state workspace */
if (PG_ARGISNULL(0)) if (PG_ARGISNULL(0))
...@@ -293,7 +360,7 @@ ordered_set_transition(PG_FUNCTION_ARGS) ...@@ -293,7 +360,7 @@ ordered_set_transition(PG_FUNCTION_ARGS)
/* safety check */ /* safety check */
if (AggCheckCallContext(fcinfo, NULL) != AGG_CONTEXT_AGGREGATE) if (AggCheckCallContext(fcinfo, NULL) != AGG_CONTEXT_AGGREGATE)
elog(ERROR, "ordered-set aggregate called in non-aggregate context"); elog(ERROR, "ordered-set aggregate called in non-aggregate context");
osastate = (OrderedSetAggState *) PG_GETARG_POINTER(0); osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
} }
/* Load the datum into the tuplesort object, but only if it's not null */ /* Load the datum into the tuplesort object, but only if it's not null */
...@@ -313,7 +380,7 @@ ordered_set_transition(PG_FUNCTION_ARGS) ...@@ -313,7 +380,7 @@ ordered_set_transition(PG_FUNCTION_ARGS)
Datum Datum
ordered_set_transition_multi(PG_FUNCTION_ARGS) ordered_set_transition_multi(PG_FUNCTION_ARGS)
{ {
OrderedSetAggState *osastate; OSAPerGroupState *osastate;
TupleTableSlot *slot; TupleTableSlot *slot;
int nargs; int nargs;
int i; int i;
...@@ -326,11 +393,11 @@ ordered_set_transition_multi(PG_FUNCTION_ARGS) ...@@ -326,11 +393,11 @@ ordered_set_transition_multi(PG_FUNCTION_ARGS)
/* safety check */ /* safety check */
if (AggCheckCallContext(fcinfo, NULL) != AGG_CONTEXT_AGGREGATE) if (AggCheckCallContext(fcinfo, NULL) != AGG_CONTEXT_AGGREGATE)
elog(ERROR, "ordered-set aggregate called in non-aggregate context"); elog(ERROR, "ordered-set aggregate called in non-aggregate context");
osastate = (OrderedSetAggState *) PG_GETARG_POINTER(0); osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
} }
/* Form a tuple from all the other inputs besides the transition value */ /* Form a tuple from all the other inputs besides the transition value */
slot = osastate->tupslot; slot = osastate->qstate->tupslot;
ExecClearTuple(slot); ExecClearTuple(slot);
nargs = PG_NARGS() - 1; nargs = PG_NARGS() - 1;
for (i = 0; i < nargs; i++) for (i = 0; i < nargs; i++)
...@@ -338,7 +405,7 @@ ordered_set_transition_multi(PG_FUNCTION_ARGS) ...@@ -338,7 +405,7 @@ ordered_set_transition_multi(PG_FUNCTION_ARGS)
slot->tts_values[i] = PG_GETARG_DATUM(i + 1); slot->tts_values[i] = PG_GETARG_DATUM(i + 1);
slot->tts_isnull[i] = PG_ARGISNULL(i + 1); slot->tts_isnull[i] = PG_ARGISNULL(i + 1);
} }
if (osastate->aggref->aggkind == AGGKIND_HYPOTHETICAL) if (osastate->qstate->aggref->aggkind == AGGKIND_HYPOTHETICAL)
{ {
/* Add a zero flag value to mark this row as a normal input row */ /* Add a zero flag value to mark this row as a normal input row */
slot->tts_values[i] = Int32GetDatum(0); slot->tts_values[i] = Int32GetDatum(0);
...@@ -362,7 +429,7 @@ ordered_set_transition_multi(PG_FUNCTION_ARGS) ...@@ -362,7 +429,7 @@ ordered_set_transition_multi(PG_FUNCTION_ARGS)
Datum Datum
percentile_disc_final(PG_FUNCTION_ARGS) percentile_disc_final(PG_FUNCTION_ARGS)
{ {
OrderedSetAggState *osastate; OSAPerGroupState *osastate;
double percentile; double percentile;
Datum val; Datum val;
bool isnull; bool isnull;
...@@ -388,7 +455,7 @@ percentile_disc_final(PG_FUNCTION_ARGS) ...@@ -388,7 +455,7 @@ percentile_disc_final(PG_FUNCTION_ARGS)
if (PG_ARGISNULL(0)) if (PG_ARGISNULL(0))
PG_RETURN_NULL(); PG_RETURN_NULL();
osastate = (OrderedSetAggState *) PG_GETARG_POINTER(0); osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
/* number_of_rows could be zero if we only saw NULL input values */ /* number_of_rows could be zero if we only saw NULL input values */
if (osastate->number_of_rows == 0) if (osastate->number_of_rows == 0)
...@@ -465,7 +532,7 @@ percentile_cont_final_common(FunctionCallInfo fcinfo, ...@@ -465,7 +532,7 @@ percentile_cont_final_common(FunctionCallInfo fcinfo,
Oid expect_type, Oid expect_type,
LerpFunc lerpfunc) LerpFunc lerpfunc)
{ {
OrderedSetAggState *osastate; OSAPerGroupState *osastate;
double percentile; double percentile;
int64 first_row = 0; int64 first_row = 0;
int64 second_row = 0; int64 second_row = 0;
...@@ -495,13 +562,13 @@ percentile_cont_final_common(FunctionCallInfo fcinfo, ...@@ -495,13 +562,13 @@ percentile_cont_final_common(FunctionCallInfo fcinfo,
if (PG_ARGISNULL(0)) if (PG_ARGISNULL(0))
PG_RETURN_NULL(); PG_RETURN_NULL();
osastate = (OrderedSetAggState *) PG_GETARG_POINTER(0); osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
/* number_of_rows could be zero if we only saw NULL input values */ /* number_of_rows could be zero if we only saw NULL input values */
if (osastate->number_of_rows == 0) if (osastate->number_of_rows == 0)
PG_RETURN_NULL(); PG_RETURN_NULL();
Assert(expect_type == osastate->datumtype); Assert(expect_type == osastate->qstate->sortColType);
/* Finish the sort */ /* Finish the sort */
tuplesort_performsort(osastate->sortstate); tuplesort_performsort(osastate->sortstate);
...@@ -672,7 +739,7 @@ setup_pct_info(int num_percentiles, ...@@ -672,7 +739,7 @@ setup_pct_info(int num_percentiles,
Datum Datum
percentile_disc_multi_final(PG_FUNCTION_ARGS) percentile_disc_multi_final(PG_FUNCTION_ARGS)
{ {
OrderedSetAggState *osastate; OSAPerGroupState *osastate;
ArrayType *param; ArrayType *param;
Datum *percentiles_datum; Datum *percentiles_datum;
bool *percentiles_null; bool *percentiles_null;
...@@ -693,7 +760,7 @@ percentile_disc_multi_final(PG_FUNCTION_ARGS) ...@@ -693,7 +760,7 @@ percentile_disc_multi_final(PG_FUNCTION_ARGS)
if (PG_ARGISNULL(0)) if (PG_ARGISNULL(0))
PG_RETURN_NULL(); PG_RETURN_NULL();
osastate = (OrderedSetAggState *) PG_GETARG_POINTER(0); osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
/* number_of_rows could be zero if we only saw NULL input values */ /* number_of_rows could be zero if we only saw NULL input values */
if (osastate->number_of_rows == 0) if (osastate->number_of_rows == 0)
...@@ -712,7 +779,7 @@ percentile_disc_multi_final(PG_FUNCTION_ARGS) ...@@ -712,7 +779,7 @@ percentile_disc_multi_final(PG_FUNCTION_ARGS)
&num_percentiles); &num_percentiles);
if (num_percentiles == 0) if (num_percentiles == 0)
PG_RETURN_POINTER(construct_empty_array(osastate->datumtype)); PG_RETURN_POINTER(construct_empty_array(osastate->qstate->sortColType));
pct_info = setup_pct_info(num_percentiles, pct_info = setup_pct_info(num_percentiles,
percentiles_datum, percentiles_datum,
...@@ -779,10 +846,10 @@ percentile_disc_multi_final(PG_FUNCTION_ARGS) ...@@ -779,10 +846,10 @@ percentile_disc_multi_final(PG_FUNCTION_ARGS)
ARR_NDIM(param), ARR_NDIM(param),
ARR_DIMS(param), ARR_DIMS(param),
ARR_LBOUND(param), ARR_LBOUND(param),
osastate->datumtype, osastate->qstate->sortColType,
osastate->typLen, osastate->qstate->typLen,
osastate->typByVal, osastate->qstate->typByVal,
osastate->typAlign)); osastate->qstate->typAlign));
} }
/* /*
...@@ -794,7 +861,7 @@ percentile_cont_multi_final_common(FunctionCallInfo fcinfo, ...@@ -794,7 +861,7 @@ percentile_cont_multi_final_common(FunctionCallInfo fcinfo,
int16 typLen, bool typByVal, char typAlign, int16 typLen, bool typByVal, char typAlign,
LerpFunc lerpfunc) LerpFunc lerpfunc)
{ {
OrderedSetAggState *osastate; OSAPerGroupState *osastate;
ArrayType *param; ArrayType *param;
Datum *percentiles_datum; Datum *percentiles_datum;
bool *percentiles_null; bool *percentiles_null;
...@@ -816,13 +883,13 @@ percentile_cont_multi_final_common(FunctionCallInfo fcinfo, ...@@ -816,13 +883,13 @@ percentile_cont_multi_final_common(FunctionCallInfo fcinfo,
if (PG_ARGISNULL(0)) if (PG_ARGISNULL(0))
PG_RETURN_NULL(); PG_RETURN_NULL();
osastate = (OrderedSetAggState *) PG_GETARG_POINTER(0); osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
/* number_of_rows could be zero if we only saw NULL input values */ /* number_of_rows could be zero if we only saw NULL input values */
if (osastate->number_of_rows == 0) if (osastate->number_of_rows == 0)
PG_RETURN_NULL(); PG_RETURN_NULL();
Assert(expect_type == osastate->datumtype); Assert(expect_type == osastate->qstate->sortColType);
/* Deconstruct the percentile-array input */ /* Deconstruct the percentile-array input */
if (PG_ARGISNULL(1)) if (PG_ARGISNULL(1))
...@@ -837,7 +904,7 @@ percentile_cont_multi_final_common(FunctionCallInfo fcinfo, ...@@ -837,7 +904,7 @@ percentile_cont_multi_final_common(FunctionCallInfo fcinfo,
&num_percentiles); &num_percentiles);
if (num_percentiles == 0) if (num_percentiles == 0)
PG_RETURN_POINTER(construct_empty_array(osastate->datumtype)); PG_RETURN_POINTER(construct_empty_array(osastate->qstate->sortColType));
pct_info = setup_pct_info(num_percentiles, pct_info = setup_pct_info(num_percentiles,
percentiles_datum, percentiles_datum,
...@@ -967,7 +1034,7 @@ percentile_cont_interval_multi_final(PG_FUNCTION_ARGS) ...@@ -967,7 +1034,7 @@ percentile_cont_interval_multi_final(PG_FUNCTION_ARGS)
Datum Datum
mode_final(PG_FUNCTION_ARGS) mode_final(PG_FUNCTION_ARGS)
{ {
OrderedSetAggState *osastate; OSAPerGroupState *osastate;
Datum val; Datum val;
bool isnull; bool isnull;
Datum mode_val = (Datum) 0; Datum mode_val = (Datum) 0;
...@@ -975,7 +1042,7 @@ mode_final(PG_FUNCTION_ARGS) ...@@ -975,7 +1042,7 @@ mode_final(PG_FUNCTION_ARGS)
Datum last_val = (Datum) 0; Datum last_val = (Datum) 0;
int64 last_val_freq = 0; int64 last_val_freq = 0;
bool last_val_is_mode = false; bool last_val_is_mode = false;
FmgrInfo equalfn; FmgrInfo *equalfn;
bool shouldfree; bool shouldfree;
/* safety check */ /* safety check */
...@@ -986,16 +1053,19 @@ mode_final(PG_FUNCTION_ARGS) ...@@ -986,16 +1053,19 @@ mode_final(PG_FUNCTION_ARGS)
if (PG_ARGISNULL(0)) if (PG_ARGISNULL(0))
PG_RETURN_NULL(); PG_RETURN_NULL();
osastate = (OrderedSetAggState *) PG_GETARG_POINTER(0); osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
/* number_of_rows could be zero if we only saw NULL input values */ /* number_of_rows could be zero if we only saw NULL input values */
if (osastate->number_of_rows == 0) if (osastate->number_of_rows == 0)
PG_RETURN_NULL(); PG_RETURN_NULL();
/* Look up the equality function for the datatype */ /* Look up the equality function for the datatype, if we didn't already */
fmgr_info(get_opcode(osastate->eqOperator), &equalfn); equalfn = &(osastate->qstate->equalfn);
if (!OidIsValid(equalfn->fn_oid))
fmgr_info_cxt(get_opcode(osastate->qstate->eqOperator), equalfn,
osastate->qstate->qcontext);
shouldfree = !(osastate->typByVal); shouldfree = !(osastate->qstate->typByVal);
/* Finish the sort */ /* Finish the sort */
tuplesort_performsort(osastate->sortstate); tuplesort_performsort(osastate->sortstate);
...@@ -1014,7 +1084,7 @@ mode_final(PG_FUNCTION_ARGS) ...@@ -1014,7 +1084,7 @@ mode_final(PG_FUNCTION_ARGS)
mode_freq = last_val_freq = 1; mode_freq = last_val_freq = 1;
last_val_is_mode = true; last_val_is_mode = true;
} }
else if (DatumGetBool(FunctionCall2(&equalfn, val, last_val))) else if (DatumGetBool(FunctionCall2(equalfn, val, last_val)))
{ {
/* value equal to previous value, count it */ /* value equal to previous value, count it */
if (last_val_is_mode) if (last_val_is_mode)
...@@ -1099,7 +1169,7 @@ hypothetical_rank_common(FunctionCallInfo fcinfo, int flag, ...@@ -1099,7 +1169,7 @@ hypothetical_rank_common(FunctionCallInfo fcinfo, int flag,
{ {
int nargs = PG_NARGS() - 1; int nargs = PG_NARGS() - 1;
int64 rank = 1; int64 rank = 1;
OrderedSetAggState *osastate; OSAPerGroupState *osastate;
TupleTableSlot *slot; TupleTableSlot *slot;
int i; int i;
...@@ -1114,7 +1184,7 @@ hypothetical_rank_common(FunctionCallInfo fcinfo, int flag, ...@@ -1114,7 +1184,7 @@ hypothetical_rank_common(FunctionCallInfo fcinfo, int flag,
return 1; return 1;
} }
osastate = (OrderedSetAggState *) PG_GETARG_POINTER(0); osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
*number_of_rows = osastate->number_of_rows; *number_of_rows = osastate->number_of_rows;
/* Adjust nargs to be the number of direct (or aggregated) args */ /* Adjust nargs to be the number of direct (or aggregated) args */
...@@ -1122,10 +1192,10 @@ hypothetical_rank_common(FunctionCallInfo fcinfo, int flag, ...@@ -1122,10 +1192,10 @@ hypothetical_rank_common(FunctionCallInfo fcinfo, int flag,
elog(ERROR, "wrong number of arguments in hypothetical-set function"); elog(ERROR, "wrong number of arguments in hypothetical-set function");
nargs /= 2; nargs /= 2;
hypothetical_check_argtypes(fcinfo, nargs, osastate->tupdesc); hypothetical_check_argtypes(fcinfo, nargs, osastate->qstate->tupdesc);
/* insert the hypothetical row into the sort */ /* insert the hypothetical row into the sort */
slot = osastate->tupslot; slot = osastate->qstate->tupslot;
ExecClearTuple(slot); ExecClearTuple(slot);
for (i = 0; i < nargs; i++) for (i = 0; i < nargs; i++)
{ {
...@@ -1225,8 +1295,7 @@ hypothetical_dense_rank_final(PG_FUNCTION_ARGS) ...@@ -1225,8 +1295,7 @@ hypothetical_dense_rank_final(PG_FUNCTION_ARGS)
int nargs = PG_NARGS() - 1; int nargs = PG_NARGS() - 1;
int64 rank = 1; int64 rank = 1;
int64 duplicate_count = 0; int64 duplicate_count = 0;
OrderedSetAggState *osastate; OSAPerGroupState *osastate;
List *sortlist;
int numDistinctCols; int numDistinctCols;
AttrNumber *sortColIdx; AttrNumber *sortColIdx;
FmgrInfo *equalfns; FmgrInfo *equalfns;
...@@ -1234,7 +1303,6 @@ hypothetical_dense_rank_final(PG_FUNCTION_ARGS) ...@@ -1234,7 +1303,6 @@ hypothetical_dense_rank_final(PG_FUNCTION_ARGS)
TupleTableSlot *extraslot; TupleTableSlot *extraslot;
TupleTableSlot *slot2; TupleTableSlot *slot2;
MemoryContext tmpcontext; MemoryContext tmpcontext;
ListCell *lc;
int i; int i;
/* safety check */ /* safety check */
...@@ -1245,41 +1313,44 @@ hypothetical_dense_rank_final(PG_FUNCTION_ARGS) ...@@ -1245,41 +1313,44 @@ hypothetical_dense_rank_final(PG_FUNCTION_ARGS)
if (PG_ARGISNULL(0)) if (PG_ARGISNULL(0))
PG_RETURN_INT64(rank); PG_RETURN_INT64(rank);
osastate = (OrderedSetAggState *) PG_GETARG_POINTER(0); osastate = (OSAPerGroupState *) PG_GETARG_POINTER(0);
/* Adjust nargs to be the number of direct (or aggregated) args */ /* Adjust nargs to be the number of direct (or aggregated) args */
if (nargs % 2 != 0) if (nargs % 2 != 0)
elog(ERROR, "wrong number of arguments in hypothetical-set function"); elog(ERROR, "wrong number of arguments in hypothetical-set function");
nargs /= 2; nargs /= 2;
hypothetical_check_argtypes(fcinfo, nargs, osastate->tupdesc); hypothetical_check_argtypes(fcinfo, nargs, osastate->qstate->tupdesc);
/* /*
* Construct list of columns to compare for uniqueness. We can omit the * When comparing tuples, we can omit the flag column since we will only
* flag column since we will only compare rows with flag == 0. * compare rows with flag == 0.
*/ */
sortlist = osastate->aggref->aggorder; numDistinctCols = osastate->qstate->numSortCols - 1;
numDistinctCols = list_length(sortlist);
sortColIdx = (AttrNumber *) palloc(numDistinctCols * sizeof(AttrNumber));
equalfns = (FmgrInfo *) palloc(numDistinctCols * sizeof(FmgrInfo));
i = 0; /* Look up the equality function(s), if we didn't already */
foreach(lc, sortlist) equalfns = osastate->qstate->equalfns;
if (equalfns == NULL)
{ {
SortGroupClause *sortcl = (SortGroupClause *) lfirst(lc); MemoryContext qcontext = osastate->qstate->qcontext;
TargetEntry *tle = get_sortgroupclause_tle(sortcl,
osastate->aggref->args);
sortColIdx[i] = tle->resno; equalfns = (FmgrInfo *)
fmgr_info(get_opcode(sortcl->eqop), &equalfns[i]); MemoryContextAlloc(qcontext, numDistinctCols * sizeof(FmgrInfo));
i++; for (i = 0; i < numDistinctCols; i++)
{
fmgr_info_cxt(get_opcode(osastate->qstate->eqOperators[i]),
&equalfns[i],
qcontext);
}
osastate->qstate->equalfns = equalfns;
} }
sortColIdx = osastate->qstate->sortColIdx;
/* Get short-term context we can use for execTuplesMatch */ /* Get short-term context we can use for execTuplesMatch */
tmpcontext = AggGetPerTupleEContext(fcinfo)->ecxt_per_tuple_memory; tmpcontext = AggGetPerTupleEContext(fcinfo)->ecxt_per_tuple_memory;
/* insert the hypothetical row into the sort */ /* insert the hypothetical row into the sort */
slot = osastate->tupslot; slot = osastate->qstate->tupslot;
ExecClearTuple(slot); ExecClearTuple(slot);
for (i = 0; i < nargs; i++) for (i = 0; i < nargs; i++)
{ {
...@@ -1296,11 +1367,11 @@ hypothetical_dense_rank_final(PG_FUNCTION_ARGS) ...@@ -1296,11 +1367,11 @@ hypothetical_dense_rank_final(PG_FUNCTION_ARGS)
tuplesort_performsort(osastate->sortstate); tuplesort_performsort(osastate->sortstate);
/* /*
* We alternate fetching into osastate->tupslot and extraslot so that we * We alternate fetching into tupslot and extraslot so that we have the
* have the previous row available for comparisons. This is accomplished * previous row available for comparisons. This is accomplished by
* by swapping the slot pointer variables after each row. * swapping the slot pointer variables after each row.
*/ */
extraslot = MakeSingleTupleTableSlot(osastate->tupdesc); extraslot = MakeSingleTupleTableSlot(osastate->qstate->tupdesc);
slot2 = extraslot; slot2 = extraslot;
/* iterate till we find the hypothetical row */ /* iterate till we find the hypothetical row */
......
...@@ -128,6 +128,7 @@ DATA(insert OID = 95 ( "<" PGNSP PGUID b f f 21 21 16 520 524 int2lt scalar ...@@ -128,6 +128,7 @@ DATA(insert OID = 95 ( "<" PGNSP PGUID b f f 21 21 16 520 524 int2lt scalar
DESCR("less than"); DESCR("less than");
DATA(insert OID = 96 ( "=" PGNSP PGUID b t t 23 23 16 96 518 int4eq eqsel eqjoinsel )); DATA(insert OID = 96 ( "=" PGNSP PGUID b t t 23 23 16 96 518 int4eq eqsel eqjoinsel ));
DESCR("equal"); DESCR("equal");
#define Int4EqualOperator 96
DATA(insert OID = 97 ( "<" PGNSP PGUID b f f 23 23 16 521 525 int4lt scalarltsel scalarltjoinsel )); DATA(insert OID = 97 ( "<" PGNSP PGUID b f f 23 23 16 521 525 int4lt scalarltsel scalarltjoinsel ));
DESCR("less than"); DESCR("less than");
#define Int4LessOperator 97 #define Int4LessOperator 97
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment