Commit 305cf1fd authored by Tom Lane's avatar Tom Lane

Fix AggGetAggref() so it won't lie to aggregate final functions.

If we merge the transition calculations for two different aggregates,
it's reasonable to assume that the transition function should not care
which of those Aggref structs it gets from AggGetAggref().  It is not
reasonable to make the same assumption about an aggregate final function,
however.  Commit 804163bc broke this, as it will pass whichever Aggref
was first associated with the transition state in both cases.

This doesn't create an observable bug so far as the core system is
concerned, because the only existing uses of AggGetAggref() are in
ordered-set aggregates that happen to not pay attention to anything
but the input properties of the Aggref; and besides that, we disabled
sharing of transition calculations for OSAs yesterday.  Nonetheless,
if some third-party code were using AggGetAggref() in a normal aggregate,
they would be entitled to call this a bug.  Hence, back-patch the fix
to 9.6 where the problem was introduced.

In passing, improve some of the comments about transition state sharing.

Discussion: https://postgr.es/m/CAB4ELO5RZhOamuT9Xsf72ozbenDLLXZKSk07FiSVsuJNZB861A@mail.gmail.com
parent ad4a7ed0
...@@ -1528,8 +1528,8 @@ finalize_aggregate(AggState *aggstate, ...@@ -1528,8 +1528,8 @@ finalize_aggregate(AggState *aggstate,
{ {
int numFinalArgs = peragg->numFinalArgs; int numFinalArgs = peragg->numFinalArgs;
/* set up aggstate->curpertrans for AggGetAggref() */ /* set up aggstate->curperagg for AggGetAggref() */
aggstate->curpertrans = pertrans; aggstate->curperagg = peragg;
InitFunctionCallInfoData(fcinfo, &peragg->finalfn, InitFunctionCallInfoData(fcinfo, &peragg->finalfn,
numFinalArgs, numFinalArgs,
...@@ -1562,7 +1562,7 @@ finalize_aggregate(AggState *aggstate, ...@@ -1562,7 +1562,7 @@ finalize_aggregate(AggState *aggstate,
*resultVal = FunctionCallInvoke(&fcinfo); *resultVal = FunctionCallInvoke(&fcinfo);
*resultIsNull = fcinfo.isnull; *resultIsNull = fcinfo.isnull;
} }
aggstate->curpertrans = NULL; aggstate->curperagg = NULL;
} }
else else
{ {
...@@ -2712,6 +2712,7 @@ ExecInitAgg(Agg *node, EState *estate, int eflags) ...@@ -2712,6 +2712,7 @@ ExecInitAgg(Agg *node, EState *estate, int eflags)
aggstate->current_set = 0; aggstate->current_set = 0;
aggstate->peragg = NULL; aggstate->peragg = NULL;
aggstate->pertrans = NULL; aggstate->pertrans = NULL;
aggstate->curperagg = NULL;
aggstate->curpertrans = NULL; aggstate->curpertrans = NULL;
aggstate->input_done = false; aggstate->input_done = false;
aggstate->agg_done = false; aggstate->agg_done = false;
...@@ -3060,27 +3061,29 @@ ExecInitAgg(Agg *node, EState *estate, int eflags) ...@@ -3060,27 +3061,29 @@ ExecInitAgg(Agg *node, EState *estate, int eflags)
* *
* Scenarios: * Scenarios:
* *
* 1. An aggregate function appears more than once in query: * 1. Identical aggregate function calls appear in the query:
* *
* SELECT SUM(x) FROM ... HAVING SUM(x) > 0 * SELECT SUM(x) FROM ... HAVING SUM(x) > 0
* *
* Since the aggregates are the identical, we only need to calculate * Since these aggregates are identical, we only need to calculate
* the calculate it once. Both aggregates will share the same 'aggno' * the value once. Both aggregates will share the same 'aggno' value.
* value.
* *
* 2. Two different aggregate functions appear in the query, but the * 2. Two different aggregate functions appear in the query, but the
* aggregates have the same transition function and initial value, but * aggregates have the same arguments, transition functions and
* different final function: * initial values (and, presumably, different final functions):
* *
* SELECT SUM(x), AVG(x) FROM ... * SELECT AVG(x), STDDEV(x) FROM ...
* *
* In this case we must create a new peragg for the varying aggregate, * In this case we must create a new peragg for the varying aggregate,
* and need to call the final functions separately, but can share the * and we need to call the final functions separately, but we need
* same transition state. * only run the transition function once. (This requires that the
* final functions be nondestructive of the transition state, but
* that's required anyway for other reasons.)
* *
* For either of these optimizations to be valid, the aggregate's * For either of these optimizations to be valid, all aggregate properties
* arguments must be the same, including any modifiers such as ORDER BY, * used in the transition phase must be the same, including any modifiers
* DISTINCT and FILTER, and they mustn't contain any volatile functions. * such as ORDER BY, DISTINCT and FILTER, and the arguments mustn't
* contain any volatile functions.
* ----------------- * -----------------
*/ */
aggno = -1; aggno = -1;
...@@ -3723,7 +3726,7 @@ GetAggInitVal(Datum textInitVal, Oid transtype) ...@@ -3723,7 +3726,7 @@ GetAggInitVal(Datum textInitVal, Oid transtype)
* *
* As a side-effect, this also collects a list of existing per-Trans structs * As a side-effect, this also collects a list of existing per-Trans structs
* with matching inputs. If no identical Aggref is found, the list is passed * with matching inputs. If no identical Aggref is found, the list is passed
* later to find_compatible_perstate, to see if we can at least reuse the * later to find_compatible_pertrans, to see if we can at least reuse the
* state value of another aggregate. * state value of another aggregate.
*/ */
static int static int
...@@ -3743,11 +3746,12 @@ find_compatible_peragg(Aggref *newagg, AggState *aggstate, ...@@ -3743,11 +3746,12 @@ find_compatible_peragg(Aggref *newagg, AggState *aggstate,
/* /*
* Search through the list of already seen aggregates. If we find an * Search through the list of already seen aggregates. If we find an
* existing aggregate with the same aggregate function and input * existing identical aggregate call, then we can re-use that one. While
* parameters as an existing one, then we can re-use that one. While
* searching, we'll also collect a list of Aggrefs with the same input * searching, we'll also collect a list of Aggrefs with the same input
* parameters. If no matching Aggref is found, the caller can potentially * parameters. If no matching Aggref is found, the caller can potentially
* still re-use the transition state of one of them. * still re-use the transition state of one of them. (At this stage we
* just compare the parsetrees; whether different aggregates share the
* same transition function will be checked later.)
*/ */
for (aggno = 0; aggno <= lastaggno; aggno++) for (aggno = 0; aggno <= lastaggno; aggno++)
{ {
...@@ -3796,7 +3800,7 @@ find_compatible_peragg(Aggref *newagg, AggState *aggstate, ...@@ -3796,7 +3800,7 @@ find_compatible_peragg(Aggref *newagg, AggState *aggstate,
* struct * struct
* *
* Searches the list of transnos for a per-Trans struct with the same * Searches the list of transnos for a per-Trans struct with the same
* transition state and initial condition. (The inputs have already been * transition function and initial condition. (The inputs have already been
* verified to match.) * verified to match.)
*/ */
static int static int
...@@ -3842,16 +3846,16 @@ find_compatible_pertrans(AggState *aggstate, Aggref *newagg, ...@@ -3842,16 +3846,16 @@ find_compatible_pertrans(AggState *aggstate, Aggref *newagg,
aggdeserialfn != pertrans->deserialfn_oid) aggdeserialfn != pertrans->deserialfn_oid)
continue; continue;
/* Check that the initial condition matches, too. */ /*
* Check that the initial condition matches, too.
*/
if (initValueIsNull && pertrans->initValueIsNull) if (initValueIsNull && pertrans->initValueIsNull)
return transno; return transno;
if (!initValueIsNull && !pertrans->initValueIsNull && if (!initValueIsNull && !pertrans->initValueIsNull &&
datumIsEqual(initValue, pertrans->initValue, datumIsEqual(initValue, pertrans->initValue,
pertrans->transtypeByVal, pertrans->transtypeLen)) pertrans->transtypeByVal, pertrans->transtypeLen))
{
return transno; return transno;
}
} }
return -1; return -1;
} }
...@@ -4070,6 +4074,13 @@ AggCheckCallContext(FunctionCallInfo fcinfo, MemoryContext *aggcontext) ...@@ -4070,6 +4074,13 @@ AggCheckCallContext(FunctionCallInfo fcinfo, MemoryContext *aggcontext)
* If the function is being called as an aggregate support function, * If the function is being called as an aggregate support function,
* return the Aggref node for the aggregate call. Otherwise, return NULL. * return the Aggref node for the aggregate call. Otherwise, return NULL.
* *
* Aggregates sharing the same inputs and transition functions can get
* merged into a single transition calculation. If the transition function
* calls AggGetAggref, it will get some one of the Aggrefs for which it is
* executing. It must therefore not pay attention to the Aggref fields that
* relate to the final function, as those are indeterminate. But if a final
* function calls AggGetAggref, it will get a precise result.
*
* Note that if an aggregate is being used as a window function, this will * Note that if an aggregate is being used as a window function, this will
* return NULL. We could provide a similar function to return the relevant * return NULL. We could provide a similar function to return the relevant
* WindowFunc node in such cases, but it's not needed yet. * WindowFunc node in such cases, but it's not needed yet.
...@@ -4079,8 +4090,16 @@ AggGetAggref(FunctionCallInfo fcinfo) ...@@ -4079,8 +4090,16 @@ AggGetAggref(FunctionCallInfo fcinfo)
{ {
if (fcinfo->context && IsA(fcinfo->context, AggState)) if (fcinfo->context && IsA(fcinfo->context, AggState))
{ {
AggStatePerAgg curperagg;
AggStatePerTrans curpertrans; AggStatePerTrans curpertrans;
/* check curperagg (valid when in a final function) */
curperagg = ((AggState *) fcinfo->context)->curperagg;
if (curperagg)
return curperagg->aggref;
/* check curpertrans (valid when in a transition function) */
curpertrans = ((AggState *) fcinfo->context)->curpertrans; curpertrans = ((AggState *) fcinfo->context)->curpertrans;
if (curpertrans) if (curpertrans)
......
...@@ -1808,7 +1808,8 @@ typedef struct AggState ...@@ -1808,7 +1808,8 @@ typedef struct AggState
ExprContext **aggcontexts; /* econtexts for long-lived data (per GS) */ ExprContext **aggcontexts; /* econtexts for long-lived data (per GS) */
ExprContext *tmpcontext; /* econtext for input expressions */ ExprContext *tmpcontext; /* econtext for input expressions */
ExprContext *curaggcontext; /* currently active aggcontext */ ExprContext *curaggcontext; /* currently active aggcontext */
AggStatePerTrans curpertrans; /* currently active trans state */ AggStatePerAgg curperagg; /* currently active aggregate, if any */
AggStatePerTrans curpertrans; /* currently active trans state, if any */
bool input_done; /* indicates end of input */ bool input_done; /* indicates end of input */
bool agg_done; /* indicates completion of Agg scan */ bool agg_done; /* indicates completion of Agg scan */
int projected_set; /* The last projected grouping set */ int projected_set; /* The last projected grouping set */
......
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