Commit c7b35395 authored by Tom Lane's avatar Tom Lane

Fix non-equivalence of VARIADIC and non-VARIADIC function call formats.

For variadic functions (other than VARIADIC ANY), the syntaxes foo(x,y,...)
and foo(VARIADIC ARRAY[x,y,...]) should be considered equivalent, since the
former is converted to the latter at parse time.  They have indeed been
equivalent, in all releases before 9.3.  However, commit 75b39e79 made an
ill-considered decision to record which syntax had been used in FuncExpr
nodes, and then to make equal() test that in checking node equality ---
which caused the syntaxes to not be seen as equivalent by the planner.
This is the underlying cause of bug #9817 from Dmitry Ryabov.

It might seem that a quick fix would be to make equal() disregard
FuncExpr.funcvariadic, but the same commit made that untenable, because
the field actually *is* semantically significant for some VARIADIC ANY
functions.  This patch instead adopts the approach of redefining
funcvariadic (and aggvariadic, in HEAD) as meaning that the last argument
is a variadic array, whether it got that way by parser intervention or was
supplied explicitly by the user.  Therefore the value will always be true
for non-ANY variadic functions, restoring the principle of equivalence.
(However, the planner will continue to consider use of VARIADIC as a
meaningful difference for VARIADIC ANY functions, even though some such
functions might disregard it.)

In HEAD, this change lets us simplify the decompilation logic in
ruleutils.c, since the funcvariadic/aggvariadic flag tells directly whether
to print VARIADIC.  However, in 9.3 we have to continue to cope with
existing stored rules/views that might contain the previous definition.
Fortunately, this just means no change in ruleutils.c, since its existing
behavior effectively ignores funcvariadic for all cases other than VARIADIC
ANY functions.

In HEAD, bump catversion to reflect the fact that FuncExpr.funcvariadic
changed meanings; this is sort of pro forma, since I don't believe any
built-in views are affected.

