Commit dd3bab5f authored by Tom Lane's avatar Tom Lane

Ensure that whole-row junk Vars are always of composite type.

The EvalPlanQual machinery assumes that whole-row Vars generated for the
outputs of non-table RTEs will be of composite types.  However, for the
case where the RTE is a function call returning a scalar type, we were
doing the wrong thing, as a result of sharing code with a parser case
where the function's scalar output is wanted.  (Or at least, that's what
that case has done historically; it does seem a bit inconsistent.)

To fix, extend makeWholeRowVar's API so that it can support both use-cases.
This fixes Belinda Cussen's report of crashes during concurrent execution
of UPDATEs involving joins to the result of UNNEST() --- in READ COMMITTED
mode, we'd run the EvalPlanQual machinery after a conflicting row update
commits, and it was expecting to get a HeapTuple not a scalar datum from
the "wholerowN" variable referencing the function RTE.

Back-patch to 9.0 where the current EvalPlanQual implementation appeared.

In 9.1 and up, this patch also fixes failure to attach the correct
collation to the Var generated for a scalar-result case.  An example:
regression=# select upper(x.*) from textcat('ab', 'cd') x;
ERROR:  could not determine which collation to use for upper() function
parent 91572ee0
...@@ -121,11 +121,17 @@ makeVarFromTargetEntry(Index varno, ...@@ -121,11 +121,17 @@ makeVarFromTargetEntry(Index varno,
* with error cases, but it's not worth changing now.) The vartype indicates * with error cases, but it's not worth changing now.) The vartype indicates
* a rowtype; either a named composite type, or RECORD. This function * a rowtype; either a named composite type, or RECORD. This function
* encapsulates the logic for determining the correct rowtype OID to use. * encapsulates the logic for determining the correct rowtype OID to use.
*
* If allowScalar is true, then for the case where the RTE is a function
* returning a non-composite result type, we produce a normal Var referencing
* the function's result directly, instead of the single-column composite
* value that the whole-row notation might otherwise suggest.
*/ */
Var * Var *
makeWholeRowVar(RangeTblEntry *rte, makeWholeRowVar(RangeTblEntry *rte,
Index varno, Index varno,
Index varlevelsup) Index varlevelsup,
bool allowScalar)
{ {
Var *result; Var *result;
Oid toid; Oid toid;
...@@ -157,39 +163,34 @@ makeWholeRowVar(RangeTblEntry *rte, ...@@ -157,39 +163,34 @@ makeWholeRowVar(RangeTblEntry *rte,
InvalidOid, InvalidOid,
varlevelsup); varlevelsup);
} }
else else if (allowScalar)
{ {
/* /* func returns scalar; just return its output as-is */
* func returns scalar; instead of making a whole-row Var,
* just reference the function's scalar output. (XXX this
* seems a tad inconsistent, especially if "f.*" was
* explicitly written ...)
*/
result = makeVar(varno, result = makeVar(varno,
1, 1,
toid, toid,
-1, -1,
InvalidOid, exprCollation(rte->funcexpr),
varlevelsup); varlevelsup);
} }
break; else
case RTE_VALUES: {
toid = RECORDOID; /* func returns scalar, but we want a composite result */
/* returns composite; same as relation case */
result = makeVar(varno, result = makeVar(varno,
InvalidAttrNumber, InvalidAttrNumber,
toid, RECORDOID,
-1, -1,
InvalidOid, InvalidOid,
varlevelsup); varlevelsup);
}
break; break;
default: default:
/* /*
* RTE is a join or subselect. We represent this as a whole-row * RTE is a join, subselect, or VALUES. We represent this as a
* Var of RECORD type. (Note that in most cases the Var will be * whole-row Var of RECORD type. (Note that in most cases the Var
* expanded to a RowExpr during planning, but that is not our * will be expanded to a RowExpr during planning, but that is not
* concern here.) * our concern here.)
*/ */
result = makeVar(varno, result = makeVar(varno,
InvalidAttrNumber, InvalidAttrNumber,
......
...@@ -129,7 +129,8 @@ preprocess_targetlist(PlannerInfo *root, List *tlist) ...@@ -129,7 +129,8 @@ preprocess_targetlist(PlannerInfo *root, List *tlist)
/* Not a table, so we need the whole row as a junk var */ /* Not a table, so we need the whole row as a junk var */
var = makeWholeRowVar(rt_fetch(rc->rti, range_table), var = makeWholeRowVar(rt_fetch(rc->rti, range_table),
rc->rti, rc->rti,
0); 0,
false);
snprintf(resname, sizeof(resname), "wholerow%u", rc->rowmarkId); snprintf(resname, sizeof(resname), "wholerow%u", rc->rowmarkId);
tle = makeTargetEntry((Expr *) var, tle = makeTargetEntry((Expr *) var,
list_length(tlist) + 1, list_length(tlist) + 1,
......
...@@ -2059,8 +2059,15 @@ transformWholeRowRef(ParseState *pstate, RangeTblEntry *rte, int location) ...@@ -2059,8 +2059,15 @@ transformWholeRowRef(ParseState *pstate, RangeTblEntry *rte, int location)
/* Find the RTE's rangetable location */ /* Find the RTE's rangetable location */
vnum = RTERangeTablePosn(pstate, rte, &sublevels_up); vnum = RTERangeTablePosn(pstate, rte, &sublevels_up);
/* Build the appropriate referencing node */ /*
result = makeWholeRowVar(rte, vnum, sublevels_up); * Build the appropriate referencing node. Note that if the RTE is a
* function returning scalar, we create just a plain reference to the
* function value, not a composite containing a single column. This is
* pretty inconsistent at first sight, but it's what we've done
* historically. One argument for it is that "rel" and "rel.*" mean the
* same thing for composite relations, so why not for scalar functions...
*/
result = makeWholeRowVar(rte, vnum, sublevels_up, true);
/* location is not filled in by makeWholeRowVar */ /* location is not filled in by makeWholeRowVar */
result->location = location; result->location = location;
......
...@@ -1188,7 +1188,8 @@ rewriteTargetListUD(Query *parsetree, RangeTblEntry *target_rte, ...@@ -1188,7 +1188,8 @@ rewriteTargetListUD(Query *parsetree, RangeTblEntry *target_rte,
*/ */
var = makeWholeRowVar(target_rte, var = makeWholeRowVar(target_rte,
parsetree->resultRelation, parsetree->resultRelation,
0); 0,
false);
attrname = "wholerow"; attrname = "wholerow";
} }
......
...@@ -35,7 +35,8 @@ extern Var *makeVarFromTargetEntry(Index varno, ...@@ -35,7 +35,8 @@ extern Var *makeVarFromTargetEntry(Index varno,
extern Var *makeWholeRowVar(RangeTblEntry *rte, extern Var *makeWholeRowVar(RangeTblEntry *rte,
Index varno, Index varno,
Index varlevelsup); Index varlevelsup,
bool allowScalar);
extern TargetEntry *makeTargetEntry(Expr *expr, extern TargetEntry *makeTargetEntry(Expr *expr,
AttrNumber resno, AttrNumber resno,
......
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