Unfortunately, this patch doesn't magically fix everything for affected
9.3 users.  After installing 9.3.5, they might need to recreate their
rules/views/indexes containing variadic function calls in order to get
everything consistent with the new definition.  As in the cited bug,
the symptom of a problem would be failure to use a nominally matching
index that has a variadic function call in its definition.  We'll need
to mention this in the 9.3.5 release notes.
parent 741364bf
...@@ -1548,15 +1548,7 @@ deparseFuncExpr(FuncExpr *node, deparse_expr_cxt *context) ...@@ -1548,15 +1548,7 @@ deparseFuncExpr(FuncExpr *node, deparse_expr_cxt *context)
procform = (Form_pg_proc) GETSTRUCT(proctup); procform = (Form_pg_proc) GETSTRUCT(proctup);
/* Check if need to print VARIADIC (cf. ruleutils.c) */ /* Check if need to print VARIADIC (cf. ruleutils.c) */
if (OidIsValid(procform->provariadic)) use_variadic = node->funcvariadic;
{
if (procform->provariadic != ANYOID)
use_variadic = true;
else
use_variadic = node->funcvariadic;
}
else
use_variadic = false;
/* Print schema name only if it's not pg_catalog */ /* Print schema name only if it's not pg_catalog */
if (procform->pronamespace != PG_CATALOG_NAMESPACE) if (procform->pronamespace != PG_CATALOG_NAMESPACE)
......
...@@ -3152,9 +3152,10 @@ CREATE OR REPLACE FUNCTION retcomposite(IN integer, IN integer, ...@@ -3152,9 +3152,10 @@ CREATE OR REPLACE FUNCTION retcomposite(IN integer, IN integer,
is zero based. <function>get_call_result_type</> can also be used is zero based. <function>get_call_result_type</> can also be used
as an alternative to <function>get_fn_expr_rettype</>. as an alternative to <function>get_fn_expr_rettype</>.
There is also <function>get_fn_expr_variadic</>, which can be used to There is also <function>get_fn_expr_variadic</>, which can be used to
find out whether the call contained an explicit <literal>VARIADIC</> find out whether variadic arguments have been merged into an array.
keyword. This is primarily useful for <literal>VARIADIC "any"</> This is primarily useful for <literal>VARIADIC "any"</> functions,
functions, as described below. since such merging will always have occurred for variadic functions
taking ordinary array types.
</para> </para>
<para> <para>
......
...@@ -555,6 +555,17 @@ ParseFuncOrColumn(ParseState *pstate, List *funcname, List *fargs, ...@@ -555,6 +555,17 @@ ParseFuncOrColumn(ParseState *pstate, List *funcname, List *fargs,
/* perform the necessary typecasting of arguments */ /* perform the necessary typecasting of arguments */
make_fn_arguments(pstate, fargs, actual_arg_types, declared_arg_types); make_fn_arguments(pstate, fargs, actual_arg_types, declared_arg_types);
/*
* If the function isn't actually variadic, forget any VARIADIC decoration
* on the call. (Perhaps we should throw an error instead, but
* historically we've allowed people to write that.)
*/
if (!OidIsValid(vatype))
{
Assert(nvargs == 0);
func_variadic = false;
}
/* /*
* If it's a variadic function call, transform the last nvargs arguments * If it's a variadic function call, transform the last nvargs arguments
* into an array --- unless it's an "any" variadic. * into an array --- unless it's an "any" variadic.
...@@ -584,6 +595,11 @@ ParseFuncOrColumn(ParseState *pstate, List *funcname, List *fargs, ...@@ -584,6 +595,11 @@ ParseFuncOrColumn(ParseState *pstate, List *funcname, List *fargs,
newa->location = exprLocation((Node *) vargs); newa->location = exprLocation((Node *) vargs);
fargs = lappend(fargs, newa); fargs = lappend(fargs, newa);
/* We could not have had VARIADIC marking before ... */
Assert(!func_variadic);
/* ... but now, it's a VARIADIC call */
func_variadic = true;
} }
/* /*
......
...@@ -401,7 +401,7 @@ static char *get_relation_name(Oid relid); ...@@ -401,7 +401,7 @@ static char *get_relation_name(Oid relid);
static char *generate_relation_name(Oid relid, List *namespaces); static char *generate_relation_name(Oid relid, List *namespaces);
static char *generate_function_name(Oid funcid, int nargs, static char *generate_function_name(Oid funcid, int nargs,
List *argnames, Oid *argtypes, List *argnames, Oid *argtypes,
bool was_variadic, bool *use_variadic_p); bool has_variadic, bool *use_variadic_p);
static char *generate_operator_name(Oid operid, Oid arg1, Oid arg2); static char *generate_operator_name(Oid operid, Oid arg1, Oid arg2);
static text *string_to_text(char *str); static text *string_to_text(char *str);
static char *flatten_reloptions(Oid relid); static char *flatten_reloptions(Oid relid);
...@@ -8849,16 +8849,16 @@ generate_relation_name(Oid relid, List *namespaces) ...@@ -8849,16 +8849,16 @@ generate_relation_name(Oid relid, List *namespaces)
* *
* If we're dealing with a potentially variadic function (in practice, this * If we're dealing with a potentially variadic function (in practice, this
* means a FuncExpr or Aggref, not some other way of calling a function), then * means a FuncExpr or Aggref, not some other way of calling a function), then
* was_variadic must specify whether VARIADIC appeared in the original call, * has_variadic must specify whether variadic arguments have been merged,
* and *use_variadic_p will be set to indicate whether to print VARIADIC in * and *use_variadic_p will be set to indicate whether to print VARIADIC in
* the output. For non-FuncExpr cases, was_variadic should be FALSE and * the output. For non-FuncExpr cases, has_variadic should be FALSE and
* use_variadic_p can be NULL. * use_variadic_p can be NULL.
* *
* The result includes all necessary quoting and schema-prefixing. * The result includes all necessary quoting and schema-prefixing.
*/ */
static char * static char *
generate_function_name(Oid funcid, int nargs, List *argnames, Oid *argtypes, generate_function_name(Oid funcid, int nargs, List *argnames, Oid *argtypes,
bool was_variadic, bool *use_variadic_p) bool has_variadic, bool *use_variadic_p)
{ {
char *result; char *result;
HeapTuple proctup; HeapTuple proctup;
...@@ -8884,32 +8884,27 @@ generate_function_name(Oid funcid, int nargs, List *argnames, Oid *argtypes, ...@@ -8884,32 +8884,27 @@ generate_function_name(Oid funcid, int nargs, List *argnames, Oid *argtypes,
* Determine whether VARIADIC should be printed. We must do this first * Determine whether VARIADIC should be printed. We must do this first
* since it affects the lookup rules in func_get_detail(). * since it affects the lookup rules in func_get_detail().
* *
* Currently, we always print VARIADIC if the function is variadic and * Currently, we always print VARIADIC if the function has a merged
* takes a variadic type other than ANY. (In principle, if VARIADIC * variadic-array argument. Note that this is always the case for
* wasn't originally specified and the array actual argument is * functions taking a VARIADIC argument type other than VARIADIC ANY.
* deconstructable, we could print the array elements separately and not *
* print VARIADIC, thus more nearly reproducing the original input. For * In principle, if VARIADIC wasn't originally specified and the array
* the moment that seems like too much complication for the benefit.) * actual argument is deconstructable, we could print the array elements
* However, if the function takes VARIADIC ANY, then the parser didn't * separately and not print VARIADIC, thus more nearly reproducing the
* fold the arguments together into an array, so we must print VARIADIC if * original input. For the moment that seems like too much complication
* and only if it was used originally. * for the benefit, and anyway we do not know whether VARIADIC was
* originally specified if it's a non-ANY type.
*/ */
if (use_variadic_p) if (use_variadic_p)
{ {
if (OidIsValid(procform->provariadic)) /* Parser should not have set funcvariadic unless fn is variadic */
{ Assert(!has_variadic || OidIsValid(procform->provariadic));
if (procform->provariadic != ANYOID) use_variadic = has_variadic;
use_variadic = true;
else
use_variadic = was_variadic;
}
else
use_variadic = false;
*use_variadic_p = use_variadic; *use_variadic_p = use_variadic;
} }
else else
{ {
Assert(!was_variadic); Assert(!has_variadic);
use_variadic = false; use_variadic = false;
} }
......
...@@ -2449,6 +2449,8 @@ get_call_expr_arg_stable(Node *expr, int argnum) ...@@ -2449,6 +2449,8 @@ get_call_expr_arg_stable(Node *expr, int argnum)
* Get the VARIADIC flag from the function invocation * Get the VARIADIC flag from the function invocation
* *
* Returns false (the default assumption) if information is not available * Returns false (the default assumption) if information is not available
*
* Note this is generally only of interest to VARIADIC ANY functions
*/ */
bool bool
get_fn_expr_variadic(FmgrInfo *flinfo) get_fn_expr_variadic(FmgrInfo *flinfo)
......
...@@ -53,6 +53,6 @@ ...@@ -53,6 +53,6 @@
*/ */
/* yyyymmddN */ /* yyyymmddN */
#define CATALOG_VERSION_NO 201403261 #define CATALOG_VERSION_NO 201404031
#endif #endif
...@@ -257,7 +257,8 @@ typedef struct Aggref ...@@ -257,7 +257,8 @@ typedef struct Aggref
List *aggdistinct; /* DISTINCT (list of SortGroupClause) */ List *aggdistinct; /* DISTINCT (list of SortGroupClause) */
Expr *aggfilter; /* FILTER expression, if any */ Expr *aggfilter; /* FILTER expression, if any */
bool aggstar; /* TRUE if argument list was really '*' */ bool aggstar; /* TRUE if argument list was really '*' */
bool aggvariadic; /* TRUE if VARIADIC was used in call */ bool aggvariadic; /* true if variadic arguments have been
* combined into an array last argument */
char aggkind; /* aggregate kind (see pg_aggregate.h) */ char aggkind; /* aggregate kind (see pg_aggregate.h) */
Index agglevelsup; /* > 0 if agg belongs to outer query */ Index agglevelsup; /* > 0 if agg belongs to outer query */
int location; /* token location, or -1 if unknown */ int location; /* token location, or -1 if unknown */
...@@ -358,7 +359,8 @@ typedef struct FuncExpr ...@@ -358,7 +359,8 @@ typedef struct FuncExpr
Oid funcid; /* PG_PROC OID of the function */ Oid funcid; /* PG_PROC OID of the function */
Oid funcresulttype; /* PG_TYPE OID of result value */ Oid funcresulttype; /* PG_TYPE OID of result value */
bool funcretset; /* true if function returns set */ bool funcretset; /* true if function returns set */
bool funcvariadic; /* true if VARIADIC was used in call */ bool funcvariadic; /* true if variadic arguments have been
* combined into an array last argument */
CoercionForm funcformat; /* how to display this function call */ CoercionForm funcformat; /* how to display this function call */
Oid funccollid; /* OID of collation of result */ Oid funccollid; /* OID of collation of result */
Oid inputcollid; /* OID of collation that function should use */ Oid inputcollid; /* OID of collation that function should use */
......
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